tripal_analysis_blast.module 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839
  1. <?php
  2. require_once "parse_blast_XML.inc";
  3. /*******************************************************************************
  4. * Tripal Blast Result lets users show/hide blast results associated
  5. * with a tripal feature
  6. ******************************************************************************/
  7. function tripal_analysis_blast_init(){
  8. // Add javascript and style sheet
  9. drupal_add_css(drupal_get_path('theme', 'tripal').'/css/tripal_analysis_blast.css');
  10. drupal_add_js(drupal_get_path('theme', 'tripal').'/js/tripal_analysis_blast.js');
  11. }
  12. /*******************************************************************************
  13. * tripal_analysis_blast_menu()
  14. * HOOK: Implementation of hook_menu()
  15. * Entry points and paths of the module
  16. */
  17. function tripal_analysis_blast_menu() {
  18. // Show top 10/25/all blast results for ajax calls
  19. $items['tripal_top_blast'] = array(
  20. 'path' => 'top_blast',
  21. 'title' => t('Blast Hits'),
  22. 'page callback' => 'tripal_get_feature_blast_results_ajax',
  23. 'page arguments' => array(1,2,3),
  24. 'access arguments' => array('access content'),
  25. 'type' => MENU_CALLBACK
  26. );
  27. // Show regular expressions for selected database in Blast admin page
  28. $items['admin/tripal/tripal_analysis/tripal_blast_regex/%'] = array(
  29. 'title' => t('Blast Regex'),
  30. 'page callback' => 'tripal_get_blast_regex',
  31. 'page arguments' => array(4),
  32. 'access arguments' => array('administer site configuration'),
  33. 'type' => MENU_CALLBACK
  34. );
  35. $items['tripal_blast_report'] = array(
  36. 'title' => t('Homology Report'),
  37. 'page callback' => 'tripal_get_blast_report',
  38. 'page arguments' => array(1,2,3,4,5),
  39. 'access arguments' => array('access chado_analysis_blast content'),
  40. 'type' => MENU_CALLBACK,
  41. 'file' => 'tripal_analysis_blast_htmlreport.inc'
  42. );
  43. return $items;
  44. }
  45. /*******************************************************************************
  46. * tripal_analysis_blast_nodeapi()
  47. * HOOK: Implementation of hook_nodeapi()
  48. * Display blast results for allowed node types
  49. */
  50. function tripal_analysis_blast_nodeapi(&$node, $op, $teaser, $page) {
  51. switch ($op) {
  52. case 'view':
  53. // Find out which node types for showing the blast
  54. $types_to_show = variable_get('tripal_analysis_blast_setting',
  55. array('chado_feature'));
  56. // Abort if this node is not one of the types we should show.
  57. if (!in_array($node->type, $types_to_show, TRUE)) {
  58. break;
  59. }
  60. // Add blast to the content item if it's not a teaser
  61. if (!$teaser && $node->feature->feature_id) {
  62. if($node->build_mode == NODE_BUILD_SEARCH_INDEX){
  63. $node->content['tripal_analysis_blast_index_version'] = array(
  64. '#value' => theme('tripal_analysis_blast_results_index_version',$node),
  65. '#weight' => 8,
  66. );
  67. } else {
  68. if(strcmp($node->type,'chado_feature')==0){
  69. // Show blast result if not at teaser view
  70. $node->content['tripal_feature_blast_results'] = array(
  71. '#value' => theme('tripal_feature_blast_results', $node),
  72. '#weight' => 8
  73. );
  74. }
  75. }
  76. }
  77. }
  78. }
  79. /************************************************************************
  80. * We need to let drupal know about our theme functions and their arguments.
  81. * We create theme functions to allow users of the module to customize the
  82. * look and feel of the output generated in this module
  83. */
  84. function tripal_analysis_blast_theme () {
  85. return array(
  86. 'tripal_analysis_blast_results_index_version' => array (
  87. 'arguments' => array('node'),
  88. ),
  89. 'tripal_feature_blast_results' => array(
  90. 'arguments' => array('node'=> null),
  91. 'template' => 'tripal_feature_blast_results',
  92. )
  93. );
  94. }
  95. /*******************************************************************************
  96. *
  97. */
  98. function tripal_get_feature_blast_results_ajax($feature_id, $db_id, $max){
  99. $sql = "SELECT nid FROM {chado_feature} WHERE feature_id = %d";
  100. $nid = db_fetch_object(db_query($sql,$feature_id));
  101. $node = node_load($nid->nid);
  102. // add the additional variables that the theme needs to generate the output
  103. $node->db_id = $db_id;
  104. $node->max = $max;
  105. // call the theme to rebuild the blast results
  106. drupal_json(array('update' => theme('tripal_feature_blast_results',$node)));
  107. }
  108. /*******************************************************************************
  109. *
  110. */
  111. function tripal_analysis_blast_preprocess_tripal_feature_blast_results(&$variables){
  112. $feature = $variables['node']->feature;
  113. $db_id = $variables['node']->db_id;
  114. $max = 10;
  115. if(isset($variables['node']->max)){
  116. $max = $variables['node']->max;
  117. }
  118. $blast_results = tripal_get_feature_blast_results($feature->feature_id, $db_id, $max);
  119. $variables['tripal_analysis_blast']['blast_results_list'] = $blast_results;
  120. }
  121. /*******************************************************************************
  122. * Prepare blast result for the feature shown on the page
  123. */
  124. function theme_tripal_analysis_blast_results_index_version ($node) {
  125. $feature = $node->feature;
  126. $content = tripal_get_blast_results_index_version($feature->feature_id);
  127. return $content;
  128. }
  129. /*******************************************************************************
  130. * tripal_get_feature_blast_results()
  131. * Get blast result from featureprop table for the feature
  132. */
  133. function tripal_get_feature_blast_results($feature_id, $db_id, $max){
  134. // Get the blast results stored as XML from the analysisfeatureprop table
  135. // the type for the property is named 'analysis_blast_output_iteration_hits'
  136. // and is found in the 'tripal' controlled vocabulary. This CV term was
  137. // added by this module.
  138. $sql = "SELECT AP.value AS apvalue, AFP.value AS afpvalue, AF.analysis_id AS aid
  139. FROM {analysisfeatureprop} AFP
  140. INNER JOIN {analysisfeature} AF ON AF.analysisfeature_id = AFP.analysisfeature_id
  141. INNER JOIN {analysisprop} AP ON AP.analysis_id = AF.analysis_id
  142. INNER JOIN {cvterm} CVT on AFP.type_id = CVT.cvterm_id
  143. INNER JOIN {cv} CV on CVT.cv_id = CV.cv_id
  144. WHERE AF.feature_id = %d AND CV.name = '%s' AND
  145. CVT.name = '%s' AND AP.value like '%|%' ";
  146. $previous_db = tripal_db_set_active('chado');
  147. $result = db_query($sql, $feature_id,'tripal','analysis_blast_output_iteration_hits');
  148. tripal_db_set_active($previous_db);
  149. // get the HTML content for viewing each of the XML file
  150. $blast_obj_array = array ();
  151. $blast_obj_counter = 0;
  152. while ($analysisfeatureprop = db_fetch_object($result)) {
  153. // the database db_id is stored in the value field of the table
  154. // along with the XML. The two are separated by a bar. We
  155. // will separate these two:
  156. $blastsettings = explode("|", $analysisfeatureprop->apvalue);
  157. // if we don't have the proper number of fields in the value column then
  158. // skip this entry
  159. if(count($blastsettings) != 3){
  160. continue;
  161. }
  162. if(!$blastsettings[0]){
  163. continue;
  164. }
  165. $att_db_id = $blastsettings[0];
  166. // get analysis name and date
  167. $previous_db = tripal_db_set_active('chado');
  168. $sql = "SELECT analysis_id AS aid, name, to_char(timeexecuted, 'MM-DD-YYYY') AS time
  169. FROM {analysis}
  170. WHERE analysis_id = %d";
  171. $analysis = db_fetch_object(db_query($sql, $analysisfeatureprop->aid));
  172. tripal_db_set_active($previous_db);
  173. // Get db object using the db_id
  174. $previous_db = tripal_db_set_active('chado');
  175. $sql = "SELECT * FROM {db} WHERE db_id=%d";
  176. $db = db_fetch_object(db_query($sql, $att_db_id));
  177. tripal_db_set_active($previous_db);
  178. // we want to add this blast result to our list if a database id has
  179. // not been specified or if it has been specified and the database id
  180. // for this analysis matches that of the one requested
  181. if(!$db_id or ($db_id and $att_db_id == $db_id)) {
  182. $blast_obj = tripal_analysis_blast_get_result_object($analysisfeatureprop->afpvalue,$db,$max,$feature_id, $analysis);
  183. $blast_obj->analysis = $analysis;
  184. $blast_obj_array [$blast_obj_counter] = $blast_obj;
  185. $blast_obj_counter ++;
  186. }
  187. }
  188. return $blast_obj_array;
  189. }
  190. /*******************************************************************************
  191. * Scanning the file folder for blast results and prepare content for indexing
  192. */
  193. function tripal_get_blast_results_index_version ($feature_id){
  194. // Get cvterm_id for 'analysis_blast_output_iteration_hits' which is required
  195. // for inserting into the analysisfeatureprop table
  196. $previous_db = tripal_db_set_active('chado');
  197. $sql = "SELECT CVT.cvterm_id FROM {cvterm} CVT ".
  198. "INNER JOIN cv ON cv.cv_id = CVT.cv_id ".
  199. "WHERE CVT.name = 'analysis_blast_output_iteration_hits' ".
  200. "AND CV.name = 'tripal'";
  201. $type_id = db_result(db_query($sql));
  202. // Get xml string from analysisfeatureprop value column, get db_id from analysisprop value column
  203. // , and get analysis_id from analysisfeature table
  204. $sql = "SELECT AP.value AS apvalue, AFP.value AS afpvalue, AF.analysis_id AS aid
  205. FROM {analysisfeatureprop} AFP
  206. INNER JOIN analysisfeature AF ON AF.analysisfeature_id = AFP.analysisfeature_id
  207. INNER JOIN analysisprop AP ON AP.analysis_id = AF.analysis_id
  208. WHERE feature_id = %d
  209. AND AFP.type_id = %d ";
  210. $result = db_query($sql, $feature_id, $type_id);
  211. tripal_db_set_active($previous_db);
  212. // get the HTML content for viewing each of the XML file
  213. while ($analysisfeatureprop = db_fetch_object($result)) {
  214. // get analysis name and date
  215. $previous_db = tripal_db_set_active('chado');
  216. $sql = "SELECT analysis_id AS aid, name, to_char(timeexecuted, 'MM-DD-YYYY') AS time
  217. FROM {analysis} WHERE analysis_id = %d";
  218. $analysis = db_fetch_object(db_query($sql, $analysisfeatureprop->aid));
  219. tripal_db_set_active($previous_db);
  220. $blastsettings = explode("|", $analysisfeatureprop->apvalue);
  221. $att_db_id = $blastsettings [0];
  222. // Get db object using the db_id
  223. $previous_db = tripal_db_set_active('chado');
  224. $sql = "SELECT * FROM {db} WHERE db_id=%d";
  225. $db = db_fetch_object(db_query($sql, $att_db_id));
  226. tripal_db_set_active($previous_db);
  227. // Only index best 10 hits because the default page only shows 10 blast results
  228. $max = 10;
  229. $content .= parse_NCBI_Blast_XML_index_version($analysisfeatureprop->afpvalue,$db,$max,$feature_id,$ajax, $analysis);
  230. }
  231. return $content;
  232. }
  233. /*******************************************************************************
  234. * Tripal Blast administrative setting form. This function is called by
  235. * tripal_analysis module which asks for an admin form to show on the page
  236. */
  237. function tripal_analysis_blast_get_settings() {
  238. // Get an array of node types with internal names as keys
  239. $options = node_get_types('names');
  240. // Add 'chado_feature' to allowed content types for showing blast results
  241. $allowedoptions ['chado_feature'] = "Show blast results on feature pages";
  242. $form['description'] = array(
  243. '#type' => 'item',
  244. '#value' => t("Most chado features were analyzed by blast against major sequence databases. This option allows user to display the blast analysis results. Please read user manual for storage and display of blast files. Check the box to enable the analysis results. Uncheck to disable it."),
  245. '#weight' => 0,
  246. );
  247. $form['tripal_analysis_blast_setting'] = array(
  248. '#type' => 'checkboxes',
  249. '#options' => $allowedoptions,
  250. '#default_value' => variable_get('tripal_analysis_blast_setting',
  251. array('chado_feature')),
  252. );
  253. $form['blast_parser'] = array(
  254. '#title' => t('Blast Parser Settings'),
  255. '#type' => 'fieldset',
  256. '#description' => t('Configure parsers for showing blast results. Each database is '.
  257. 'allowed to have one xml parser.'),
  258. '#weight' => 10
  259. );
  260. $previous_db = tripal_db_set_active('chado'); // use chado database
  261. // get a list of db from chado for user to choose
  262. $sql = 'SELECT db_id, name FROM {db} ORDER BY lower(name)';
  263. $results = db_query ($sql);
  264. $blastdbs = array();
  265. while ($db = db_fetch_object($results)){
  266. $blastdbs[$db->db_id] = $db->name;
  267. }
  268. $form['db_options'] = array(
  269. '#type' => 'value',
  270. '#value' => $blastdbs
  271. );
  272. $form['blast_parser']['blastdb'] = array(
  273. '#title' => t('Database'),
  274. '#type' => 'select',
  275. '#description' => t('The database used for the blast analysis.'),
  276. '#options' => $form['db_options']['#value'],
  277. '#attributes' => array(
  278. 'onChange' => "return tripal_update_regex(this)",
  279. )
  280. );
  281. $form['blast_parser']['displayname'] = array(
  282. '#title' => t('Title for the blast analysis'),
  283. '#type' => 'textfield',
  284. );
  285. $form['blast_parser']['gb_style_parser'] = array(
  286. '#title' => t('Use Genebank style parser. This will clear all regular expression settings for the selected database.'),
  287. '#type' => 'checkbox',
  288. '#attributes' => array(
  289. 'onClick' => "return tripal_set_genbank_style(this)",
  290. )
  291. );
  292. $form['blast_parser']['hit_id'] = array(
  293. '#title' => t('Regular expression for Hit Name'),
  294. '#type' => 'textfield',
  295. );
  296. $form['blast_parser']['hit_def'] = array(
  297. '#title' => t('Regular expression for Hit Description'),
  298. '#type' => 'textfield',
  299. );
  300. $form['blast_parser']['hit_accession'] = array(
  301. '#title' => t('Regular expression for Hit Accession'),
  302. '#type' => 'textfield',
  303. );
  304. $form['blast_parser']['button'] = array(
  305. '#type' => 'submit',
  306. '#value' => t('Save settings')
  307. );
  308. tripal_db_set_active($previous_db); // use drupal database
  309. $settings->form = $form;
  310. $settings->title = "Tripal Blast";
  311. return $settings;
  312. }
  313. /*******************************************************************************
  314. * This function is only called by ajax to get regular expressions for blast
  315. * admin page
  316. */
  317. function tripal_get_blast_regex ($db_id) {
  318. $sql = "SELECT * FROM {tripal_analysis_blast} WHERE db_id = %d";
  319. $blast_regexs = db_fetch_object(db_query($sql, $db_id));
  320. drupal_json(array(
  321. 'name' => $blast_regexs->displayname,
  322. 'genbank_style' => $blast_regexs->genbank_style,
  323. 'reg1' => $blast_regexs->regex_hit_id,
  324. 'reg2' => $blast_regexs->regex_hit_def,
  325. 'reg3' => $blast_regexs->regex_hit_accession)
  326. );
  327. }
  328. /*******************************************************************************
  329. * Provide information to drupal about the node types that we're creating
  330. * in this module
  331. */
  332. function tripal_analysis_blast_node_info() {
  333. $nodes = array();
  334. $nodes['chado_analysis_blast'] = array(
  335. 'name' => t('Analysis: Blast'),
  336. 'module' => 'chado_analysis_blast',
  337. 'description' => t('A blast analysis from the chado database'),
  338. 'has_title' => FALSE,
  339. 'title_label' => t('Analysis: Blast'),
  340. 'has_body' => FALSE,
  341. 'body_label' => t('Blast Analysis Description'),
  342. 'locked' => TRUE
  343. );
  344. return $nodes;
  345. }
  346. /*******************************************************************************
  347. * Provide a Blast Analysis form
  348. */
  349. function chado_analysis_blast_form ($node){
  350. //dprint_r($node);
  351. // add in the default fields
  352. $form = chado_analysis_form($node);
  353. // set the default values
  354. $blast = $node->analysis->tripal_analysis_blast;
  355. $blastdb = $blast->blastdb;
  356. $blastfile = $blast->blastfile;
  357. $blastparameters = $blast->blastparameters;
  358. $query_re = $blast->query_re;
  359. $query_type = $blast->query_type;
  360. $query_uniquename = $blast->query_uniquename;
  361. $form['blast'] = array(
  362. '#title' => t('Blast Settings'),
  363. '#type' => 'fieldset',
  364. '#description' => t('Specific Settings for Blast Analysis.'),
  365. '#collapsible' => TRUE,
  366. '#attributes' => array('id' => 'blast-extra-settings'),
  367. '#weight' => 11
  368. );
  369. $previous_db = tripal_db_set_active('chado'); // use chado database
  370. // get a list of db from chado for user to choose
  371. $sql = 'SELECT db_id, name FROM {db} ORDER BY lower(name)';
  372. $results = db_query ($sql);
  373. tripal_db_set_active($previous_db);
  374. $blastdbs = array();
  375. while ($db = db_fetch_object($results)){
  376. $blastdbs[$db->db_id] = $db->name;
  377. }
  378. $form['db_options'] = array(
  379. '#type' => 'value',
  380. '#value' => $blastdbs
  381. );
  382. $form['blast']['blastdb'] = array(
  383. '#title' => t('Database'),
  384. '#type' => 'select',
  385. '#description' => t('The database used for the blast analysis. If the database does not appear in this list, please add it.'),
  386. '#options' => $form['db_options']['#value'],
  387. '#default_value' => $blastdb,
  388. );
  389. $form['blast']['blastfile'] = array(
  390. '#title' => t('Blast XML File/Directory: (if you input a directory without the tailing slash, all xml files in the directory will be loaded)'),
  391. '#type' => 'textfield',
  392. '#description' => t('The xml output file generated by blast in full path.'),
  393. '#default_value' => $blastfile,
  394. );
  395. $form['blast']['blastfile_ext'] = array(
  396. '#title' => t('Blast XML file extension'),
  397. '#type' => 'textfield',
  398. '#description' => t('If a directory is provide for the blast file setting above, then a file extension can be provided here. Files with this extension in the directory will be parsed. If no extension is provided then files with a .xml extension will be parsed within the directory. Please provide the extension without the preceeding period (e.g. "out" rather than ".out"'),
  399. '#default_value' => $blastfile_ext,
  400. );
  401. $form['blast']['no_parsed'] = array(
  402. '#title' => t('Number of hits to be parsed'),
  403. '#type' => 'textfield',
  404. '#description' => t("The number of hits to be parsed. Tripal will parse only top 10 hits if you input '10'' in this field."),
  405. '#default_value' => 'all',
  406. );
  407. $form['blast']['query_re'] = array(
  408. '#title' => t('Query Name RE'),
  409. '#type' => 'textfield',
  410. '#description' => t('Enter the regular expression that will extract the '.
  411. 'feature name from the query line in the blast results. This should be '.
  412. 'the same as the definition line in the query FASTA file. This option is '.
  413. 'is only required when the query does not identically match a feature '.
  414. 'in the database.'),
  415. '#default_value' => $query_re,
  416. );
  417. $form['blast']['query_type'] = array(
  418. '#title' => t('Query Type'),
  419. '#type' => 'textfield',
  420. '#description' => t('Please enter the Sequence Ontology term that describes '.
  421. 'the query sequences used for blasting. This is only necessary if two '.
  422. 'or more sequences have the same name.'),
  423. '#default_value' => $query_type,
  424. );
  425. $form['blast']['query_uniquename'] = array(
  426. '#title' => t('Use Unique Name'),
  427. '#type' => 'checkbox',
  428. '#description' => t('Select this checboxk if the query name in the blast file '.
  429. 'matches the uniquename of the feature. By default, the blast results will '.
  430. 'mapped to the "name" of the feature.'),
  431. '#default_value' => $query_uniquename,
  432. );
  433. $form['blast']['blastparameters'] = array(
  434. '#title' => t('Parameters'),
  435. '#type' => 'textfield',
  436. '#description' => t('The parameters for running the blast analysis.'),
  437. '#default_value' => $blastparameters,
  438. );
  439. $form['blast']['blastjob'] = array(
  440. '#type' => 'checkbox',
  441. '#title' => t('Submit a job to parse the xml output into analysisfeatureprop table'),
  442. '#description' => t('Note: features associated with the blast results must '.
  443. 'exist in chado before parsing the file. Otherwise, blast '.
  444. 'results that cannot be linked to a feature will be '.
  445. 'discarded. '),
  446. '#default_value' => $blastjob
  447. );
  448. $form['blast']['blastbesthit'] = array(
  449. '#type' => 'checkbox',
  450. '#title' => t('Submit a job to generate a "best hits" report.'),
  451. '#description' => t('Note: the checkbox above must also be selected.'),
  452. '#default_value' => $blastbesthit
  453. );
  454. return $form;
  455. }
  456. function chado_analysis_blast_validate($node, &$form){
  457. ##dprint_r($node);
  458. // This validation is being used for three activities:
  459. // CASE A: Update a node that exists in both drupal and chado
  460. // CASE B: Synchronizing a node from chado to drupal
  461. // CASE C: Inserting a new node that exists in niether drupal nor chado
  462. // Only nodes being updated will have an nid already
  463. if($node->nid){
  464. //---------------------------------------------------
  465. // CASE A: We are validating a form for updating an existing node
  466. //---------------------------------------------------
  467. // TO DO: check that the new fields don't yield a non-unique primary key in chado
  468. }
  469. else{
  470. // To differentiate if we are syncing or creating a new analysis altogther, see if an
  471. // analysis_id already exists
  472. if($node->analysis_id){
  473. //---------------------------------------------------
  474. // CASE B: Synchronizing a node from chado to drupal
  475. //---------------------------------------------------
  476. }
  477. else{
  478. //---------------------------------------------------
  479. // CASE C: We are validating a form for inserting a new node
  480. //---------------------------------------------------
  481. // The primary key for the chado analysis table is
  482. // program, programversion, sourcename
  483. // Check to see if this analysis really is new -ie, it doesn't have the same
  484. // primary key as any other analysis
  485. $sql = "SELECT analysis_id ".
  486. "FROM {analysis} ".
  487. "WHERE program='%s'".
  488. "AND programversion='%s'".
  489. "AND sourcename='%s'";
  490. $previous_db = tripal_db_set_active('chado');
  491. $analysis_id = db_result(db_query($sql, $node->program, $node->programversion, $node->sourcename));
  492. tripal_db_set_active($previous_db);
  493. if($analysis_id){
  494. //---------------------------------------------------
  495. // this primary key already exists in chado analysis table!
  496. //---------------------------------------------------
  497. // check to see if it has also been synced with drupal
  498. $sql = "SELECT nid FROM {chado_analysis} ".
  499. "WHERE analysis_id = %d";
  500. $node_id = db_result(db_query($sql, $analysis_id));
  501. if($node_id){
  502. //---------------------------------------------------
  503. // the analysis has already been synced with drupal, redirect the user
  504. // to modify that node or start over
  505. //---------------------------------------------------
  506. $error = 'This analysis already exists in the chado database (analysis id ';
  507. $error .= $analysis_id.') and has been synchronized ';
  508. $error .= 'with drupal. See node '.$node_id.' if you wish to update that analysis. ';
  509. $error .= ' For a new analysis, please select a unique primary key ';
  510. $error .= '(primary key consists of sourcename, program and programversion).';
  511. form_set_error('sourcename', t($error));
  512. }
  513. else{
  514. //---------------------------------------------------
  515. // the analysis does not exist in drupal - tell the user
  516. // to sync from chado or create a new unique primary key
  517. //---------------------------------------------------
  518. $error = 'This analysis already exists in the chado database (analysis id ';
  519. $error .= $analysis_id.') but has not been synchronized ';
  520. $error .= 'with drupal. See the tripal admin pages to synchronize. ';
  521. $error .= ' For a new analysis, please select a unique primary key ';
  522. $error .= '(primary key consists of sourcename, program and programversion).';
  523. form_set_error('sourcename', t($error));
  524. }
  525. }
  526. }
  527. }
  528. }
  529. /*******************************************************************************
  530. * When a node is requested by the user this function is called to allow us
  531. * to add auxiliary data to the node object.
  532. */
  533. function chado_analysis_blast_load($node){
  534. // load the default set of analysis fields
  535. $additions = chado_analysis_load($node);
  536. // create some variables for easier lookup
  537. $analysis = $additions->analysis;
  538. $analysis_id = $analysis->analysis_id;
  539. $blast_settings = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_settings');
  540. $blastdb = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_blastdb');
  541. $blastfile = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_blastfile');
  542. $blastparameters = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_blastparameters');
  543. $no_parsed = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_no_parsed');
  544. $query_re = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_query_re');
  545. $query_type = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_query_type');
  546. $query_uniquename= tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_query_uniquename');
  547. $blastfile_ext = tripal_analysis_get_property($analysis->analysis_id,'analysis_blast_blastfile_ext');
  548. $analysis->tripal_analysis_blast->blastdb = $blastdb->value;
  549. $analysis->tripal_analysis_blast->blastfile = $blastfile->value;
  550. $analysis->tripal_analysis_blast->blastparameters = $blastparameters->value;
  551. $analysis->tripal_analysis_blast->no_parsed = $no_parsed->value;
  552. $analysis->tripal_analysis_blast->query_re = $query_re->value;
  553. $analysis->tripal_analysis_blast->query_type = $query_type->value;
  554. $analysis->tripal_analysis_blast->query_uniquename= $query_uniquename->value;
  555. $analysis->tripal_analysis_blast->blastfile_ext = $blastfile_ext->value;
  556. // get the database information so that we don't have to require callers
  557. // to do the lookup
  558. $select = array('db_id' => $blastdb->value);
  559. $analysis->tripal_analysis_blast->db = tripal_core_generate_chado_var('db',$select);
  560. // if there is an old style 'blast_settings' array, then break these out for
  561. // use in the new format
  562. if(count($blast_settings)>0){
  563. $prop_values = explode ("|", $blast_settings->value);
  564. $analysis->tripal_analysis_blast->blastdb = $prop_values[0];
  565. $analysis->tripal_analysis_blast->blastfile = $prop_values[1];
  566. $analysis->tripal_analysis_blast->blastparameters = $prop_values[2];
  567. }
  568. return $additions;
  569. }
  570. /**
  571. *
  572. */
  573. function chado_analysis_blast_insert($node){
  574. // insert the analysis
  575. chado_analysis_insert($node);
  576. // set the type for this analysis
  577. tripal_analysis_insert_property($node->analysis->analysis_id,'analysis_type','tripal_analysis_blast');
  578. // now add in the remaining settings as a single property but separated by bars
  579. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_blastdb',$node->blastdb);
  580. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_blastfile',$node->blastfile);
  581. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_blastparameters',$node->blastparameters);
  582. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_no_parsed',$node->no_parsed);
  583. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_query_re',$node->query_re);
  584. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_query_type',$node->query_type);
  585. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_query_uniquename',$node->query_uniquename);
  586. tripal_analysis_insert_property($node->analysis_id,'analysis_blast_blastfile_ext',$node->blastfile_ext);
  587. // submit the parsing jobs
  588. chado_analysis_blast_submit_jobs($node);
  589. }
  590. /**
  591. *
  592. */
  593. function chado_analysis_blast_update($node){
  594. // update the anlaysis
  595. chado_analysis_update($node);
  596. // add the blast settings
  597. tripal_analysis_update_property($node->analysis_id,'analysis_type','tripal_analysis_blast',1);
  598. tripal_analysis_update_property($node->analysis_id,'analysis_blast_blastdb',$node->blastdb,1);
  599. tripal_analysis_update_property($node->analysis_id,'analysis_blast_blastfile',$node->blastfile,1);
  600. tripal_analysis_update_property($node->analysis_id,'analysis_blast_blastparameters',$node->blastparameters,1);
  601. tripal_analysis_update_property($node->analysis_id,'analysis_blast_no_parsed',$node->no_parsed,1);
  602. tripal_analysis_update_property($node->analysis_id,'analysis_blast_query_re',$node->query_re,1);
  603. tripal_analysis_update_property($node->analysis_id,'analysis_blast_query_type',$node->query_type,1);
  604. tripal_analysis_update_property($node->analysis_id,'analysis_blast_query_uniquename',$node->query_uniquename,1);
  605. tripal_analysis_update_property($node->analysis_id,'analysis_blast_blastfile_ext',$node->blastfile_ext,1);
  606. // if this analysis uses the old style blast settings cvterm then remove that term
  607. $old = tripal_analysis_get_property($node->analysis_id,'analysis_blast_settings');
  608. if(count($old) > 0){
  609. tripal_analysis_delete_property($node->analysis_id,'analysis_blast_settings');
  610. }
  611. // submit the parsing jobs
  612. chado_analysis_blast_submit_jobs($node);
  613. }
  614. /**
  615. *
  616. */
  617. function chado_analysis_blast_submit_jobs($node){
  618. global $user;
  619. // add a job if the user wants to parse the XML
  620. if($node->blastjob) {
  621. $job_args = array($analysis_id,
  622. $node->blastdb,
  623. $node->blastfile,
  624. $node->no_parsed,
  625. $node->blastfile_ext,
  626. $node->query_re,
  627. $node->query_type,
  628. $node->query_uniquename
  629. );
  630. if (is_readable($node->blastfile)) {
  631. tripal_add_job("Parse blast: $node->blastfile",'tripal_analysis_blast',
  632. 'tripal_analysis_blast_parseXMLFile', $job_args, $user->uid);
  633. } else {
  634. drupal_set_message("Can not open blast output file. Job not scheduled.");
  635. }
  636. }
  637. // add a job if the user wants to create a best hits report.
  638. if($node->blastbesthit) {
  639. $j_args[0] = $analysis_id;
  640. tripal_add_job("Parse best hit: $node->blastfile",'tripal_analysis_blast',
  641. 'tripal_analysis_blast_parse_best_hit', $j_args, $user->uid);
  642. }
  643. }
  644. /*******************************************************************************
  645. * Delete blast anlysis
  646. */
  647. function chado_analysis_blast_delete($node){
  648. // Before removing, get analysis_id so we can remove it from chado database
  649. // later
  650. $sql_drupal = "SELECT analysis_id ".
  651. "FROM {chado_analysis} ".
  652. "WHERE nid = %d ".
  653. "AND vid = %d";
  654. $analysis_id = db_result(db_query($sql_drupal, $node->nid, $node->vid));
  655. // Remove data from the {chado_analysis}, {node}, and {node_revisions} tables
  656. $sql_del = "DELETE FROM {chado_analysis} ".
  657. "WHERE nid = %d ".
  658. "AND vid = %d";
  659. db_query($sql_del, $node->nid, $node->vid);
  660. $sql_del = "DELETE FROM {node} ".
  661. "WHERE nid = %d ".
  662. "AND vid = %d";
  663. db_query($sql_del, $node->nid, $node->vid);
  664. $sql_del = "DELETE FROM {node_revisions} ".
  665. "WHERE nid = %d ".
  666. "AND vid = %d";
  667. db_query($sql_del, $node->nid, $node->vid);
  668. //Remove from analysisfeatureprop, analysisfeature, analysis, and analysisprop tables
  669. $previous_db = tripal_db_set_active('chado');
  670. $sql = "SELECT analysisfeature_id FROM {analysisfeature} WHERE analysis_id=%d";
  671. $results = db_query($sql, $analysis_id);
  672. while ($af = db_fetch_object($results)) {
  673. db_query("DELETE FROM {analysisfeatureprop} WHERE analysisfeature_id = %d", $af->analysisfeature_id);
  674. }
  675. db_query("DELETE FROM {analysisfeature} WHERE analysis_id = %d", $analysis_id);
  676. db_query("DELETE FROM {analysisprop} WHERE analysis_id = %d", $analysis_id);
  677. db_query("DELETE FROM {analysis} WHERE analysis_id = %d", $analysis_id);
  678. tripal_db_set_active($previous_db);
  679. }
  680. /*******************************************************************************
  681. * This function customizes the view of the chado_analysis node. It allows
  682. * us to generate the markup.
  683. */
  684. function chado_analysis_blast_view ($node, $teaser = FALSE, $page = FALSE) {
  685. // use drupal's default node view:
  686. //dprint_r($node);
  687. if (!$teaser) {
  688. $node = node_prepare($node, $teaser);
  689. // When previewing a node submitting form, it shows 'Array' instead of
  690. // correct date format. We need to format the date here
  691. $time = $node->timeexecuted;
  692. if(is_array($time)){
  693. $month = $time['month'];
  694. $day = $time['day'];
  695. $year = $time['year'];
  696. $timestamp = $year.'-'.$month.'-'.$day;
  697. $node->timeexecuted = $timestamp;
  698. }
  699. // When viewing a node, we need to reformat the analysisprop since we
  700. // separate each value with a bar |
  701. if (preg_match("/.*\|.*\|.*/",$node->blastdb)) {
  702. $prop_values = explode("|", $node->blastdb);
  703. $node->blastdb = $prop_values[0];
  704. $node->blastfile = $prop_values[1];
  705. $node->blastparameters = $prop_values[2];
  706. }
  707. }
  708. return $node;
  709. }
  710. /*******************************************************************************
  711. * Set the permission types that the chado module uses. Essentially we
  712. * want permissionis that protect creation, editing and deleting of chado
  713. * data objects
  714. */
  715. function tripal_analysis_blast_perm(){
  716. return array(
  717. 'access chado_analysis_blast content',
  718. 'create chado_analysis_blast content',
  719. 'delete chado_analysis_blast content',
  720. 'edit chado_analysis_blast content',
  721. );
  722. }
  723. /*******************************************************************************
  724. * The following function proves access control for users trying to
  725. * perform actions on data managed by this module
  726. */
  727. function chado_analysis_blast_access($op, $node, $account){
  728. if ($op == 'create') {
  729. return user_access('create chado_analysis_blast content', $account);
  730. }
  731. if ($op == 'update') {
  732. if (user_access('edit chado_analysis_blast content', $account)) {
  733. return TRUE;
  734. }
  735. }
  736. if ($op == 'delete') {
  737. if (user_access('delete chado_analysis_blast content', $account)) {
  738. return TRUE;
  739. }
  740. }
  741. if ($op == 'view') {
  742. if (user_access('access chado_analysis_blast content', $account)) {
  743. return TRUE;
  744. }
  745. }
  746. return FALSE;
  747. }