tripal_feature.fasta_loader.inc 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002
  1. <?php
  2. /**
  3. * @file
  4. * Provides fasta loading functionality. Creates features based on their specification
  5. * in a fasta file.
  6. */
  7. /**
  8. * @defgroup fasta_loader FASTA Feature Loader
  9. * @ingroup tripal_feature
  10. * @{
  11. * Provides fasta loading functionality.
  12. * Creates features based on their specification
  13. * in a fasta file.
  14. * @}
  15. */
  16. /**
  17. * The form to submit a fasta loading job
  18. *
  19. * @ingroup fasta_loader
  20. */
  21. function tripal_feature_fasta_load_form() {
  22. $form['fasta_file'] = array('#type' => 'textfield','#title' => t('FASTA File'),
  23. '#description' => t('Please enter the full system path for the FASTA file, or a path within the Drupal
  24. installation (e.g. /sites/default/files/xyz.obo). The path must be accessible to the
  25. server on which this Drupal instance is running.'),'#required' => TRUE
  26. );
  27. // get the list of organisms
  28. $sql = "SELECT * FROM {organism} ORDER BY genus, species";
  29. $org_rset = chado_query($sql);
  30. $organisms = array();
  31. $organisms[''] = '';
  32. while ($organism = $org_rset->fetchObject()) {
  33. $organisms[$organism->organism_id] = "$organism->genus $organism->species ($organism->common_name)";
  34. }
  35. $form['organism_id'] = array('#title' => t('Organism'),'#type' => t('select'),
  36. '#description' => t("Choose the organism to which these sequences are associated"),
  37. '#required' => TRUE,'#options' => $organisms
  38. );
  39. // get the sequence ontology CV ID
  40. $values = array('name' => 'sequence'
  41. );
  42. $cv = chado_select_record('cv', array('cv_id'
  43. ), $values);
  44. $cv_id = $cv[0]->cv_id;
  45. $form['seqtype'] = array('#type' => 'textfield','#title' => t('Sequence Type'),
  46. '#required' => TRUE,
  47. '#description' => t('Please enter the Sequence Ontology (SO) term name that describes the sequences in the FASTA file (e.g. gene, mRNA, polypeptide, etc...)'),
  48. '#autocomplete_path' => "admin/tripal/chado/tripal_cv/cvterm/auto_name/$cv_id"
  49. );
  50. $form['method'] = array('#type' => 'radios','#title' => 'Method','#required' => TRUE,
  51. '#options' => array(t('Insert only'),t('Update only'),t('Insert and update')
  52. ),
  53. '#description' => t('Select how features in the FASTA file are handled.
  54. Select "Insert only" to insert the new features. If a feature already
  55. exists with the same name or unique name and type then it is skipped.
  56. Select "Update only" to only update featues that already exist in the
  57. database. Select "Insert and Update" to insert features that do
  58. not exist and upate those that do.'),'#default_value' => 2
  59. );
  60. $form['match_type'] = array('#type' => 'radios','#title' => 'Name Match Type','#required' => TRUE,
  61. '#options' => array(t('Name'),t('Unique name')
  62. ),
  63. '#description' => t('Used for "updates only" or "insert and update" methods. Not required if method type is "insert".
  64. Feature data is stored in Chado with both a human-readable
  65. name and a unique name. If the features in your FASTA file are uniquely identified using
  66. a human-readable name then select the "Name" button. If your features are
  67. uniquely identified using the unique name then select the "Unique name" button. If you
  68. loaded your features first using the GFF loader then the unique name of each
  69. features were indicated by the "ID=" attribute and the name by the "Name=" attribute.
  70. By default, the FASTA loader will use the first word (character string
  71. before the first space) as the name for your feature. If
  72. this does not uniquely identify your feature consider specifying a regular expression in the advanced section below.
  73. Additionally, you may import both a name and a unique name for each sequence using the advanced options.'),
  74. '#default_value' => 1
  75. );
  76. $form['analysis'] = array('#type' => 'fieldset','#title' => t('Analysis Used to Derive Features'),
  77. '#collapsed' => TRUE
  78. );
  79. $form['analysis']['desc'] = array(
  80. '#markup' => t("Why specify an analysis for a data load? All data comes
  81. from some place, even if downloaded from Genbank. By specifying
  82. analysis details for all data uploads, it allows an end user to reproduce the
  83. data set, but at least indicates the source of the data.")
  84. );
  85. // get the list of organisms
  86. $sql = "SELECT * FROM {analysis} ORDER BY name";
  87. $org_rset = chado_query($sql);
  88. $analyses = array();
  89. $analyses[''] = '';
  90. while ($analysis = $org_rset->fetchObject()) {
  91. $analyses[$analysis->analysis_id] = "$analysis->name ($analysis->program $analysis->programversion, $analysis->sourcename)";
  92. }
  93. $form['analysis']['analysis_id'] = array('#title' => t('Analysis'),'#type' => t('select'),
  94. '#description' => t("Choose the analysis to which these features are associated"),
  95. '#required' => TRUE,'#options' => $analyses
  96. );
  97. // Advanced Options
  98. $form['advanced'] = array('#type' => 'fieldset','#title' => t('Advanced Options'),
  99. '#collapsible' => TRUE,'#collapsed' => TRUE
  100. );
  101. $form['advanced']['re_help'] = array('#type' => 'item',
  102. '#value' => t('A regular expression is an advanced method for extracting information from a string of text.
  103. Your FASTA file may contain both a human-readable name and a unique name for each sequence.
  104. If you want to import
  105. both the name and unique name for all sequences, then you must provide regular expressions
  106. so that the loader knows how to separate them.
  107. Otherwise the name and uniquename will be the same.
  108. By default, this loader will use the first word in the definition
  109. lines of the FASTA file
  110. as the name or unique name of the feature.')
  111. );
  112. $form['advanced']['re_name'] = array('#type' => 'textfield',
  113. '#title' => t('Regular expression for the name'),'#required' => FALSE,
  114. '#description' => t('Enter the regular expression that will extract the
  115. feature name from the FASTA definition line. For example, for a
  116. defintion line with a name and unique name separated by a bar \'|\' (>seqname|uniquename),
  117. the regular expression for the name would be, "^(.*?)\|.*$".')
  118. );
  119. $form['advanced']['re_uname'] = array('#type' => 'textfield',
  120. '#title' => t('Regular expression for the unique name'),'#required' => FALSE,
  121. '#description' => t('Enter the regular expression that will extract the
  122. feature name from the FASTA definition line. For example, for a
  123. defintion line with a name and unique name separated by a bar \'|\' (>seqname|uniquename),
  124. the regular expression for the unique name would be "^.*?\|(.*)$").')
  125. );
  126. // Advanced database cross-reference optoins
  127. $form['advanced']['db'] = array('#type' => 'fieldset',
  128. '#title' => t('External Database Reference'),'#weight' => 6,'#collapsed' => TRUE
  129. );
  130. $form['advanced']['db']['re_accession'] = array('#type' => 'textfield',
  131. '#title' => t('Regular expression for the accession'),'#required' => FALSE,
  132. '#description' => t('Enter the regular expression that will extract the accession for the external database for each feature from the FASTA definition line.'),
  133. '#weight' => 2
  134. );
  135. // get the list of databases
  136. $sql = "SELECT * FROM {db} ORDER BY name";
  137. $db_rset = chado_query($sql);
  138. $dbs = array();
  139. $dbs[''] = '';
  140. while ($db = $db_rset->fetchObject()) {
  141. $dbs[$db->db_id] = "$db->name";
  142. }
  143. $form['advanced']['db']['db_id'] = array('#title' => t('External Database'),
  144. '#type' => t('select'),
  145. '#description' => t("Plese choose an external database for which these sequences have a cross reference."),
  146. '#required' => FALSE,'#options' => $dbs,'#weight' => 1
  147. );
  148. $form['advanced']['relationship'] = array('#type' => 'fieldset','#title' => t('Relationships'),
  149. '#weight' => 6,'#collapsed' => TRUE
  150. );
  151. $rels = array();
  152. $rels[''] = '';
  153. $rels['part_of'] = 'part of';
  154. $rels['derives_from'] = 'produced by';
  155. // Advanced references options
  156. $form['advanced']['relationship']['rel_type'] = array('#title' => t('Relationship Type'),
  157. '#type' => t('select'),
  158. '#description' => t("Use this option to create associations, or relationships between the
  159. features of this FASTA file and existing features in the database. For
  160. example, to associate a FASTA file of peptides to existing genes or transcript sequence,
  161. select the type 'produced by'. For a CDS sequences select the type 'part of'"),
  162. '#required' => FALSE,'#options' => $rels,'#weight' => 5
  163. );
  164. $form['advanced']['relationship']['re_subject'] = array('#type' => 'textfield',
  165. '#title' => t('Regular expression for the parent'),'#required' => FALSE,
  166. '#description' => t('Enter the regular expression that will extract the unique
  167. name needed to identify the existing sequence for which the
  168. relationship type selected above will apply.'),'#weight' => 6
  169. );
  170. $form['advanced']['relationship']['parent_type'] = array('#type' => 'textfield',
  171. '#title' => t('Parent Type'),'#required' => FALSE,
  172. '#description' => t('Please enter the Sequence Ontology term for the parent. For example
  173. if the FASTA file being loaded is a set of proteins that are
  174. products of genes, then use the SO term \'gene\' or \'transcript\' or equivalent. However,
  175. this type must match the type for already loaded features.'),
  176. '#weight' => 7
  177. );
  178. $form['button'] = array('#type' => 'submit','#value' => t('Import FASTA file'),'#weight' => 10
  179. );
  180. return $form;
  181. }
  182. /**
  183. * Validate the fasta loader job form
  184. *
  185. * @ingroup fasta_loader
  186. */
  187. function tripal_feature_fasta_load_form_validate($form, &$form_state) {
  188. $fasta_file = trim($form_state['values']['fasta_file']);
  189. $organism_id = $form_state['values']['organism_id'];
  190. $type = trim($form_state['values']['seqtype']);
  191. $method = trim($form_state['values']['method']);
  192. $match_type = trim($form_state['values']['match_type']);
  193. $re_name = trim($form_state['values']['re_name']);
  194. $re_uname = trim($form_state['values']['re_uname']);
  195. $re_accession = trim($form_state['values']['re_accession']);
  196. $db_id = $form_state['values']['db_id'];
  197. $rel_type = $form_state['values']['rel_type'];
  198. $re_subject = trim($form_state['values']['re_subject']);
  199. $parent_type = trim($form_state['values']['parent_type']);
  200. if ($method == 0) {
  201. $method = 'Insert only';
  202. }
  203. if ($method == 1) {
  204. $method = 'Update only';
  205. }
  206. if ($method == 2) {
  207. $method = 'Insert and update';
  208. }
  209. if ($match_type == 0) {
  210. $match_type = 'Name';
  211. }
  212. if ($match_type == 1) {
  213. $match_type = 'Unique name';
  214. }
  215. if ($re_name and !$re_uname and strcmp($match_type, 'Unique name') == 0) {
  216. form_set_error('re_uname', t("You must provide a regular expression to identify the sequence unique name"));
  217. }
  218. if (!$re_name and $re_uname and strcmp($match_type, 'Name') == 0) {
  219. form_set_error('re_name', t("You must provide a regular expression to identify the sequence name"));
  220. }
  221. // check to see if the file is located local to Drupal
  222. $fasta_file = trim($fasta_file);
  223. $dfile = $_SERVER['DOCUMENT_ROOT'] . base_path() . $fasta_file;
  224. if (!file_exists($dfile)) {
  225. // if not local to Drupal, the file must be someplace else, just use
  226. // the full path provided
  227. $dfile = $fasta_file;
  228. }
  229. if (!file_exists($dfile)) {
  230. form_set_error('fasta_file', t("Cannot find the file on the system. Check that the file exists or that the web server has permissions to read the file."));
  231. }
  232. // make sure if a relationship is specified that all fields are provided.
  233. if (($rel_type or $parent_type) and !$re_subject) {
  234. form_set_error('re_subject', t("Please provide a regular expression for the parent"));
  235. }
  236. if (($rel_type or $re_subject) and !$parent_type) {
  237. form_set_error('parent_type', t("Please provide a SO term for the parent"));
  238. }
  239. if (($parent_type or $re_subject) and !$rel_type) {
  240. form_set_error('rel_type', t("Please select a relationship type"));
  241. }
  242. // make sure if a database is specified that all fields are provided
  243. if ($db_id and !$re_accession) {
  244. form_set_error('re_accession', t("Please provide a regular expression for the accession"));
  245. }
  246. if ($re_accession and !$db_id) {
  247. form_set_error('db_id', t("Please select a database"));
  248. }
  249. // check to make sure the types exists
  250. $cvtermsql = "SELECT CVT.cvterm_id
  251. FROM {cvterm} CVT
  252. INNER JOIN {cv} CV on CVT.cv_id = CV.cv_id
  253. LEFT JOIN {cvtermsynonym} CVTS on CVTS.cvterm_id = CVT.cvterm_id
  254. WHERE cv.name = :cvname and (CVT.name = :name or CVTS.synonym = :synonym)";
  255. $cvterm = chado_query($cvtermsql, array(':cvname' => 'sequence',':name' => $type,
  256. ':synonym' => $type
  257. ))->fetchObject();
  258. if (!$cvterm) {
  259. form_set_error('type', t("The Sequence Ontology (SO) term selected for the sequence type is not available in the database. Please check spelling or select another."));
  260. }
  261. if ($rel_type) {
  262. $cvterm = chado_query($cvtermsql, array(':cvname' => 'sequence',':name' => $parent_type,
  263. ':synonym' => $parent_type
  264. ))->fetchObject();
  265. if (!$cvterm) {
  266. form_set_error('parent_type', t("The Sequence Ontology (SO) term selected for the parent relationship is not available in the database. Please check spelling or select another."));
  267. }
  268. }
  269. // check to make sure the 'relationship' and 'sequence' ontologies are loaded
  270. $form_state['storage']['dfile'] = $dfile;
  271. }
  272. /**
  273. * Submit a fasta loading job
  274. *
  275. * @ingroup fasta_loader
  276. */
  277. function tripal_feature_fasta_load_form_submit($form, &$form_state) {
  278. global $user;
  279. $dfile = $form_state['storage']['dfile'];
  280. $organism_id = $form_state['values']['organism_id'];
  281. $type = trim($form_state['values']['seqtype']);
  282. $method = trim($form_state['values']['method']);
  283. $match_type = trim($form_state['values']['match_type']);
  284. $re_name = trim($form_state['values']['re_name']);
  285. $re_uname = trim($form_state['values']['re_uname']);
  286. $re_accession = trim($form_state['values']['re_accession']);
  287. $db_id = $form_state['values']['db_id'];
  288. $rel_type = $form_state['values']['rel_type'];
  289. $re_subject = trim($form_state['values']['re_subject']);
  290. $parent_type = trim($form_state['values']['parent_type']);
  291. $analysis_id = $form_state['values']['analysis_id'];
  292. if ($method == 0) {
  293. $method = 'Insert only';
  294. }
  295. if ($method == 1) {
  296. $method = 'Update only';
  297. }
  298. if ($method == 2) {
  299. $method = 'Insert and update';
  300. }
  301. if ($match_type == 0) {
  302. $match_type = 'Name';
  303. }
  304. if ($match_type == 1) {
  305. $match_type = 'Unique name';
  306. }
  307. $args = array($dfile,$organism_id,$type,$re_name,$re_uname,$re_accession,$db_id,$rel_type,
  308. $re_subject,$parent_type,$method,$user->uid,$analysis_id,$match_type
  309. );
  310. $fname = preg_replace("/.*\/(.*)/", "$1", $dfile);
  311. tripal_add_job("Import FASTA file: $fname", 'tripal_feature', 'tripal_feature_load_fasta', $args, $user->uid);
  312. }
  313. /**
  314. * Actually load a fasta file.
  315. * This is the function called by tripal jobs
  316. *
  317. * @param $dfile The
  318. * full path to the fasta file to load
  319. * @param $organism_id The
  320. * organism_id of the organism these features are from
  321. * @param $type The
  322. * type of features contained in the fasta file
  323. * @param $re_name A
  324. * regular expression to extract the feature.name from the fasta header
  325. * @param $re_uname A
  326. * regular expression to extract the feature.uniquename from the fasta header
  327. * @param $re_accession A
  328. * regular expression to extract the accession of the feature.dbxref_id
  329. * @param $db_id The
  330. * db_id of the above dbxref
  331. * @param $rel_type The
  332. * type of relationship when creating a feature_relationship between this
  333. * feature (object) and an extracted subject
  334. * @param $re_subject The
  335. * regular expression to extract the uniquename of the feature to be the subject
  336. * of the above specified relationship
  337. * @param $parent_type The
  338. * type of the parent feature
  339. * @param $method The
  340. * method of feature adding. (ie: 'Insert only', 'Update only', 'Insert and update')
  341. * @param $uid The
  342. * user id of the user who submitted the job
  343. * @param $analysis_id The
  344. * analysis_id to associate the features in this fasta file with
  345. * @param $match_type Whether
  346. * to match existing features based on the 'Name' or 'Unique name'
  347. * @param $job =
  348. * NULL
  349. * The tripal job
  350. *
  351. * @ingroup fasta_loader
  352. */
  353. function tripal_feature_load_fasta($dfile, $organism_id, $type, $re_name, $re_uname, $re_accession,
  354. $db_id, $rel_type, $re_subject, $parent_type, $method, $uid, $analysis_id, $match_type,
  355. $job = NULL) {
  356. $transaction = db_transaction();
  357. print "\nNOTE: Loading of this GFF file is performed using a database transaction. \n" .
  358. "If the load fails or is terminated prematurely then the entire set of \n" .
  359. "insertions/updates is rolled back and will not be found in the database\n\n";
  360. try {
  361. // First get the type for this sequence.
  362. $cvtermsql = "
  363. SELECT CVT.cvterm_id
  364. FROM {cvterm} CVT
  365. INNER JOIN {cv} CV on CVT.cv_id = CV.cv_id
  366. LEFT JOIN {cvtermsynonym} CVTS on CVTS.cvterm_id = CVT.cvterm_id
  367. WHERE cv.name = :cvname and (CVT.name = :name or CVTS.synonym = :synonym)
  368. ";
  369. $cvterm = chado_query($cvtermsql, array(':cvname' => 'sequence',':name' => $type,
  370. ':synonym' => $type
  371. ))->fetchObject();
  372. if (!$cvterm) {
  373. tripal_report_error("T_fasta_loader", TRIPAL_ERROR, "Cannot find the term type: '%type'", array(
  374. '%type' => $type
  375. ));
  376. return 0;
  377. }
  378. // Second, if there is a parent type then get that.
  379. if ($parent_type) {
  380. $parentcvterm = chado_query($cvtermsql, array(':cvname' => 'sequence',
  381. ':name' => $parent_type,':synonym' => $parent_type
  382. ))->fetchObject();
  383. if (!$parentcvterm) {
  384. tripal_report_error("T_fasta_loader", TRIPAL_ERROR, "Cannot find the paretne term type: '%type'", array(
  385. '%type' => $parentcvterm
  386. ));
  387. return 0;
  388. }
  389. }
  390. // Third, if there is a relationship type then get that.
  391. if ($rel_type) {
  392. $relcvterm = chado_query($cvtermsql, array(':cvname' => 'sequence',':name' => $rel_type,
  393. ':synonym' => $rel_type
  394. ))->fetchObject();
  395. if (!$relcvterm) {
  396. tripal_report_error("T_fasta_loader", TRIPAL_ERROR, "Cannot find the relationship term type: '%type'", array(
  397. '%type' => $relcvterm
  398. ));
  399. return 0;
  400. }
  401. }
  402. // We need to get the table schema to make sure we don't overrun the
  403. // size of fields with what our regular expressions retrieve
  404. $feature_tbl = chado_get_schema('feature');
  405. $dbxref_tbl = chado_get_schema('dbxref');
  406. print "Step 1: finding sequences\n";
  407. $filesize = filesize($dfile);
  408. $fh = fopen($dfile, 'r');
  409. if (!$fh) {
  410. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "cannot open file: %dfile", array(
  411. '%dfile' => $dfile
  412. ));
  413. return 0;
  414. }
  415. // Calculate the interval at which we will print to the screen that status.
  416. $interval = intval($filesize * 0.01);
  417. if ($interval < 1) {
  418. $interval = 1;
  419. }
  420. $inv_read = 0;
  421. $num_read = 0;
  422. // Iterate through the lines of the file. Keep a record for
  423. // where in the file each line is at for later import.
  424. $seqs = array();
  425. $num_seqs = 0;
  426. $prev_pos = 0;
  427. $set_start = FALSE;
  428. while ($line = fgets($fh)) {
  429. $num_read += strlen($line);
  430. $intv_read += strlen($line);
  431. // If we encounter a definition line then get the name, uniquename,
  432. // accession and relationship subject from the definition line.
  433. if (preg_match('/^>/', $line)) {
  434. // Remove the > symbol from the defline.
  435. $defline = preg_replace("/^>/", '', $line);
  436. // Get the feature name if a regular expression is provided.
  437. if ($re_name) {
  438. if (!preg_match("/$re_name/", $defline, $matches)) {
  439. tripal_report_error('trp-fasta', "ERROR: Regular expression for the feature name finds nothing. Line %line.", array(
  440. '%line' => $i
  441. ), 'error');
  442. }
  443. elseif (strlen($matches[1]) > $feature_tbl['fields']['name']['length']) {
  444. tripal_report_error('trp-fasta', "WARNING: Regular expression retrieves a value too long for the feature name. Line %line.", array(
  445. '%line' => $i
  446. ), 'error');
  447. }
  448. else {
  449. $name = trim($matches[1]);
  450. }
  451. }
  452. // If the match_type is name and no regular expression was provided
  453. // then use the first word as the name, otherwise we don't set the name.
  454. elseif (strcmp($match_type, 'Name') == 0) {
  455. if (preg_match("/^\s*(.*?)[\s\|].*$/", $defline, $matches)) {
  456. if (strlen($matches[1]) > $feature_tbl['fields']['name']['length']) {
  457. tripal_report_error('trp-fasta', "WARNING: Regular expression retrieves a feature name too long for the feature name. Line %line.", array(
  458. '%line' => $i
  459. ), 'error');
  460. }
  461. else {
  462. $name = trim($matches[1]);
  463. }
  464. }
  465. else {
  466. tripal_report_error('trp-fasta', "ERROR: Cannot find a feature name. Line %line.", array(
  467. '%line' => $i
  468. ), 'error');
  469. }
  470. }
  471. // Get the feature uniquename if a regular expression is provided.
  472. if ($re_uname) {
  473. if (!preg_match("/$re_uname/", $defline, $matches)) {
  474. tripal_report_error('trp-fasta', "ERROR: Regular expression for the feature unique name finds nothing. Line %line.", array(
  475. '%line' => $i
  476. ), 'error');
  477. }
  478. $uname = trim($matches[1]);
  479. }
  480. // If the match_type is name and no regular expression was provided
  481. // then use the first word as the name, otherwise, we don't set the
  482. // unqiuename.
  483. elseif (strcmp($match_type, 'Unique name') == 0) {
  484. if (preg_match("/^\s*(.*?)[\s\|].*$/", $defline, $matches)) {
  485. $uname = trim($matches[1]);
  486. }
  487. else {
  488. tripal_report_error('trp-fasta', "ERROR: Cannot find a feature unique name. Line %line.", array(
  489. '%line' => $i
  490. ), 'error');
  491. }
  492. }
  493. // Get the accession if a regular expression is provided.
  494. preg_match("/$re_accession/", $defline, $matches);
  495. if (strlen($matches[1]) > $dbxref_tbl['fields']['accession']['length']) {
  496. tripal_report_error('trp-fasta', "WARNING: Regular expression retrieves an accession too long for the feature name. " .
  497. "Cannot add cross reference. Line %line.", array('%line' => $i
  498. ), 'warning');
  499. }
  500. else {
  501. $accession = trim($matches[1]);
  502. }
  503. // Get the relationship subject
  504. preg_match("/$re_subject/", $line, $matches);
  505. $subject = trim($matches[1]);
  506. // Add the details to the sequence.
  507. $seqs[$num_seqs] = array('name' => $name,'uname' => $uname,'accession' => $accession,
  508. 'subject' => $subject,'seq_start' => ftell($fh)
  509. );
  510. $set_start = TRUE;
  511. // If this isn't the first sequence, then we want to specify where
  512. // the previous sequence ended.
  513. if ($num_seqs > 0) {
  514. $seqs[$num_seqs - 1]['seq_end'] = $prev_pos;
  515. }
  516. $num_seqs++;
  517. }
  518. // Keep the current file position so we can use it to set the sequence
  519. // ending position
  520. $prev_pos = ftell($fh);
  521. // update the job status every % bytes
  522. if ($job and $intv_read >= $interval) {
  523. $intv_read = 0;
  524. $percent = sprintf("%.2f", ($num_read / $filesize) * 100);
  525. if ($name) {
  526. print "Parsing Line $i (" . $percent . "%). Memory: " . number_format(memory_get_usage()) .
  527. " bytes.\r";
  528. }
  529. else {
  530. print "Parsing Line $i (" . $percent . "%). Memory: " . number_format(memory_get_usage()) .
  531. " bytes.\r";
  532. }
  533. tripal_set_job_progress($job, intval(($num_read / $filesize) * 100));
  534. }
  535. }
  536. $percent = sprintf("%.2f", ($num_read / $filesize) * 100);
  537. print "Parsing Line $i (" . $percent . "%). Memory: " . number_format(memory_get_usage()) .
  538. " bytes.\r";
  539. tripal_set_job_progress($job, 50);
  540. // Set the end position for the last sequence.
  541. $seqs[$num_seqs - 1]['seq_end'] = $num_read - strlen($line);
  542. // Now that we know where the sequences are in the file we need to add them.
  543. print "\nStep 2: Importing sequences\n";
  544. for ($i = 0; $i < $num_seqs; $i++) {
  545. $seq = $seqs[$i];
  546. print "Importing " . ($i + 1) . " of $num_seqs. ";
  547. if ($name) {
  548. print "Current feature: " . $seq['name'] . ".\n";
  549. }
  550. else {
  551. print "Current feature: " . $seq['uname'] . ".\n";
  552. }
  553. tripal_feature_load_fasta_feature($fh, $seq['name'], $seq['uname'], $db_id, $seq['accession'], $seq['subject'], $rel_type, $parent_type, $analysis_id, $organism_id, $cvterm, $source, $method, $re_name, $match_type, $parentcvterm, $relcvterm, $seq['seq_start'], $seq['seq_end']);
  554. }
  555. tripal_set_job_progress($job, 100);
  556. fclose($fh);
  557. }
  558. catch (Exception $e) {
  559. fclose($fh);
  560. $transaction->rollback();
  561. print "\n"; // make sure we start errors on new line
  562. watchdog_exception('T_fasta_loader', $e);
  563. print "FAILED: Rolling back database changes...\n";
  564. }
  565. print "\nDone\n";
  566. }
  567. /**
  568. * A helper function for tripal_feature_load_fasta() to load a single feature
  569. *
  570. * @ingroup fasta_loader
  571. */
  572. function tripal_feature_load_fasta_feature($fh, $name, $uname, $db_id, $accession, $parent,
  573. $rel_type, $parent_type, $analysis_id, $organism_id, $cvterm, $source, $method, $re_name,
  574. $match_type, $parentcvterm, $relcvterm, $seq_start, $seq_end) {
  575. // Check to see if this feature already exists if the match_type is 'Name'.
  576. if (strcmp($match_type, 'Name') == 0) {
  577. $values = array('organism_id' => $organism_id,'name' => $name,'type_id' => $cvterm->cvterm_id
  578. );
  579. $results = chado_select_record('feature', array('feature_id'
  580. ), $values);
  581. if (count($results) > 1) {
  582. tripal_report_error('T_fasta_loader', "Multiple features exist with the name '%name' of type
  583. '%type' for the organism. skipping", array(
  584. '%name' => $name,'%type' => $type
  585. ));
  586. return 0;
  587. }
  588. if (count($results) == 1) {
  589. $feature = $results[0];
  590. }
  591. }
  592. // Check if this feature already exists if the match_type is 'Unique Name'.
  593. if (strcmp($match_type, 'Unique name') == 0) {
  594. $values = array('organism_id' => $organism_id,'uniquename' => $uname,
  595. 'type_id' => $cvterm->cvterm_id
  596. );
  597. $results = chado_select_record('feature', array('feature_id'
  598. ), $values);
  599. if (count($results) > 1) {
  600. tripal_report_error('T_fasta_loader', TRIPAL_WARNING, "Multiple features exist with the name '%name' of type '%type' for the organism. skipping", array(
  601. '%name' => $name,'%type' => $type
  602. ));
  603. return 0;
  604. }
  605. if (count($results) == 1) {
  606. $feature = $results[0];
  607. }
  608. // If the feature exists but this is an "insert only" then skip.
  609. if ($feature and (strcmp($method, 'Insert only') == 0)) {
  610. tripal_report_error('T_fasta_loader', TRIPAL_WARNING, "Feature already exists '%name' ('%uname') while matching on %type. Skipping insert.", array(
  611. '%name' => $name,'%uname' => $uname,'%type' => drupal_strtolower($match_type)
  612. ));
  613. return 0;
  614. }
  615. }
  616. // If we don't have a feature and we're doing an insert then do the insert.
  617. $inserted = 0;
  618. if (!$feature and (strcmp($method, 'Insert only') == 0 or strcmp($method, 'Insert and update') == 0)) {
  619. // If we have a unique name but not a name then set them to be the same
  620. if (!$uname) {
  621. $uname = $name;
  622. }
  623. elseif (!$name) {
  624. $name = $uname;
  625. }
  626. // Insert the feature record.
  627. $values = array('organism_id' => $organism_id,'name' => $name,'uniquename' => $uname,
  628. 'type_id' => $cvterm->cvterm_id
  629. );
  630. $success = chado_insert_record('feature', $values);
  631. if (!$success) {
  632. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to insert feature '%name (%uname)'", array(
  633. '%name' => $name,'%uname' => $numane
  634. ));
  635. return 0;
  636. }
  637. // now get the feature we just inserted
  638. $values = array('organism_id' => $organism_id,'uniquename' => $uname,
  639. 'type_id' => $cvterm->cvterm_id
  640. );
  641. $results = chado_select_record('feature', array('feature_id'
  642. ), $values);
  643. if (count($results) == 1) {
  644. $inserted = 1;
  645. $feature = $results[0];
  646. }
  647. else {
  648. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to retreive newly inserted feature '%name (%uname)'", array(
  649. '%name' => $name,'%uname' => $numane
  650. ));
  651. return 0;
  652. }
  653. // Add the residues for this feature
  654. tripal_feature_load_fasta_residues($fh, $feature->feature_id, $seq_start, $seq_end);
  655. }
  656. // if we don't have a feature and the user wants to do an update then fail
  657. if (!$feature and (strcmp($method, 'Update only') == 0 or
  658. drupal_strcmp($method, 'Insert and update') == 0)) {
  659. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to find feature '%name' ('%uname') while matching on " .
  660. drupal_strtolower($match_type), array('%name' => $name,'%uname' => $uname
  661. ));
  662. return 0;
  663. }
  664. // if we do have a feature and this is an update then proceed with the update
  665. if ($feature and !$inserted and (strcmp($method, 'Update only') == 0 or
  666. strcmp($method, 'Insert and update') == 0)) {
  667. // if the user wants to match on the Name field
  668. if (strcmp($match_type, 'Name') == 0) {
  669. // if we're matching on the name but do not have a unique name then we
  670. // don't want to update the uniquename.
  671. $values = array();
  672. if ($uname) {
  673. // First check to make sure that by changing the unique name of this
  674. // feature that we won't conflict with another existing feature of
  675. // the same name
  676. $values = array('organism_id' => $organism_id,'uniquename' => $uname,
  677. 'type_id' => $cvterm->cvterm_id
  678. );
  679. $results = chado_select_record('feature', array('feature_id'
  680. ), $values);
  681. if (count($results) > 0) {
  682. tripal_report_error('T_fasta_loader', "Cannot update the feature '%name' with a uniquename of '%uname' and type of '%type' as it
  683. conflicts with an existing feature with the same uniquename and type.", array(
  684. '%name' => $name,'%uname' => $uname,'%type' => $type
  685. ));
  686. return 0;
  687. }
  688. // the changes to the uniquename don't conflict so proceed with the update
  689. $values = array('uniquename' => $uname
  690. );
  691. $match = array('name' => $name,'organism_id' => $organism_id,
  692. 'type_id' => $cvterm->cvterm_id
  693. );
  694. // perform the update
  695. $success = chado_update_record('feature', $match, $values);
  696. if (!$success) {
  697. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to update feature '%name' ('%name')", array(
  698. '%name' => $name,'%uiname' => $uname
  699. ));
  700. return 0;
  701. }
  702. }
  703. }
  704. // If the user wants to match on the unique name field.
  705. if (strcmp($match_type, 'Unique name') == 0) {
  706. // If we're matching on the uniquename and have a new name then
  707. // we want to update the name.
  708. $values = array();
  709. if ($name) {
  710. $values = array('name' => $name
  711. );
  712. $match = array('uniquename' => $uname,'organism_id' => $organism_id,
  713. 'type_id' => $cvterm->cvterm_id
  714. );
  715. $success = chado_update_record('feature', $match, $values);
  716. if (!$success) {
  717. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to update feature '%name' ('%name')", array(
  718. '%name' => $name,'%uiname' => $uname
  719. ));
  720. return 0;
  721. }
  722. }
  723. }
  724. }
  725. // Update the residues for this feature
  726. tripal_feature_load_fasta_residues($fh, $feature->feature_id, $seq_start, $seq_end);
  727. // add in the analysis link
  728. if ($analysis_id) {
  729. // if the association doens't alredy exist then add one
  730. $values = array('analysis_id' => $analysis_id,'feature_id' => $feature->feature_id
  731. );
  732. $results = chado_select_record('analysisfeature', array('analysisfeature_id'
  733. ), $values);
  734. if (count($results) == 0) {
  735. $success = chado_insert_record('analysisfeature', $values);
  736. if (!$success) {
  737. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to associate analysis and feature '%name' ('%name')", array(
  738. '%name' => $name,'%uname' => $uname
  739. ));
  740. return 0;
  741. }
  742. }
  743. }
  744. // now add the database cross reference
  745. if ($db_id) {
  746. // check to see if this accession reference exists, if not add it
  747. $values = array('db_id' => $db_id,'accession' => $accession
  748. );
  749. $results = chado_select_record('dbxref', array('dbxref_id'
  750. ), $values);
  751. // if the accession doesn't exist then add it
  752. if (count($results) == 0) {
  753. $results = chado_insert_record('dbxref', $values);
  754. if (!$results) {
  755. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to add database accession '%accession'", array(
  756. '%accession' => $accession
  757. ));
  758. return 0;
  759. }
  760. $results = chado_select_record('dbxref', array('dbxref_id'
  761. ), $values);
  762. if (count($results) == 1) {
  763. $dbxref = $results[0];
  764. }
  765. else {
  766. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to retreive newly inserted dbxref '%name (%uname)'", array(
  767. '%name' => $name,'%uname' => $numane
  768. ));
  769. return 0;
  770. }
  771. }
  772. else {
  773. $dbxref = $results[0];
  774. }
  775. // check to see if the feature dbxref record exists if not, then add it
  776. $values = array('feature_id' => $feature->feature_id,'dbxref_id' => $dbxref->dbxref_id
  777. );
  778. $results = chado_select_record('feature_dbxref', array('feature_dbxref_id'
  779. ), $values);
  780. if (count($results) == 0) {
  781. $success = chado_insert_record('feature_dbxref', $values);
  782. if (!$success) {
  783. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to add associate database accession '%accession' with feature", array(
  784. '%accession' => $accession
  785. ));
  786. return 0;
  787. }
  788. }
  789. }
  790. // now add in the relationship if one exists. If not, then add it
  791. if ($rel_type) {
  792. $values = array('organism_id' => $organism_id,'uniquename' => $parent,
  793. 'type_id' => $parentcvterm->cvterm_id
  794. );
  795. $results = chado_select_record('feature', array('feature_id'
  796. ), $values);
  797. if (count($results) != 1) {
  798. tripal_report_error('T_fasta_loader', "Cannot find a unique fature for the parent '%parent' of type
  799. '%type' for the feature.", array(
  800. '%parent' => $parent,'%type' => $parent_type
  801. ));
  802. return 0;
  803. }
  804. $parent_feature = $results[0];
  805. // check to see if the relationship already exists if not then add it
  806. $values = array('subject_id' => $feature->feature_id,'object_id' => $parent_feature->feature_id,
  807. 'type_id' => $relcvterm->cvterm_id
  808. );
  809. $results = chado_select_record('feature_relationship', array('feature_relationship_id'
  810. ), $values);
  811. if (count($results) == 0) {
  812. $success = chado_insert_record('feature_relationship', $values);
  813. if (!$success) {
  814. tripal_report_error('T_fasta_loader', TRIPAL_ERROR, "Failed to add associate database accession '%accession' with feature", array(
  815. '%accession' => $accession
  816. ));
  817. return 0;
  818. }
  819. }
  820. }
  821. }
  822. /**
  823. * Adds the residues column to the feature.
  824. *
  825. * This function seeks to the proper location in the file for the sequence
  826. * and reads in chunks of sequence and appends them to the feature.residues
  827. * column in the database.
  828. *
  829. * @param unknown $fh
  830. * @param unknown $feature_id
  831. * @param unknown $seq_start
  832. * @param unknown $seq_end
  833. */
  834. function tripal_feature_load_fasta_residues($fh, $feature_id, $seq_start, $seq_end) {
  835. // First position the file at the beginning of the sequence
  836. fseek($fh, $seq_start, SEEK_SET);
  837. $chunk_size = 100000000;
  838. $chunk = '';
  839. $seqlen = ($seq_end - $seq_start) + 1;
  840. // Calculate the interval at which we updated the precent complete.
  841. $interval = intval($seqlen * 0.01);
  842. if ($interval < 1) {
  843. $interval = 1;
  844. }
  845. // We don't to repeat the update too often or it slows things down, so
  846. // if the interval is less than 1000 then bring it up to that.
  847. if ($interval < 100000) {
  848. $interval = 100000;
  849. }
  850. $chunk_intv_read = 0;
  851. $intv_read = 0;
  852. $num_read = 0;
  853. $total_seq_size = 0;
  854. // First, make sure we don't have a null in the residues
  855. $sql = "UPDATE {feature} SET residues = '' WHERE feature_id = :feature_id";
  856. chado_query($sql, array(':feature_id' => $feature_id
  857. ));
  858. // Read in the lines until we reach the end of the sequence. Once we
  859. // get a specific bytes read then append the sequence to the one in the
  860. // database.
  861. print "Sequence complete: 0%. Memory: " . number_format(memory_get_usage()) . " bytes. \r";
  862. while ($line = fgets($fh)) {
  863. $num_read += strlen($line) + 1;
  864. $chunk_intv_read += strlen($line) + 1;
  865. $intv_read += strlen($line) + 1;
  866. $chunk .= trim($line);
  867. // If we've read in enough of the sequence then append it to the database.
  868. if ($chunk_intv_read >= $chunk_size) {
  869. $sql = "
  870. UPDATE {feature}
  871. SET residues = residues || :chunk
  872. WHERE feature_id = :feature_id
  873. ";
  874. $success = chado_query($sql, array(':feature_id' => $feature_id,':chunk' => $chunk
  875. ));
  876. if (!$success) {
  877. return FALSE;
  878. }
  879. $total_seq_size += strlen($chunk);
  880. $chunk = '';
  881. $chunk_intv_read = 0;
  882. }
  883. if ($intv_read >= $interval) {
  884. $percent = sprintf("%.2f", ($total_seq_size / $seqlen) * 100);
  885. print "Sequence complete: " . $percent . "%. Memory: " . number_format(memory_get_usage()) .
  886. " bytes. \r";
  887. $intv_read = 0;
  888. }
  889. // If we've reached the ned of the sequence then break out of the loop
  890. if (ftell($fh) == $seq_end) {
  891. break;
  892. }
  893. }
  894. // write the last bit of sequence if it remains
  895. if (strlen($chunk) > 0) {
  896. $sql = "
  897. UPDATE {feature}
  898. SET residues = residues || :chunk
  899. WHERE feature_id = :feature_id
  900. ";
  901. $success = chado_query($sql, array(':feature_id' => $feature_id,':chunk' => $chunk
  902. ));
  903. if (!$success) {
  904. return FALSE;
  905. }
  906. $total_seq_size += strlen($chunk);
  907. $chunk = '';
  908. $chunk_intv_read = 0;
  909. }
  910. // Now update the seqlen and md5checksum fields
  911. $sql = "UPDATE {feature} SET seqlen = char_length(residues), md5checksum = md5(residues) WHERE feature_id = :feature_id";
  912. chado_query($sql, array(':feature_id' => $feature_id
  913. ));
  914. $percent = sprintf("%.2f", ($num_read / $seqlen) * 100);
  915. print "Sequence complete: " . $percent . "%. Memory: " . number_format(memory_get_usage()) .
  916. " bytes. \r";
  917. }