-package jalview.analysis;\r
-\r
-import java.util.Enumeration;\r
-import java.util.Vector;\r
-import java.util.Hashtable;\r
-\r
-import jalview.datamodel.AlignedCodonFrame;\r
-import jalview.datamodel.Alignment;\r
-import jalview.datamodel.AlignmentI;\r
-import jalview.datamodel.DBRefSource;\r
-import jalview.datamodel.DBRefEntry;\r
-import jalview.datamodel.Sequence;\r
-import jalview.datamodel.SequenceI;\r
-import jalview.ws.SequenceFetcher;\r
-import jalview.ws.seqfetcher.ASequenceFetcher;\r
-\r
-/**\r
- * Functions for cross-referencing sequence databases. user must first specify\r
- * if cross-referencing from protein or dna (set dna==true)\r
- * \r
- * @author JimP\r
- * \r
- */\r
-public class CrossRef\r
-{\r
- /**\r
- * get the DNA or protein references for a protein or dna sequence\r
- * \r
- * @param dna\r
- * @param rfs\r
- * @return\r
- */\r
- public static DBRefEntry[] findXDbRefs(boolean dna, DBRefEntry[] rfs)\r
- {\r
- if (dna)\r
- {\r
- rfs = jalview.util.DBRefUtils.selectRefs(rfs, DBRefSource.PROTEINDBS);\r
- }\r
- else\r
- {\r
- rfs = jalview.util.DBRefUtils.selectRefs(rfs,\r
- DBRefSource.DNACODINGDBS); // could attempt to find other cross\r
- // refs and return here - ie PDB xrefs\r
- // (not dna, not protein seq)\r
- }\r
- return rfs;\r
- }\r
-\r
- public static Hashtable classifyDbRefs(DBRefEntry[] rfs)\r
- {\r
- Hashtable classes = new Hashtable();\r
- classes.put(DBRefSource.PROTEINDBS, jalview.util.DBRefUtils.selectRefs(\r
- rfs, DBRefSource.PROTEINDBS));\r
- classes.put(DBRefSource.DNACODINGDBS, jalview.util.DBRefUtils\r
- .selectRefs(rfs, DBRefSource.DNACODINGDBS));\r
- classes.put(DBRefSource.DOMAINDBS, jalview.util.DBRefUtils.selectRefs(\r
- rfs, DBRefSource.DOMAINDBS));\r
- // classes.put(OTHER, )\r
- return classes;\r
- }\r
-\r
- /**\r
- * @param dna\r
- * true if seqs are DNA seqs\r
- * @param seqs\r
- * @return a list of sequence database cross reference source types\r
- */\r
- public static String[] findSequenceXrefTypes(boolean dna, SequenceI[] seqs)\r
- {\r
- return findSequenceXrefTypes(dna, seqs, null);\r
- }\r
-\r
- /**\r
- * Indirect references are references from other sequences from the dataset to\r
- * any of the direct DBRefEntrys on the given sequences.\r
- * \r
- * @param dna\r
- * true if seqs are DNA seqs\r
- * @param seqs\r
- * @return a list of sequence database cross reference source types\r
- */\r
- public static String[] findSequenceXrefTypes(boolean dna,\r
- SequenceI[] seqs, AlignmentI dataset)\r
- {\r
- String[] dbrefs = null;\r
- Vector refs = new Vector();\r
- for (int s = 0; s < seqs.length; s++)\r
- {\r
- SequenceI dss = seqs[s];\r
- while (dss.getDatasetSequence() != null)\r
- {\r
- dss = dss.getDatasetSequence();\r
- }\r
- DBRefEntry[] rfs = findXDbRefs(dna, dss.getDBRef());\r
- for (int r = 0; rfs != null && r < rfs.length; r++)\r
- {\r
- if (!refs.contains(rfs[r].getSource()))\r
- {\r
- refs.addElement(rfs[r].getSource());\r
- }\r
- }\r
- if (dataset != null)\r
- {\r
- // search for references to this sequence's direct references.\r
- DBRefEntry[] lrfs = CrossRef.findXDbRefs(!dna, seqs[s].getDBRef());\r
- Vector rseqs = new Vector();\r
- CrossRef.searchDatasetXrefs(seqs[s], !dna, lrfs, dataset, rseqs,\r
- null); // don't need to specify codon frame for mapping here\r
- Enumeration lr = rseqs.elements();\r
- while (lr.hasMoreElements())\r
- {\r
- SequenceI rs = (SequenceI) lr.nextElement();\r
- DBRefEntry[] xrs = findXDbRefs(dna, rs.getDBRef());\r
- for (int r = 0; rfs != null && r < rfs.length; r++)\r
- {\r
- if (!refs.contains(rfs[r].getSource()))\r
- {\r
- refs.addElement(rfs[r].getSource());\r
- }\r
- }\r
- }\r
- }\r
- }\r
- if (refs.size() > 0)\r
- {\r
- dbrefs = new String[refs.size()];\r
- refs.copyInto(dbrefs);\r
- }\r
- return dbrefs;\r
- }\r
-\r
- /*\r
- * if (dna) { if (rfs[r].hasMap()) { // most likely this is a protein cross\r
- * reference if (!refs.contains(rfs[r].getSource())) {\r
- * refs.addElement(rfs[r].getSource()); } } }\r
- */\r
- public static boolean hasCdnaMap(SequenceI[] seqs)\r
- {\r
- String[] reftypes = findSequenceXrefTypes(false, seqs);\r
- for (int s = 0; s < reftypes.length; s++)\r
- {\r
- if (reftypes.equals(DBRefSource.EMBLCDS))\r
- {\r
- return true;\r
- // no map\r
- }\r
- }\r
- return false;\r
- }\r
-\r
- public static SequenceI[] getCdnaMap(SequenceI[] seqs)\r
- {\r
- Vector cseqs = new Vector();\r
- for (int s = 0; s < seqs.length; s++)\r
- {\r
- DBRefEntry[] cdna = findXDbRefs(true, seqs[s].getDBRef());\r
- for (int c = 0; c < cdna.length; c++)\r
- {\r
- if (cdna[c].getSource().equals(DBRefSource.EMBLCDS))\r
- {\r
- System.err.println("TODO: unimplemented sequence retrieval for coding region sequence.");\r
- // TODO: retrieve CDS dataset sequences\r
- // need global dataset sequence retriever/resolver to reuse refs\r
- // and construct Mapping entry.\r
- // insert gaps in CDS according to peptide gaps.\r
- // add gapped sequence to cseqs\r
- }\r
- }\r
- }\r
- if (cseqs.size() > 0)\r
- {\r
- SequenceI[] rsqs = new SequenceI[cseqs.size()];\r
- cseqs.copyInto(rsqs);\r
- return rsqs;\r
- }\r
- return null;\r
-\r
- }\r
-\r
- /**\r
- * \r
- * @param dna\r
- * @param seqs\r
- * @return\r
- */\r
- public static Alignment findXrefSequences(SequenceI[] seqs, boolean dna,\r
- String source)\r
- {\r
- return findXrefSequences(seqs, dna, source, null);\r
- }\r
-\r
- /**\r
- * \r
- * @param seqs\r
- * @param dna\r
- * @param source\r
- * @param dataset\r
- * alignment to search for product sequences.\r
- * @return products (as dataset sequences)\r
- */\r
- public static Alignment findXrefSequences(SequenceI[] seqs, boolean dna,\r
- String source, AlignmentI dataset)\r
- {\r
- Vector rseqs = new Vector();\r
- Alignment ral = null;\r
- AlignedCodonFrame cf = new AlignedCodonFrame(0); // nominal width\r
- for (int s = 0; s < seqs.length; s++)\r
- {\r
- SequenceI dss = seqs[s];\r
- while (dss.getDatasetSequence() != null)\r
- {\r
- dss = dss.getDatasetSequence();\r
- }\r
- boolean found = false;\r
- DBRefEntry[] xrfs = CrossRef.findXDbRefs(dna, dss.getDBRef());\r
- if ((xrfs == null || xrfs.length == 0) && dataset != null)\r
- {\r
- System.out.println("Attempting to find ds Xrefs refs.");\r
- DBRefEntry[] lrfs = CrossRef.findXDbRefs(!dna, seqs[s].getDBRef()); // less\r
- // ambiguous\r
- // would\r
- // be a\r
- // 'find\r
- // primary\r
- // dbRefEntry'\r
- // method.\r
- // filter for desired source xref here\r
- found = CrossRef.searchDatasetXrefs(dss, !dna, lrfs, dataset,\r
- rseqs, cf);\r
- }\r
- for (int r = 0; xrfs != null && r < xrfs.length; r++)\r
- {\r
- if (source != null && !source.equals(xrfs[r].getSource()))\r
- continue;\r
- if (xrfs[r].hasMap())\r
- {\r
- if (xrfs[r].getMap().getTo() != null)\r
- {\r
- Sequence rsq = new Sequence(xrfs[r].getMap().getTo());\r
- rseqs.addElement(rsq);\r
- if (xrfs[r].getMap().getMap().getFromRatio() != xrfs[r]\r
- .getMap().getMap().getToRatio())\r
- {\r
- // get sense of map correct for adding to product alignment.\r
- if (dna)\r
- {\r
- // map is from dna seq to a protein product\r
- cf.addMap(dss, rsq, xrfs[r].getMap().getMap());\r
- }\r
- else\r
- {\r
- // map should be from protein seq to its coding dna\r
- cf.addMap(rsq, dss, xrfs[r].getMap().getMap().getInverse());\r
- }\r
- }\r
- found = true;\r
- }\r
- }\r
- if (!found)\r
- {\r
- // do a bit more work - search for sequences with references matching\r
- // xrefs on this sequence.\r
- if (dataset != null)\r
- {\r
- found |= searchDataset(dss, xrfs[r], dataset, rseqs, cf);\r
- if (found)\r
- xrfs[r] = null; // we've recovered seqs for this one.\r
- }\r
- }\r
- }\r
- if (!found)\r
- {\r
- if (xrfs != null && xrfs.length > 0)\r
- {\r
- // Try and get the sequence reference...\r
- /*\r
- * Ideal world - we ask for a sequence fetcher implementation here if\r
- * (jalview.io.RunTimeEnvironment.getSequenceFetcher()) (\r
- */\r
- ASequenceFetcher sftch = new SequenceFetcher();\r
- SequenceI[] retrieved = null;\r
- int l = xrfs.length;\r
- for (int r = 0; r < xrfs.length; r++)\r
- {\r
- // filter out any irrelevant or irretrievable references\r
- if (xrfs[r] == null\r
- || ((source != null && !source.equals(xrfs[r]\r
- .getSource())) || !sftch.isFetchable(xrfs[r]\r
- .getSource())))\r
- {\r
- l--;\r
- xrfs[r] = null;\r
- }\r
- }\r
- if (l > 0)\r
- {\r
- System.out\r
- .println("Attempting to retrieve cross referenced sequences.");\r
- DBRefEntry[] t = new DBRefEntry[l];\r
- l = 0;\r
- for (int r = 0; r < xrfs.length; r++)\r
- {\r
- if (xrfs[r] != null)\r
- t[l++] = xrfs[r];\r
- }\r
- xrfs = t;\r
- try\r
- {\r
- retrieved = sftch.getSequences(xrfs); // problem here is we don't know which of xrfs resulted in which retrieved element\r
- } catch (Exception e)\r
- {\r
- System.err\r
- .println("Problem whilst retrieving cross references for Sequence : "\r
- + seqs[s].getName());\r
- e.printStackTrace();\r
- }\r
- if (retrieved != null)\r
- {\r
- for (int rs = 0; rs < retrieved.length; rs++)\r
- {\r
- // TODO: examine each sequence for 'redundancy'\r
- jalview.datamodel.DBRefEntry[] dbr = retrieved[rs].getDBRef();\r
- if (dbr != null && dbr.length > 0)\r
- {\r
- for (int di = 0; di < dbr.length; di++)\r
- {\r
- // find any entry where we should put in the sequence being cross-referenced into the map\r
- jalview.datamodel.Mapping map = dbr[di].getMap();\r
- if (map != null)\r
- {\r
- if (map.getTo() != null && map.getMap() != null)\r
- {\r
- // should search the local dataset to find any existing candidates for To !\r
- try\r
- {\r
- // compare ms with dss and replace with dss in mapping if map is congruent\r
- SequenceI ms = map.getTo();\r
- int sf = map.getMap().getToLowest();\r
- int st = map.getMap().getToHighest();\r
- SequenceI mappedrg = ms.getSubSequence(sf, st);\r
- SequenceI loc = dss.getSubSequence(sf, st);\r
- if (mappedrg.getLength()>0 && mappedrg.getSequenceAsString().equals(\r
- loc.getSequenceAsString()))\r
- {\r
- System.err\r
- .println("Mapping updated for retrieved crossreference");\r
- // method to update all refs of existing To on retrieved sequence with dss and merge any props on To onto dss.\r
- map.setTo(dss);\r
- }\r
- } catch (Exception e)\r
- {\r
- System.err\r
- .println("Exception when consolidating Mapped sequence set...");\r
- e.printStackTrace(System.err);\r
- }\r
- }\r
- }\r
- }\r
- }\r
- retrieved[rs].updatePDBIds();\r
- rseqs.addElement(retrieved[rs]);\r
- }\r
- }\r
- }\r
- }\r
- }\r
- }\r
- if (rseqs.size() > 0)\r
- {\r
- SequenceI[] rsqs = new SequenceI[rseqs.size()];\r
- rseqs.copyInto(rsqs);\r
- ral = new Alignment(rsqs);\r
- if (cf != null && cf.getProtMappings() != null)\r
- {\r
- ral.addCodonFrame(cf);\r
- }\r
- }\r
- return ral;\r
- }\r
-\r
- /**\r
- * find references to lrfs in the cross-reference set of each sequence in\r
- * dataset (that is not equal to sequenceI) Identifies matching DBRefEntry\r
- * based on source and accession string only - Map and Version are nulled.\r
- * \r
- * @param sequenceI\r
- * @param lrfs\r
- * @param dataset\r
- * @param rseqs\r
- * @return true if matches were found.\r
- */\r
- private static boolean searchDatasetXrefs(SequenceI sequenceI,\r
- boolean dna, DBRefEntry[] lrfs, AlignmentI dataset, Vector rseqs,\r
- AlignedCodonFrame cf)\r
- {\r
- boolean found = false;\r
- if (lrfs == null)\r
- return false;\r
- for (int i = 0; i < lrfs.length; i++)\r
- {\r
- DBRefEntry xref = new DBRefEntry(lrfs[i]);\r
- // add in wildcards\r
- xref.setVersion(null);\r
- xref.setMap(null);\r
- found = searchDataset(sequenceI, xref, dataset, rseqs, cf, false, dna);\r
- }\r
- return found;\r
- }\r
-\r
- /**\r
- * search a given sequence dataset for references matching cross-references to\r
- * the given sequence\r
- * \r
- * @param sequenceI\r
- * @param xrf\r
- * @param dataset\r
- * @param rseqs\r
- * set of unique sequences\r
- * @param cf\r
- * @return true if one or more unique sequences were found and added\r
- */\r
- public static boolean searchDataset(SequenceI sequenceI, DBRefEntry xrf,\r
- AlignmentI dataset, Vector rseqs, AlignedCodonFrame cf)\r
- {\r
- return searchDataset(sequenceI, xrf, dataset, rseqs, cf, true, false);\r
- }\r
-\r
- /**\r
- * TODO: generalise to different protein classifications Search dataset for\r
- * DBRefEntrys matching the given one (xrf) and add the associated sequence to\r
- * rseq.\r
- * \r
- * @param sequenceI\r
- * @param xrf\r
- * @param dataset\r
- * @param rseqs\r
- * @param direct -\r
- * search all references or only subset\r
- * @param dna\r
- * search dna or protein xrefs (if direct=false)\r
- * @return true if relationship found and sequence added.\r
- */\r
- public static boolean searchDataset(SequenceI sequenceI, DBRefEntry xrf,\r
- AlignmentI dataset, Vector rseqs, AlignedCodonFrame cf,\r
- boolean direct, boolean dna)\r
- {\r
- boolean found = false;\r
- SequenceI[] typer=new SequenceI[1];\r
- if (dataset == null)\r
- return false;\r
- if (dataset.getSequences() == null)\r
- {\r
- System.err.println("Empty dataset sequence set - NO VECTOR");\r
- return false;\r
- }\r
- Enumeration e = dataset.getSequences().elements();\r
- while (e.hasMoreElements())\r
- {\r
- SequenceI nxt = (SequenceI) e.nextElement();\r
- if (nxt != null)\r
- {\r
- if (nxt.getDatasetSequence() != null)\r
- {\r
- System.err\r
- .println("Implementation warning: getProducts passed a dataset alignment without dataset sequences in it!");\r
- }\r
- if (nxt != sequenceI && nxt != sequenceI.getDatasetSequence())\r
- {\r
- // check if this is the correct sequence type\r
- {\r
- typer[0] = nxt;\r
- boolean isDna = jalview.util.Comparison.isNucleotide(typer);\r
- if ((direct && isDna == dna) || (!direct && isDna!=dna))\r
- {\r
- // skip this sequence because it is same molecule type\r
- continue;\r
- }\r
- }\r
-\r
- // look for direct or indirect references in common\r
- DBRefEntry[] poss = null, cands = null;\r
- if (direct)\r
- {\r
- cands = jalview.util.DBRefUtils.searchRefs(poss = nxt\r
- .getDBRef(), xrf);\r
- }\r
- else\r
- {\r
- cands = jalview.util.DBRefUtils.searchRefs(poss = CrossRef\r
- .findXDbRefs(dna, nxt.getDBRef()), xrf);\r
- }\r
- if (cands != null)\r
- {\r
- if (!rseqs.contains(nxt))\r
- {\r
- rseqs.addElement(nxt);\r
- boolean foundmap = cf != null; // don't search if we aren't given\r
- // a codon map object\r
- for (int r = 0; foundmap && r < cands.length; r++)\r
- {\r
- if (cands[r].hasMap())\r
- {\r
- if (cands[r].getMap().getTo() != null\r
- && cands[r].getMap().getMap().getFromRatio() != cands[r]\r
- .getMap().getMap().getToRatio())\r
- {\r
- foundmap = true;\r
- // get sense of map correct for adding to product alignment.\r
- if (dna)\r
- {\r
- // map is from dna seq to a protein product\r
- cf.addMap(sequenceI, nxt, cands[r].getMap().getMap());\r
- }\r
- else\r
- {\r
- // map should be from protein seq to its coding dna\r
- cf.addMap(nxt, sequenceI, cands[r].getMap().getMap()\r
- .getInverse());\r
- }\r
- }\r
- }\r
- }\r
- // TODO: add mapping between sequences if necessary\r
- found = true;\r
- }\r
- }\r
-\r
- }\r
- }\r
- }\r
- return found;\r
- }\r
-\r
- /**\r
- * precalculate different products that can be found for seqs in dataset and\r
- * return them.\r
- * \r
- * @param dna\r
- * @param seqs\r
- * @param dataset\r
- * @param fake -\r
- * don't actually build lists - just get types\r
- * @return public static Object[] buildXProductsList(boolean dna, SequenceI[]\r
- * seqs, AlignmentI dataset, boolean fake) { String types[] =\r
- * jalview.analysis.CrossRef.findSequenceXrefTypes( dna, seqs,\r
- * dataset); if (types != null) { System.out.println("Xref Types for:\r
- * "+(dna ? "dna" : "prot")); for (int t = 0; t < types.length; t++) {\r
- * System.out.println("Type: " + types[t]); SequenceI[] prod =\r
- * jalview.analysis.CrossRef.findXrefSequences(seqs, dna, types[t]);\r
- * System.out.println("Found " + ((prod == null) ? "no" : "" +\r
- * prod.length) + " products"); if (prod!=null) { for (int p=0; p<prod.length;\r
- * p++) { System.out.println("Prod "+p+":\r
- * "+prod[p].getDisplayId(true)); } } }\r
- * } else { System.out.println("Trying getProducts for\r
- * "+al.getSequenceAt(0).getDisplayId(true)); System.out.println("Search DS\r
- * Xref for: "+(dna ? "dna" : "prot")); // have a bash at finding the products\r
- * amongst all the retrieved sequences. SequenceI[] prod =\r
- * jalview.analysis.CrossRef.findXrefSequences(al .getSequencesArray(), dna,\r
- * null, ds); System.out.println("Found " + ((prod == null) ? "no" : "" +\r
- * prod.length) + " products"); if (prod!=null) { // select non-equivalent\r
- * sequences from dataset list for (int p=0; p<prod.length; p++) {\r
- * System.out.println("Prod "+p+": "+prod[p].getDisplayId(true)); } }\r
- * } }\r
- */\r
-}
\ No newline at end of file
+/*
+ * Jalview - A Sequence Alignment Editor and Viewer (Version 2.8.2)
+ * Copyright (C) 2014 The Jalview Authors
+ *
+ * This file is part of Jalview.
+ *
+ * Jalview is free software: you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version.
+ *
+ * Jalview is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty
+ * of MERCHANTABILITY or FITNESS FOR A PARTICULAR
+ * PURPOSE. See the GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along with Jalview. If not, see <http://www.gnu.org/licenses/>.
+ * The Jalview Authors are detailed in the 'AUTHORS' file.
+ */
+package jalview.analysis;
+
+import java.util.Enumeration;
+import java.util.List;
+import java.util.Vector;
+import java.util.Hashtable;
+
+import jalview.datamodel.AlignedCodonFrame;
+import jalview.datamodel.Alignment;
+import jalview.datamodel.AlignmentI;
+import jalview.datamodel.DBRefSource;
+import jalview.datamodel.DBRefEntry;
+import jalview.datamodel.Sequence;
+import jalview.datamodel.SequenceI;
+import jalview.ws.SequenceFetcher;
+import jalview.ws.seqfetcher.ASequenceFetcher;
+
+/**
+ * Functions for cross-referencing sequence databases. user must first specify
+ * if cross-referencing from protein or dna (set dna==true)
+ *
+ * @author JimP
+ *
+ */
+public class CrossRef
+{
+ /**
+ * get the DNA or protein references for a protein or dna sequence
+ *
+ * @param dna
+ * @param rfs
+ * @return
+ */
+ public static DBRefEntry[] findXDbRefs(boolean dna, DBRefEntry[] rfs)
+ {
+ if (dna)
+ {
+ rfs = jalview.util.DBRefUtils.selectRefs(rfs, DBRefSource.PROTEINDBS);
+ }
+ else
+ {
+ rfs = jalview.util.DBRefUtils.selectRefs(rfs,
+ DBRefSource.DNACODINGDBS); // could attempt to find other cross
+ // refs and return here - ie PDB xrefs
+ // (not dna, not protein seq)
+ }
+ return rfs;
+ }
+
+ public static Hashtable classifyDbRefs(DBRefEntry[] rfs)
+ {
+ Hashtable classes = new Hashtable();
+ classes.put(DBRefSource.PROTEINDBS,
+ jalview.util.DBRefUtils.selectRefs(rfs, DBRefSource.PROTEINDBS));
+ classes.put(DBRefSource.DNACODINGDBS, jalview.util.DBRefUtils
+ .selectRefs(rfs, DBRefSource.DNACODINGDBS));
+ classes.put(DBRefSource.DOMAINDBS,
+ jalview.util.DBRefUtils.selectRefs(rfs, DBRefSource.DOMAINDBS));
+ // classes.put(OTHER, )
+ return classes;
+ }
+
+ /**
+ * @param dna
+ * true if seqs are DNA seqs
+ * @param seqs
+ * @return a list of sequence database cross reference source types
+ */
+ public static String[] findSequenceXrefTypes(boolean dna, SequenceI[] seqs)
+ {
+ return findSequenceXrefTypes(dna, seqs, null);
+ }
+
+ /**
+ * Indirect references are references from other sequences from the dataset to
+ * any of the direct DBRefEntrys on the given sequences.
+ *
+ * @param dna
+ * true if seqs are DNA seqs
+ * @param seqs
+ * @return a list of sequence database cross reference source types
+ */
+ public static String[] findSequenceXrefTypes(boolean dna,
+ SequenceI[] seqs, AlignmentI dataset)
+ {
+ String[] dbrefs = null;
+ Vector refs = new Vector();
+ for (int s = 0; s < seqs.length; s++)
+ {
+ if (seqs[s] != null)
+ {
+
+ SequenceI dss = seqs[s];
+ while (dss.getDatasetSequence() != null)
+ {
+ dss = dss.getDatasetSequence();
+ }
+ DBRefEntry[] rfs = findXDbRefs(dna, dss.getDBRef());
+ for (int r = 0; rfs != null && r < rfs.length; r++)
+ {
+ if (!refs.contains(rfs[r].getSource()))
+ {
+ refs.addElement(rfs[r].getSource());
+ }
+ }
+ if (dataset != null)
+ {
+ // search for references to this sequence's direct references.
+ DBRefEntry[] lrfs = CrossRef
+ .findXDbRefs(!dna, seqs[s].getDBRef());
+ Vector rseqs = new Vector();
+ CrossRef.searchDatasetXrefs(seqs[s], !dna, lrfs, dataset, rseqs,
+ null); // don't need to specify codon frame for mapping here
+ Enumeration lr = rseqs.elements();
+ while (lr.hasMoreElements())
+ {
+ SequenceI rs = (SequenceI) lr.nextElement();
+ DBRefEntry[] xrs = findXDbRefs(dna, rs.getDBRef());
+ for (int r = 0; rfs != null && r < rfs.length; r++)
+ {
+ if (!refs.contains(rfs[r].getSource()))
+ {
+ refs.addElement(rfs[r].getSource());
+ }
+ }
+ }
+ }
+ }
+ }
+ if (refs.size() > 0)
+ {
+ dbrefs = new String[refs.size()];
+ refs.copyInto(dbrefs);
+ }
+ return dbrefs;
+ }
+
+ /*
+ * if (dna) { if (rfs[r].hasMap()) { // most likely this is a protein cross
+ * reference if (!refs.contains(rfs[r].getSource())) {
+ * refs.addElement(rfs[r].getSource()); } } }
+ */
+ public static boolean hasCdnaMap(SequenceI[] seqs)
+ {
+ String[] reftypes = findSequenceXrefTypes(false, seqs);
+ for (int s = 0; s < reftypes.length; s++)
+ {
+ if (reftypes.equals(DBRefSource.EMBLCDS))
+ {
+ return true;
+ // no map
+ }
+ }
+ return false;
+ }
+
+ public static SequenceI[] getCdnaMap(SequenceI[] seqs)
+ {
+ Vector cseqs = new Vector();
+ for (int s = 0; s < seqs.length; s++)
+ {
+ DBRefEntry[] cdna = findXDbRefs(true, seqs[s].getDBRef());
+ for (int c = 0; c < cdna.length; c++)
+ {
+ if (cdna[c].getSource().equals(DBRefSource.EMBLCDS))
+ {
+ System.err
+ .println("TODO: unimplemented sequence retrieval for coding region sequence.");
+ // TODO: retrieve CDS dataset sequences
+ // need global dataset sequence retriever/resolver to reuse refs
+ // and construct Mapping entry.
+ // insert gaps in CDS according to peptide gaps.
+ // add gapped sequence to cseqs
+ }
+ }
+ }
+ if (cseqs.size() > 0)
+ {
+ SequenceI[] rsqs = new SequenceI[cseqs.size()];
+ cseqs.copyInto(rsqs);
+ return rsqs;
+ }
+ return null;
+
+ }
+
+ /**
+ *
+ * @param dna
+ * @param seqs
+ * @return
+ */
+ public static Alignment findXrefSequences(SequenceI[] seqs, boolean dna,
+ String source)
+ {
+ return findXrefSequences(seqs, dna, source, null);
+ }
+
+ /**
+ *
+ * @param seqs
+ * @param dna
+ * @param source
+ * @param dataset
+ * alignment to search for product sequences.
+ * @return products (as dataset sequences)
+ */
+ public static Alignment findXrefSequences(SequenceI[] seqs, boolean dna,
+ String source, AlignmentI dataset)
+ {
+ Vector rseqs = new Vector();
+ Alignment ral = null;
+ AlignedCodonFrame cf = new AlignedCodonFrame(0); // nominal width
+ for (int s = 0; s < seqs.length; s++)
+ {
+ SequenceI dss = seqs[s];
+ while (dss.getDatasetSequence() != null)
+ {
+ dss = dss.getDatasetSequence();
+ }
+ boolean found = false;
+ DBRefEntry[] xrfs = CrossRef.findXDbRefs(dna, dss.getDBRef());
+ if ((xrfs == null || xrfs.length == 0) && dataset != null)
+ {
+ System.out.println("Attempting to find ds Xrefs refs.");
+ DBRefEntry[] lrfs = CrossRef.findXDbRefs(!dna, seqs[s].getDBRef()); // less
+ // ambiguous
+ // would
+ // be a
+ // 'find
+ // primary
+ // dbRefEntry'
+ // method.
+ // filter for desired source xref here
+ found = CrossRef.searchDatasetXrefs(dss, !dna, lrfs, dataset,
+ rseqs, cf);
+ }
+ for (int r = 0; xrfs != null && r < xrfs.length; r++)
+ {
+ if (source != null && !source.equals(xrfs[r].getSource()))
+ continue;
+ if (xrfs[r].hasMap())
+ {
+ if (xrfs[r].getMap().getTo() != null)
+ {
+ Sequence rsq = new Sequence(xrfs[r].getMap().getTo());
+ rseqs.addElement(rsq);
+ if (xrfs[r].getMap().getMap().getFromRatio() != xrfs[r]
+ .getMap().getMap().getToRatio())
+ {
+ // get sense of map correct for adding to product alignment.
+ if (dna)
+ {
+ // map is from dna seq to a protein product
+ cf.addMap(dss, rsq, xrfs[r].getMap().getMap());
+ }
+ else
+ {
+ // map should be from protein seq to its coding dna
+ cf.addMap(rsq, dss, xrfs[r].getMap().getMap().getInverse());
+ }
+ }
+ found = true;
+ }
+ }
+ if (!found)
+ {
+ // do a bit more work - search for sequences with references matching
+ // xrefs on this sequence.
+ if (dataset != null)
+ {
+ found |= searchDataset(dss, xrfs[r], dataset, rseqs, cf); // ,false,!dna);
+ if (found)
+ xrfs[r] = null; // we've recovered seqs for this one.
+ }
+ }
+ }
+ if (!found)
+ {
+ if (xrfs != null && xrfs.length > 0)
+ {
+ // Try and get the sequence reference...
+ /*
+ * Ideal world - we ask for a sequence fetcher implementation here if
+ * (jalview.io.RunTimeEnvironment.getSequenceFetcher()) (
+ */
+ ASequenceFetcher sftch = new SequenceFetcher();
+ SequenceI[] retrieved = null;
+ int l = xrfs.length;
+ for (int r = 0; r < xrfs.length; r++)
+ {
+ // filter out any irrelevant or irretrievable references
+ if (xrfs[r] == null
+ || ((source != null && !source.equals(xrfs[r]
+ .getSource())) || !sftch.isFetchable(xrfs[r]
+ .getSource())))
+ {
+ l--;
+ xrfs[r] = null;
+ }
+ }
+ if (l > 0)
+ {
+ System.out
+ .println("Attempting to retrieve cross referenced sequences.");
+ DBRefEntry[] t = new DBRefEntry[l];
+ l = 0;
+ for (int r = 0; r < xrfs.length; r++)
+ {
+ if (xrfs[r] != null)
+ t[l++] = xrfs[r];
+ }
+ xrfs = t;
+ try
+ {
+ retrieved = sftch.getSequences(xrfs); // problem here is we don't
+ // know which of xrfs
+ // resulted in which
+ // retrieved element
+ } catch (Exception e)
+ {
+ System.err
+ .println("Problem whilst retrieving cross references for Sequence : "
+ + seqs[s].getName());
+ e.printStackTrace();
+ }
+ if (retrieved != null)
+ {
+ for (int rs = 0; rs < retrieved.length; rs++)
+ {
+ // TODO: examine each sequence for 'redundancy'
+ jalview.datamodel.DBRefEntry[] dbr = retrieved[rs]
+ .getDBRef();
+ if (dbr != null && dbr.length > 0)
+ {
+ for (int di = 0; di < dbr.length; di++)
+ {
+ // find any entry where we should put in the sequence being
+ // cross-referenced into the map
+ jalview.datamodel.Mapping map = dbr[di].getMap();
+ if (map != null)
+ {
+ if (map.getTo() != null && map.getMap() != null)
+ {
+ // should search the local dataset to find any existing
+ // candidates for To !
+ try
+ {
+ // compare ms with dss and replace with dss in mapping
+ // if map is congruent
+ SequenceI ms = map.getTo();
+ int sf = map.getMap().getToLowest();
+ int st = map.getMap().getToHighest();
+ SequenceI mappedrg = ms.getSubSequence(sf, st);
+ SequenceI loc = dss.getSubSequence(sf, st);
+ if (mappedrg.getLength() > 0
+ && mappedrg.getSequenceAsString().equals(
+ loc.getSequenceAsString()))
+ {
+ System.err
+ .println("Mapping updated for retrieved crossreference");
+ // method to update all refs of existing To on
+ // retrieved sequence with dss and merge any props
+ // on To onto dss.
+ map.setTo(dss);
+ }
+ } catch (Exception e)
+ {
+ System.err
+ .println("Exception when consolidating Mapped sequence set...");
+ e.printStackTrace(System.err);
+ }
+ }
+ }
+ }
+ }
+ retrieved[rs].updatePDBIds();
+ rseqs.addElement(retrieved[rs]);
+ }
+ }
+ }
+ }
+ }
+ }
+ if (rseqs.size() > 0)
+ {
+ SequenceI[] rsqs = new SequenceI[rseqs.size()];
+ rseqs.copyInto(rsqs);
+ ral = new Alignment(rsqs);
+ if (cf != null && cf.getProtMappings() != null)
+ {
+ ral.addCodonFrame(cf);
+ }
+ }
+ return ral;
+ }
+
+ /**
+ * find references to lrfs in the cross-reference set of each sequence in
+ * dataset (that is not equal to sequenceI) Identifies matching DBRefEntry
+ * based on source and accession string only - Map and Version are nulled.
+ *
+ * @param sequenceI
+ * @param lrfs
+ * @param dataset
+ * @param rseqs
+ * @return true if matches were found.
+ */
+ private static boolean searchDatasetXrefs(SequenceI sequenceI,
+ boolean dna, DBRefEntry[] lrfs, AlignmentI dataset, Vector rseqs,
+ AlignedCodonFrame cf)
+ {
+ boolean found = false;
+ if (lrfs == null)
+ return false;
+ for (int i = 0; i < lrfs.length; i++)
+ {
+ DBRefEntry xref = new DBRefEntry(lrfs[i]);
+ // add in wildcards
+ xref.setVersion(null);
+ xref.setMap(null);
+ found = searchDataset(sequenceI, xref, dataset, rseqs, cf, false, dna);
+ }
+ return found;
+ }
+
+ /**
+ * search a given sequence dataset for references matching cross-references to
+ * the given sequence
+ *
+ * @param sequenceI
+ * @param xrf
+ * @param dataset
+ * @param rseqs
+ * set of unique sequences
+ * @param cf
+ * @return true if one or more unique sequences were found and added
+ */
+ public static boolean searchDataset(SequenceI sequenceI, DBRefEntry xrf,
+ AlignmentI dataset, Vector rseqs, AlignedCodonFrame cf)
+ {
+ return searchDataset(sequenceI, xrf, dataset, rseqs, cf, true, false);
+ }
+
+ /**
+ * TODO: generalise to different protein classifications Search dataset for
+ * DBRefEntrys matching the given one (xrf) and add the associated sequence to
+ * rseq.
+ *
+ * @param sequenceI
+ * @param xrf
+ * @param dataset
+ * @param rseqs
+ * @param direct
+ * - search all references or only subset
+ * @param dna
+ * search dna or protein xrefs (if direct=false)
+ * @return true if relationship found and sequence added.
+ */
+ public static boolean searchDataset(SequenceI sequenceI, DBRefEntry xrf,
+ AlignmentI dataset, Vector rseqs, AlignedCodonFrame cf,
+ boolean direct, boolean dna)
+ {
+ boolean found = false;
+ SequenceI[] typer = new SequenceI[1];
+ if (dataset == null)
+ return false;
+ if (dataset.getSequences() == null)
+ {
+ System.err.println("Empty dataset sequence set - NO VECTOR");
+ return false;
+ }
+ List<SequenceI> ds;
+ synchronized (ds = dataset.getSequences())
+ {
+ for (SequenceI nxt : ds)
+ if (nxt != null)
+ {
+ if (nxt.getDatasetSequence() != null)
+ {
+ System.err
+ .println("Implementation warning: getProducts passed a dataset alignment without dataset sequences in it!");
+ }
+ if (nxt != sequenceI && nxt != sequenceI.getDatasetSequence())
+ {
+ // check if this is the correct sequence type
+ {
+ typer[0] = nxt;
+ boolean isDna = jalview.util.Comparison.isNucleotide(typer);
+ if ((direct && isDna == dna) || (!direct && isDna != dna))
+ {
+ // skip this sequence because it is same molecule type
+ continue;
+ }
+ }
+
+ // look for direct or indirect references in common
+ DBRefEntry[] poss = nxt.getDBRef(), cands = null;
+ if (direct)
+ {
+ cands = jalview.util.DBRefUtils.searchRefs(poss, xrf);
+ }
+ else
+ {
+ poss = CrossRef.findXDbRefs(dna, poss); //
+ cands = jalview.util.DBRefUtils.searchRefs(poss, xrf);
+ }
+ if (cands != null)
+ {
+ if (!rseqs.contains(nxt))
+ {
+ rseqs.addElement(nxt);
+ boolean foundmap = cf != null; // don't search if we aren't
+ // given
+ // a codon map object
+ for (int r = 0; foundmap && r < cands.length; r++)
+ {
+ if (cands[r].hasMap())
+ {
+ if (cands[r].getMap().getTo() != null
+ && cands[r].getMap().getMap().getFromRatio() != cands[r]
+ .getMap().getMap().getToRatio())
+ {
+ foundmap = true;
+ // get sense of map correct for adding to product
+ // alignment.
+ if (dna)
+ {
+ // map is from dna seq to a protein product
+ cf.addMap(sequenceI, nxt, cands[r].getMap()
+ .getMap());
+ }
+ else
+ {
+ // map should be from protein seq to its coding dna
+ cf.addMap(nxt, sequenceI, cands[r].getMap()
+ .getMap().getInverse());
+ }
+ }
+ }
+ }
+ // TODO: add mapping between sequences if necessary
+ found = true;
+ }
+ }
+
+ }
+ }
+ }
+ return found;
+ }
+
+ /**
+ * precalculate different products that can be found for seqs in dataset and
+ * return them.
+ *
+ * @param dna
+ * @param seqs
+ * @param dataset
+ * @param fake
+ * - don't actually build lists - just get types
+ * @return public static Object[] buildXProductsList(boolean dna, SequenceI[]
+ * seqs, AlignmentI dataset, boolean fake) { String types[] =
+ * jalview.analysis.CrossRef.findSequenceXrefTypes( dna, seqs,
+ * dataset); if (types != null) { System.out.println("Xref Types for:
+ * "+(dna ? "dna" : "prot")); for (int t = 0; t < types.length; t++) {
+ * System.out.println("Type: " + types[t]); SequenceI[] prod =
+ * jalview.analysis.CrossRef.findXrefSequences(seqs, dna, types[t]);
+ * System.out.println("Found " + ((prod == null) ? "no" : "" +
+ * prod.length) + " products"); if (prod!=null) { for (int p=0;
+ * p<prod.length; p++) { System.out.println("Prod "+p+":
+ * "+prod[p].getDisplayId(true)); } } } } else {
+ * System.out.println("Trying getProducts for
+ * "+al.getSequenceAt(0).getDisplayId(true));
+ * System.out.println("Search DS Xref for: "+(dna ? "dna" : "prot"));
+ * // have a bash at finding the products amongst all the retrieved
+ * sequences. SequenceI[] prod =
+ * jalview.analysis.CrossRef.findXrefSequences(al
+ * .getSequencesArray(), dna, null, ds); System.out.println("Found " +
+ * ((prod == null) ? "no" : "" + prod.length) + " products"); if
+ * (prod!=null) { // select non-equivalent sequences from dataset list
+ * for (int p=0; p<prod.length; p++) { System.out.println("Prod "+p+":
+ * "+prod[p].getDisplayId(true)); } } } }
+ */
+}