-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathalign.py
96 lines (84 loc) · 3.68 KB
/
align.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
# Simple class for learning an alignment of strings, MED-style.
# Weights are learned by a Chinese Restaurant Process sampler
# that weights single alignments x:y in proportion to how many times
# such an alignment has been seen elsewhere out of all possible alignments.
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
# http://www.apache.org/licenses/LICENSE-2.0
# Usage:
# Align(wordpairs) <= wordpairs is an iterable of 2-tuples containing strings as iterables
# The resulting Align.alignedpairs is a list of aligned 2-tuples
# Relies on C-code in libalign.so built from align.c through ctypes.
# Author: Mans Hulden
# MH20151102
# Last modified 20200422
import itertools
from ctypes import *
libalign = cdll.LoadLibrary('./libalign.so')
libalign_add_int_pair = libalign.add_int_pair
libalign_clear_counts = libalign.clear_counts
libalign_initial_align = libalign.initial_align
libalign_crp_train = libalign.crp_train
libalign_crp_align = libalign.crp_align
libalign_med_align = libalign.med_align
libalign_getpairs_init = libalign.getpairs_init
libalign_getpairs_init.restype = c_void_p
libalign_getpairs_in = libalign.getpairs_in
libalign_getpairs_in.restype = POINTER(c_int)
libalign_getpairs_out = libalign.getpairs_out
libalign_getpairs_out.restype = POINTER(c_int)
libalign_getpairs_advance = libalign.getpairs_advance
libalign_getpairs_advance.restype = c_void_p
libalign_align_init = libalign.align_init
libalign_align_init.restype = None
class Aligner:
def __init__(self, wordpairs, align_symbol = ' ', iterations = 10, burnin = 1, lag = 1, mode = 'crp'):
s = set()
for wl, wr in wordpairs:
s |= {a for a in wl}
s |= {a for a in wr}
self.symboltoint = dict(zip(s, range(1,len(s)+1)))
self.inttosymbol = {v:k for k, v in self.symboltoint.items()}
self.inttosymbol[0] = align_symbol
## Map stringpairs to -1 terminated integer sequences ##
intpairs = []
for i, o in wordpairs:
intin = [self.symboltoint[x] for x in i] + [-1]
intout = [self.symboltoint[x] for x in o] + [-1]
#intin = map(lambda x: self.symboltoint[x], i) + [-1]
#intout = map(lambda x: self.symboltoint[x], o) + [-1]
intpairs.append((intin, intout))
libalign_align_init()
for i, o in intpairs:
icint = (c_int * len(i))(*i)
ocint = (c_int * len(o))(*o)
libalign_add_int_pair(icint, ocint)
# Run CRP align
if mode == 'crp':
libalign_clear_counts()
libalign_initial_align()
libalign_crp_train(c_int(iterations), c_int(burnin), c_int(lag))
libalign_crp_align()
else:
libalign_clear_counts()
libalign_initial_align()
libalign_med_align()
# Reconvert to output
self.alignedpairs = []
stringpairptr = libalign_getpairs_init()
while stringpairptr != None:
inints = libalign_getpairs_in(c_void_p(stringpairptr))
outints = libalign_getpairs_out(c_void_p(stringpairptr))
instr = []
outstr = []
for j in itertools.count():
if inints[j] == -1:
break
instr.append(self.inttosymbol[inints[j]])
for j in itertools.count():
if outints[j] == -1:
break
outstr.append(self.inttosymbol[outints[j]])
self.alignedpairs.append((instr, outstr))
stringpairptr = libalign_getpairs_advance(c_void_p(stringpairptr))