Source of file MgfReader.php
Size: 6,292 Bytes - Last Modified: 2019-05-10T12:24:09+01:00
src/Reader/MgfReader.php
1234567891011121314151617181920212223242526272829303132333435363738394041424344454647
Covered by 5 test(s):
48
Covered by 5 test(s):
495051525354555657
Covered by 4 test(s):
58596061626364656667
Covered by 2 test(s):
686970717273747576
Covered by 2 test(s):
77
Covered by 2 test(s):
78
Covered by 2 test(s):
79
Covered by 2 test(s):
80
Covered by 2 test(s):
818283848586878889
Covered by 4 test(s):
90
Covered by 1 test(s):
91
Covered by 1 test(s):
9293
Covered by 4 test(s):
94
Covered by 4 test(s):
95
Covered by 4 test(s):
96
Covered by 4 test(s):
979899100101102103104105
Covered by 2 test(s):
106
Covered by 2 test(s):
107108109
Covered by 2 test(s):
110111112113114115116117118119
Covered by 2 test(s):
120
Covered by 2 test(s):
121122123
Covered by 2 test(s):
124
Covered by 2 test(s):
125126
Covered by 2 test(s):
127128129130131132133134135136
Covered by 2 test(s):
137
Covered by 2 test(s):
138
Covered by 2 test(s):
139140
Covered by 2 test(s):
141142143144145
Covered by 2 test(s):
146147148
Covered by 2 test(s):
149
Covered by 2 test(s):
150
Covered by 2 test(s):
151
Covered by 2 test(s):
152
Covered by 2 test(s):
153154155
Covered by 2 test(s):
156
Covered by 2 test(s):
157158159
Covered by 2 test(s):
160
Covered by 2 test(s):
161162163
Covered by 2 test(s):
164
Covered by 2 test(s):
165166167
Covered by 2 test(s):
168
Covered by 2 test(s):
169
Covered by 2 test(s):
170171172
Covered by 2 test(s):
173
Covered by 2 test(s):
174175176
Covered by 2 test(s):
177
Covered by 2 test(s):
178
Covered by 2 test(s):
179180181182183
Covered by 2 test(s):
184
Covered by 2 test(s):
185
Covered by 2 test(s):
186187188
Covered by 2 test(s):
189
Covered by 2 test(s):
190191
Covered by 2 test(s):
192193
Covered by 2 test(s):
194195196197198199200201202203204
Covered by 2 test(s):
205
Covered by 2 test(s):
206207
Covered by 2 test(s):
208
Covered by 2 test(s):
209
Covered by 2 test(s):
210
Covered by 2 test(s):
211212
Covered by 2 test(s):
213
Covered by 2 test(s):
214
Covered by 2 test(s):
215
Covered by 2 test(s):
216
Covered by 2 test(s):
217218219220
Covered by 2 test(s):
221
Covered by 2 test(s):
222
Covered by 2 test(s):
223
Covered by 2 test(s):
224
Covered by 2 test(s):
225
Covered by 2 test(s):
226
Covered by 2 test(s):
227
Covered by 2 test(s):
228
Covered by 2 test(s):
229
Covered by 2 test(s):
230231232
Covered by 2 test(s):
233
Covered by 2 test(s):
234235236237238239240241242243
Covered by 2 test(s):
244245
Covered by 2 test(s):
246247248249
Covered by 2 test(s):
250251
Covered by 2 test(s):
252
Covered by 2 test(s):
253
Covered by 2 test(s):
254255
Covered by 2 test(s):
256
Covered by 2 test(s):
257
Covered by 2 test(s):
258259
Covered by 2 test(s):
260
Covered by 2 test(s):
261
Covered by 2 test(s):
262263
Covered by 2 test(s):
264265
Covered by 2 test(s):
266
Covered by 2 test(s):
267268
| <?php /** * Copyright 2018 University of Liverpool * * 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 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace pgb_liv\php_ms\Reader; use pgb_liv\php_ms\Core\Spectra\PrecursorIon; use pgb_liv\php_ms\Core\Spectra\FragmentIon; /** * An MGF reader that creates a new iterable object that will return a raw * entry on each iteration. * * @author Andrew Collins */ class MgfReader implements \Iterator { private $filePath; private $fileHandle; private $filePeek; private $current; private $key = 0; private $massCharge; private $charge; public function __construct($filePath) { $this->filePath = $filePath; } /** * * {@inheritdoc} * @return PrecursorIon */ public function current() { return $this->current; } /** * * {@inheritdoc} * @return int */ public function key() { return $this->key; } /** * * {@inheritdoc} */ public function next() { $this->current = null; if (! feof($this->fileHandle)) { $this->current = $this->parseEntry(); } } /** * * {@inheritdoc} */ public function rewind() { // Reset file parsing to start if ($this->fileHandle != null) { fclose($this->fileHandle); } $this->fileHandle = fopen($this->filePath, 'r'); $this->key = 0; $this->current = $this->parseEntry(); } /** * * {@inheritdoc} * @return bool */ public function valid() { if ($this->current instanceof PrecursorIon) { return true; } return false; } /** * Gets the next line and increments the file iterator * * @return string The next line in the file */ private function getLine() { if ($this->filePeek == null) { return fgets($this->fileHandle); } $ret = $this->filePeek; $this->filePeek = null; return $ret; } /** * Gets the next line, though does not move the file iterator * * @return string The next line in the file */ private function peekLine() { if ($this->filePeek == null) { $this->filePeek = fgets($this->fileHandle); } return $this->filePeek; } private function parseEntry() { $entry = new PrecursorIon(); // Scan to BEGIN IONS $isFound = false; while ($line = $this->getLine()) { $line = trim($line); if (strpos($line, 'BEGIN IONS') !== 0) { continue; } $isFound = true; break; } if (! $isFound) { return null; } $this->massCharge = null; $this->charge = 1; // Scan for key=value pairs while ($line = $this->peekLine()) { if (strpos($line, '=') === false) { break; } $this->parseMeta($entry); } // TODO: Better support required for charge-less data if (! is_null($this->massCharge)) { $entry->setMonoisotopicMassCharge($this->massCharge, $this->charge); } else { $entry->setCharge($this->charge); } // Scan for [m/z] [intensity] [charge] while ($line = $this->peekLine()) { if (strpos($line, 'END IONS') !== false) { break; } $this->parseFragments($entry); } $this->key ++; return $entry; } /** * Parses the meta information from the scan and writes it to the precursor entry * * @param PrecursorIon $precursor * The precursor to append to */ private function parseMeta(PrecursorIon $precursor) { $line = trim($this->getLine()); $pair = explode('=', $line, 2); $value = $pair[1]; if (is_numeric($value)) { $value += 0; } if ($pair[0] == 'TITLE') { $precursor->setTitle($pair[1]); } elseif ($pair[0] == 'PEPMASS') { $chunks = explode(' ', $pair[1]); if (count($chunks) > 1) { $precursor->setIntensity((float) $chunks[1] + 0); } $this->massCharge = (float) $chunks[0] + 0; } elseif ($pair[0] == 'CHARGE') { $this->charge = (int) $pair[1]; } elseif ($pair[0] == 'SCANS') { $precursor->setScan((int) $pair[1] + 0); } elseif ($pair[0] == 'RTINSECONDS') { $window = explode(',', $pair[1]); if (count($window) == 1) { $precursor->setRetentionTime((float) $window[0] + 0); } else { $precursor->setRetentionTimeWindow((float) $window[0] + 0, (float) $window[1] + 0); } } } /** * Parses the fragment information from the scan and writes it to the precursor entry * * @param PrecursorIon $precursor * The precursor to append to */ private function parseFragments(PrecursorIon $precursor) { $line = trim($this->getLine()); if (strlen($line) == 0) { return; } $pair = preg_split('/\\s/', $line, 3); $ion = new FragmentIon(); $fragmentMz = (float) $pair[0]; $fragmentCharge = 1; if (count($pair) > 1) { $ion->setIntensity((float) $pair[1]); } if (count($pair) > 2) { $fragmentCharge = (int) $pair[2]; } $ion->setMonoisotopicMassCharge($fragmentMz, $fragmentCharge); $precursor->addFragmentIon($ion); } } |