-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathscp.py
264 lines (214 loc) · 7.36 KB
/
scp.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
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
#!/usr/bin/env python
# -*- coding: utf-8 -*-
class ScpRecord:
"""Container for all SCP sections"""
def __init__(self):
self.crc = 0
self.len = 0
self.sections = []
def has_section(self, section_id):
"""Return True if section with section_id exists"""
return self.section(section_id) is not None
def section(self, section_id):
"""Return section with given id"""
for s in self.sections:
if s.h.id == section_id:
return s
return None
def number_of_leads(self):
"""Returns number of leads from section 3"""
s3 = self.section(3)
if s3:
return len(s3.leads)
return 0
class SectionPointer:
"""A Section pointer"""
def __init__(self):
# section id number
self.id = 0
# section length
self.len = 0
# byte position of section starting from zero
self.index = 0
def __str__(self):
return '{0}({1})'.format(self.id, self.len)
def section_has_data(self):
"""Return True if section is not empty"""
return self.len > 0
class Section():
"""Super class for each Section, which contains a header"""
def __init__(self, scpHeader):
self.h = scpHeader
class SectionHeader:
"""A section header for each section"""
def __init__(self):
self.crc = 0
# section id number
self.id = 0
self.len = 0
# section version number
self.versnr = 0
# protocol version number
self.protnr = 0
self.reserved = None
def __str__(self):
return 'crc:{0},id:{1},len:{2},ver:{3},prot:{4},resvd:{5}'\
.format(self.crc, self.id, self.len, self.versnr, self.protnr, self.reserved)
class Tag:
"""A tag"""
def __init__(self):
self.tag = 0
self.len = 0
self.data = None
class Section0(Section):
"""Section 0 wich contains pointers to other sections"""
def __init__(self, header):
super().__init__(header)
self.p = []
def has_section(self, section_id):
"""Retturn True if section with id exists"""
p = self.pointer_for_section(section_id)
if p is None:
return False
return p.len > 0
def pointer_for_section(self, section_id):
"""Returns pointer for the section with id"""
for p in self.p:
if p.id == section_id:
return p
return None
class LeadIdentification:
"""LeadIdenticitation with information about sample count (Section 3)"""
def __init__(self):
self.startsample = 0
self.endsample = 0
self.leadid = 0
def __str__(self):
return '{0} ({1})'.format(self.leadid, self.sample_count())
def sample_count(self):
"""Return number of samples for this LeadId"""
return self.endsample - self.startsample + 1
class DataSamples:
"""
Data samples for a lead, for Sections 5 (ref. beat samples) and Section 6 (samples)
"""
def __init__(self):
self.samples = []
class Section1(Section):
"""Section 1 - Patient data, ECG Aquisition data"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
# tags 10,13,30,32,35 may exist multiple times
self.tags = []
self.datalen = 0
class Section2(Section):
"""Section 2 - Huffman tables used in encoding of Ecg data"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
# Number of Huffman Tables defined (if 19 999 then the default table, defined in C.3.7.6, is used).
self.nr_huffman_tables = 0
# Number of code structures in table # 1
self.nr_code_struct = 0
class Section3(Section):
"""Section 3 - ECG Leads definition"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
self.nrleads = 0
self.flags = 0
# first bit
self.ref_beat_substr = False
# bits 3-7
self.nr_leads_sim = 0
self.leads = []
# QRS Locations if reference beats are encoded
class Section4(Section):
"""Section 4 - Reserved for legacy SCP-ECG versions (SCP Versions 1.x, 2.x)"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
# Length of reference beat type 0 in milliseconds
self.ref_beat_type_len = 0
# Sample number of the fiducial point (QRS trigger point), with respect to beginning of reference beat type 0
self.sample_nr_fidpoint = 0
# Total number of QRS complexes within the entire short-term ECG rhythm record
self.total_nr_qrs = 0
class Section5(Section):
"""Section 5 with samples"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
# avm in nanovolt
self.avm = 0
# sample time interval in ms
self.sample_time_interval = 0
# 0,1,2
self.sample_encoding = 0
self.reserved = 0
# number of bytes for encoded leads
# in the same order as the leads are stored
# lead order comes from section3
self.nr_bytes_for_leads = []
self.data = []
class Section6(Section):
"""Section 6 - Long-term ECG rythm data"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
# Amplitude Value Multiplier in nanovolt
self.avm = 0
self.sample_time_interval = 0
# 0,1,2
self.sample_encoding = 0
# rythm data compression
self.bimodal_compression = 0
# number of bytes for encoded leads
# in the same order as the leads are stored
# lead order comes from section3
self.nr_bytes_for_leads = []
self.data = []
class Section7(Section):
"""Section 7 - Global ECG Measurements"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
self.reference_count = 0
# number of pacemaker spikes
self.pace_count = 0
# Average RR interval in milliseconds for all QRS's
self.rr_interval = 0
# Average PP interval in milliseconds for all QRS's
self.pp_interval = 0
self.pace_times = []
self.pace_amplitudes = []
self.pace_types = []
self.pace_sources = []
self.pace_indexes = []
self.pace_widths = []
class Section8(Section):
"""Section 8 - Textual diagnosis"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
class Section9(Section):
"""Section 9 - Manufacturer Specific diagnostic and overreading data"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
class Section10(Section):
"""Section 10 - per Lead ECG Measurements"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
class Section11(Section):
"""Section 11 - Universal Statement Codes resulting from the interpretation"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer
class Section12(Section):
"""Section 12 - Long-Term ECG Rythm data"""
def __init__(self, header, pointer):
super().__init__(header)
self.p = pointer