001/*
002 *                    BioJava development code
003 *
004 * This code may be freely distributed and modified under the
005 * terms of the GNU Lesser General Public Licence.  This should
006 * be distributed with the code.  If you do not have a copy,
007 * see:
008 *
009 *      http://www.gnu.org/copyleft/lesser.html
010 *
011 * Copyright for this code is held jointly by the individual
012 * authors.  These should be listed in @author doc comments.
013 *
014 * For more information on the BioJava project and its aims,
015 * or to join the biojava-l mailing list, visit the home page
016 * at:
017 *
018 *      http://www.biojava.org/
019 *
020 */
021
022package org.biojava.nbio.ws.alignment.qblast;
023
024import org.biojava.nbio.core.sequence.io.util.IOUtils;
025import org.biojava.nbio.core.sequence.template.Compound;
026import org.biojava.nbio.core.sequence.template.Sequence;
027import org.biojava.nbio.ws.alignment.RemotePairwiseAlignmentOutputProperties;
028import org.biojava.nbio.ws.alignment.RemotePairwiseAlignmentProperties;
029import org.biojava.nbio.ws.alignment.RemotePairwiseAlignmentService;
030
031import java.io.*;
032import java.net.MalformedURLException;
033import java.net.URL;
034import java.net.URLConnection;
035import java.util.HashMap;
036import java.util.Map;
037
038import static org.biojava.nbio.ws.alignment.qblast.BlastAlignmentParameterEnum.*;
039import static org.biojava.nbio.ws.alignment.qblast.BlastOutputParameterEnum.RID;
040
041/**
042 * Provides a simple way of submitting BLAST request to the QBlast service at NCBI.
043 * <p>
044 * NCBI provides a Blast server through a CGI-BIN interface. This service simply encapsulates an access to it by giving
045 * users access to get/set methods to fix sequence, program and database as well as advanced options.
046 * </p>
047 * <p>
048 * The philosophy behind this service is to disconnect submission of Blast requests from collection of Blast results.
049 * This is done so to allow a user to submit multiple Blast requests while allowing recovery of the reports at a later
050 * time.
051 * </p>
052 * <p>
053 * Presently, only blastall programs are accessible.
054 * </p>
055 *
056 * @author Sylvain Foisy, Diploide BioIT
057 * @author Gediminas Rimsa
058 */
059public class NCBIQBlastService implements RemotePairwiseAlignmentService {
060        /**
061         * Number of milliseconds by which expected job execution time is incremented if it is not finished yet. Subsequent
062         * calls to {@link #isReady(String, long)} method will return false until at least this much time passes.
063         */
064        public static final long WAIT_INCREMENT = 3000;
065
066        private static final MapToStringTransformer MAP_TO_STRING_TRANSFORMER = new MapToStringTransformer();
067        private static final String SERVICE_URL = "https://blast.ncbi.nlm.nih.gov/Blast.cgi";
068        private static final String DEFAULT_EMAIL = "anonymous@biojava.org";
069        private static final String DEFAULT_TOOL = "biojava5";
070
071        private URL serviceUrl;
072        private String email = DEFAULT_EMAIL;
073        private String tool = DEFAULT_TOOL;
074
075        private Map<String, BlastJob> jobs = new HashMap<>();
076
077        /** Constructs a service object that targets the public NCBI BLAST network
078         * service.
079         */
080        public NCBIQBlastService() {
081                init(SERVICE_URL);
082        }
083
084        /** Constructs a service object which targets a custom NCBI BLAST network
085         * service (e.g.: an instance of BLAST in the cloud).
086         *
087         * @param svcUrl : a {@code String} containing the base URL to send requests to,
088         *                 e.g.: http://host.my.cloud.service.provider.com/cgi-bin/blast.cgi
089         *
090         * @see <a href="https://blast.ncbi.nlm.nih.gov/Blast.cgi?PAGE_TYPE=BlastDocs&DOC_TYPE=CloudBlast">BLAST on the cloud documentation</a>
091         */
092        public NCBIQBlastService(String svcUrl) {
093                init(svcUrl);
094        }
095
096        /** Initialize the serviceUrl data member
097         * @throws MalformedURLException on invalid URL
098         */
099        private void init(String svcUrl) {
100                try {
101                        serviceUrl = new URL(svcUrl);
102                } catch (MalformedURLException e) {
103                        throw new RuntimeException("It looks like the URL for remote NCBI BLAST service ("
104                                                                           + svcUrl + ") is wrong. Cause: " + e.getMessage(), e);
105                }
106        }
107
108        /**
109         * A simple method to check the availability of the QBlast service. Sends {@code Info} command to QBlast
110         *
111         * @return QBlast info output concatenated to String
112         * @throws Exception if unable to connect to the NCBI QBlast service
113         */
114        public String getRemoteBlastInfo() throws Exception {
115                OutputStreamWriter writer = null;
116                BufferedReader reader = null;
117                try {
118                        URLConnection serviceConnection = setQBlastServiceProperties(serviceUrl.openConnection());
119                        writer = new OutputStreamWriter(serviceConnection.getOutputStream());
120                        writer.write("CMD=Info");
121                        writer.flush();
122                        reader = new BufferedReader(new InputStreamReader(serviceConnection.getInputStream()));
123                        StringBuilder sb = new StringBuilder();
124                        String line;
125                        while ((line = reader.readLine()) != null) {
126                                sb.append(line);
127                                sb.append(System.getProperty("line.separator"));
128                        }
129                        return sb.toString();
130                } catch (IOException e) {
131                        throw new Exception("Impossible to get info from QBlast service at this time. Cause: " + e.getMessage(), e);
132                } finally {
133                        IOUtils.close(reader);
134                        IOUtils.close(writer);
135                }
136        }
137
138        /**
139         * Converts given sequence to String and calls
140         * {@link #sendAlignmentRequest(String, RemotePairwiseAlignmentProperties)}
141         */
142        @Override
143        public String sendAlignmentRequest(Sequence<Compound> seq, RemotePairwiseAlignmentProperties rpa) throws Exception {
144                return sendAlignmentRequest(seq.getSequenceAsString(), rpa);
145        }
146
147        /**
148         * Converts given GenBank GID to String and calls
149         * {@link #sendAlignmentRequest(String, RemotePairwiseAlignmentProperties)}
150         */
151        public String sendAlignmentRequest(int gid, RemotePairwiseAlignmentProperties rpa) throws Exception {
152                return sendAlignmentRequest(Integer.toString(gid), rpa);
153        }
154
155        /**
156         * Sends the Blast request via the Put command of the CGI-BIN interface. Uses all of the parameters specified in
157         * {@code alignmentProperties} (parameters PROGRAM and DATABASE are required).
158         *
159         * @param query : a {@code String} representing a sequence or Genbank ID
160         * @param alignmentProperties : a {@code RemotePairwiseAlignmentProperties} object representing alignment properties
161         * @return the request id for this sequence, necessary to fetch results after completion
162         * @throws Exception if unable to connect to the NCBI QBlast service or if no sequence or required parameters
163         *             PROGRAM and DATABASE are not set
164         */
165        @Override
166        public String sendAlignmentRequest(String query, RemotePairwiseAlignmentProperties alignmentProperties) throws Exception {
167                Map<String, String> params = new HashMap<>();
168                for (String key : alignmentProperties.getAlignmentOptions()) {
169                        params.put(key, alignmentProperties.getAlignmentOption(key));
170                }
171
172                if (query == null || query.isEmpty()) {
173                        throw new IllegalArgumentException("Impossible to execute QBlast request. The sequence has not been set.");
174                }
175                if (!params.containsKey(PROGRAM.name())) {
176                        throw new IllegalArgumentException("Impossible to execute QBlast request. Parameter PROGRAM has not been set.");
177                }
178                if (!params.containsKey(DATABASE.name())) {
179                        throw new IllegalArgumentException("Impossible to execute QBlast request. Parameter DATABASE has not been set.");
180                }
181
182                params.put(CMD.name(), "Put");
183                params.put(QUERY.name(), query);
184                params.put(TOOL.name(), getTool());
185                params.put(EMAIL.name(), getEmail());
186
187                String putCmd = MAP_TO_STRING_TRANSFORMER.transform(params);
188
189                OutputStreamWriter writer = null;
190                BufferedReader reader = null;
191                try {
192                        URLConnection serviceConnection = setQBlastServiceProperties(serviceUrl.openConnection());
193                        writer = new OutputStreamWriter(serviceConnection.getOutputStream());
194                        writer.write(putCmd);
195                        writer.flush();
196
197                        // Get the response
198                        reader = new BufferedReader(new InputStreamReader(serviceConnection.getInputStream()));
199
200                        // find request id and time of execution
201                        BlastJob job = new BlastJob();
202                        String line;
203                        while ((line = reader.readLine()) != null) {
204                                if (!line.contains("class=\"error\"") && !line.contains("Message ID#")) {
205                                        // if there is no error, capture RID and RTOE
206                                        if (line.contains("RID = ")) {
207                                                String[] arr = line.split("=");
208                                                job.setId(arr[1].trim());
209                                        } else if (line.contains("RTOE = ")) {
210                                                String[] arr = line.split("=");
211                                                job.setStartTimestamp(System.currentTimeMillis());
212                                                job.setExpectedExecutionTime(Long.parseLong(arr[1].trim()) * 1000);
213                                        }
214                                        jobs.put(job.getId(), job);
215                                } else {
216                                        // handle QBlast error message
217
218                                        // Capture everything to the left of this HTML statement...
219                                        String[] tmp = line.split("</p></li></ul>");
220
221                                        // Only the error message is on the right side of this...
222                                        String[] moreTmp = tmp[0].split("<p class=\"error\">");
223                                        throw new Exception("NCBI QBlast refused this request because: " + moreTmp[1].trim());
224                                }
225
226                        }
227                        if (job != null && job.getId() != null) {
228                                return job.getId();
229                        } else {
230                                throw new Exception("Unable to retrieve request ID");
231                        }
232                } catch (IOException e) {
233                        throw new IOException("An error occured submiting sequence to BLAST server. Cause: " + e.getMessage(), e);
234                } finally {
235                        IOUtils.close(reader);
236                        IOUtils.close(writer);
237                }
238        }
239
240        /**
241         * Wrapper method for {@link #isReady(String, long)}, omitting unnecessary {@code present} property.
242         *
243         * @see #isReady(String, long)
244         */
245        public boolean isReady(String id) throws Exception {
246                return isReady(id, 0);
247        }
248
249        /**
250         * Checks for completion of request.
251         * <p>
252         * If expected execution time (RTOE) is available for request, this method will always return false until that time
253         * passes. This is done to prevent sending unnecessary requests to the server.
254         *
255         * @param id : request id, which was returned by {@code sendAlignmentRequest} method
256         * @param present : is not used, can be any value
257         * @return a boolean value telling if the request has been completed
258         */
259        @Override
260        public boolean isReady(String id, long present) throws Exception {
261                BlastJob job = jobs.get(id);
262                if (job != null) {
263                        long expectedJobFinishTime = job.getStartTimestamp() + job.getExpectedExecutionTime();
264                        if (System.currentTimeMillis() < expectedJobFinishTime) {
265                                return false;
266                        }
267                } else {
268                        // it might be a valid job from another session
269                        job = new BlastJob();
270                        job.setId(id);
271                        job.setStartTimestamp(System.currentTimeMillis());
272                        job.setExpectedExecutionTime(0);
273                }
274
275                OutputStreamWriter writer = null;
276                BufferedReader reader = null;
277                try {
278                        String checkRequest = "CMD=Get&RID=" + job.getId() + "&FORMAT_OBJECT=SearchInfo";
279                        URLConnection serviceConnection = setQBlastServiceProperties(serviceUrl.openConnection());
280                        writer = new OutputStreamWriter(serviceConnection.getOutputStream());
281                        writer.write(checkRequest);
282                        writer.flush();
283                        reader = new BufferedReader(new InputStreamReader(serviceConnection.getInputStream()));
284
285                        String line;
286                        while ((line = reader.readLine()) != null) {
287                                if (line.contains("READY")) {
288                                        jobs.put(job.getId(), job);
289                                        return true;
290                                } else if (line.contains("WAITING")) {
291                                        job.setExpectedExecutionTime(job.getExpectedExecutionTime() + WAIT_INCREMENT);
292                                        jobs.put(job.getId(), job);
293                                        return false;
294                                } else if (line.contains("UNKNOWN")) {
295                                        throw new IllegalArgumentException("Unknown request id - no results exist for it. Given id = " + id);
296                                }
297                        }
298                        return false;
299                } catch (IOException ioe) {
300                        throw new Exception("It is not possible to fetch Blast report from NCBI at this time. Cause: " + ioe.getMessage(), ioe);
301                } finally {
302                        IOUtils.close(reader);
303                        IOUtils.close(writer);
304                }
305        }
306
307        /**
308         * Extracts the actual Blast report for given request id according to options provided in {@code outputProperties}
309         * argument.
310         * <p>
311         * If the results are not ready yet, sleeps until they are available. If sleeping is not desired, call this method
312         * after {@code isReady} returns true
313         *
314         * @param id : request id, which was returned by {@code sendAlignmentRequest} method
315         * @param outputProperties : an object specifying output formatting options
316         * @return an {@code InputStream} of results
317         * @throws Exception if it is not possible to recover the results
318         */
319        @Override
320        public InputStream getAlignmentResults(String id, RemotePairwiseAlignmentOutputProperties outputProperties) throws Exception {
321                Map<String, String> params = new HashMap<>();
322                for (String key : outputProperties.getOutputOptions()) {
323                        params.put(key, outputProperties.getOutputOption(key));
324                }
325                OutputStreamWriter writer = null;
326
327                while (!isReady(id)) {
328                        Thread.sleep(WAIT_INCREMENT + 100);
329                }
330
331                params.put(CMD.name(), "Get");
332                params.put(RID.name(), id);
333                params.put(TOOL.name(), getTool());
334                params.put(EMAIL.name(), getEmail());
335                String getCmd = MAP_TO_STRING_TRANSFORMER.transform(params);
336
337                try {
338                        URLConnection serviceConnection = setQBlastServiceProperties(serviceUrl.openConnection());
339                        writer = new OutputStreamWriter(serviceConnection.getOutputStream());
340                        writer.write(getCmd);
341                        writer.flush();
342                        return serviceConnection.getInputStream();
343                } catch (IOException ioe) {
344                        throw new Exception("It is not possible to fetch Blast report from NCBI at this time. Cause: " + ioe.getMessage(), ioe);
345                } finally {
346                        IOUtils.close(writer);
347                }
348        }
349
350        /**
351         * Sends a delete request for given request id. Optional operation, ignores IOExceptions.<br/>
352         * Can be used after results of given search are no longer needed to be kept on Blast server
353         *
354         * @param id request id, as returned by {@code sendAlignmentRequest} method
355         */
356        public void sendDeleteRequest(String id) {
357                OutputStreamWriter writer = null;
358                try {
359                        String deleteRequest = "CMD=Delete&RID=" + id;
360                        URLConnection serviceConnection = setQBlastServiceProperties(serviceUrl.openConnection());
361                        writer = new OutputStreamWriter(serviceConnection.getOutputStream());
362                        writer.write(deleteRequest);
363                        writer.flush();
364                } catch (IOException ignore) {
365                        // ignore it this is an optional operation
366                } finally {
367                        IOUtils.close(writer);
368                }
369        }
370
371        /**
372         * Sets properties for given URLConnection
373         *
374         * @param conn URLConnection to set properties for
375         * @return given object after setting properties
376         */
377        private URLConnection setQBlastServiceProperties(URLConnection conn) {
378                conn.setDoOutput(true);
379                conn.setUseCaches(false);
380                conn.setRequestProperty("User-Agent", "Biojava/NCBIQBlastService");
381                conn.setRequestProperty("Connection", "Keep-Alive");
382                conn.setRequestProperty("Content-type", "application/x-www-form-urlencoded");
383                conn.setRequestProperty("Content-length", "200");
384                return conn;
385        }
386
387        /**
388         * Set the tool identifier for QBlast. Defaults to {@value #DEFAULT_TOOL}
389         *
390         * @param tool the new identifier
391         */
392        public void setTool(String tool) {
393                this.tool = tool;
394        }
395
396        /**
397         * Get the tool identifier for QBlast. Defaults to {@value #DEFAULT_TOOL}
398         *
399         * @return the identifier
400         */
401        public String getTool() {
402                return this.tool;
403        }
404
405        /**
406         * Set the email for QBlast. Defaults to {@value #DEFAULT_EMAIL}
407         *
408         * @param email the new email
409         */
410        public void setEmail(String email) {
411                this.email = email;
412        }
413
414        /**
415         * Get the email for QBlast. Defaults to {@value #DEFAULT_EMAIL}.
416         *
417         * @return the email
418         */
419        public String getEmail() {
420                return this.email;
421        }
422}