001/*
002 *                    PDB web 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 *
015 * Created on Jul 8, 2009
016 * Created by ap3
017 *
018 */
019
020package org.biojava.nbio.structure.align.util;
021
022import org.biojava.nbio.structure.align.ce.StartupParameters;
023import org.biojava.nbio.structure.io.LocalPDBDirectory.FetchBehavior;
024import org.biojava.nbio.structure.io.LocalPDBDirectory.ObsoleteBehavior;
025import org.biojava.nbio.core.util.PrettyXMLWriter;
026import org.biojava.nbio.core.util.XMLWriter;
027import org.biojava.nbio.structure.io.StructureFiletype;
028import org.slf4j.Logger;
029import org.slf4j.LoggerFactory;
030
031import java.io.File;
032import java.io.IOException;
033import java.io.PrintWriter;
034import java.util.Map;
035import java.util.concurrent.atomic.AtomicBoolean;
036
037
038/** A container to persist config to the file system
039 *
040 * @author Andreas Prlic
041 *
042 */
043public class UserConfiguration
044{
045
046        private static final Logger logger = LoggerFactory.getLogger(UserConfiguration.class);
047
048        public static final String PDB_FORMAT   = "PDB";
049        public static final String MMCIF_FORMAT = "mmCif";
050        public static final String MMTF_FORMAT  = "mmtf";
051        public static final String BCIF_FORMAT  = "bcif";
052
053        public static final String TMP_DIR = "java.io.tmpdir";
054
055        public static final String PDB_DIR   = "PDB_DIR";
056        public static final String PDB_CACHE_DIR = "PDB_CACHE_DIR";
057
058        public static final String lineSplit = System.getProperty("file.separator");
059
060        private String pdbFilePath;
061        private String cacheFilePath;
062
063        private FetchBehavior fetchBehavior;
064        private ObsoleteBehavior obsoleteBehavior;
065
066        private String fileFormat;
067
068        private static AtomicBoolean warningShown = new AtomicBoolean(false);
069
070
071        /**
072         * Default UserConfiguration:
073         * <ul>
074         * <li>split directory</li>
075         * <li>autofetch files</li>
076         * <li>default download location. This is the first specified of:
077         *      <ol><li>{@value #PDB_DIR} system property (for instance, -D{@value #PDB_DIR}=/tmp)</li>
078         *   <li>{@value #PDB_DIR} environment variable</li>
079         *   <li>System temp directory (java.io.tmpdir property)</li>
080         *   </ol>
081         *   if the provided path is not a directory then
082         *   the system's temp directory is used. A non-writable path is allowed,
083         *   only a warning will be logged.
084         * </li>
085         * <li>default cache location. This is the first specified of:
086         *      <ol><li>{@value #PDB_CACHE_DIR} system property (for instance, -D{@value #PDB_CACHE_DIR}=/tmp)</li>
087         *   <li>{@value #PDB_CACHE_DIR} environment variable</li>
088         *   <li>the value set for {@value #PDB_DIR}</li>
089         *   </ol>
090         *   if the provided path is not a directory or is not writable then
091         *   the system's temp directory is used.
092         * </li>
093         * </ul>
094         */
095        public UserConfiguration(){
096                fetchBehavior = FetchBehavior.DEFAULT;
097                obsoleteBehavior = ObsoleteBehavior.DEFAULT;
098
099                pdbFilePath = initPdbFilePath();
100                // note that in initCacheFilePath, we set to the provided one (if readable) or to the same as pdbFilePath
101                cacheFilePath = initCacheFilePath();
102
103                fileFormat = BCIF_FORMAT;
104        }
105
106        private String initPdbFilePath() {
107
108                String path = null;
109
110                String propertyName = PDB_DIR;
111
112                String userProvidedDir = System.getProperty(propertyName);
113
114                if ( userProvidedDir != null && !userProvidedDir.trim().isEmpty()) {
115
116                        path = userProvidedDir;
117                        logger.debug("Read PDB dir from system property {}: {}", propertyName, path);
118                        File f = new File(path);
119                        if (!f.isDirectory()) {
120                                logger.warn(
121                                                "Provided path {} (with system property {}) is not a directory. Using system's temp directory instead {}",
122                                                path, propertyName, System.getProperty(TMP_DIR));
123                                path = System.getProperty(TMP_DIR);
124                        } else if (!f.canWrite()) {
125                                logger.warn(
126                                                "Provided path {} (with system property {}) is not writable. Will not be able to write cached files.",
127                                                path, propertyName);
128                                // we don't require the PDB_DIR to be writable, so that it can be used with a pre-rsynced dir
129                                // thus if not writable, we only warn and go ahead using it
130                        }
131
132
133                } else {
134                        Map<String,String> env = System.getenv();
135
136                        if( env.containsKey(propertyName) && !env.get(propertyName).trim().isEmpty()) {
137                                path = env.get(propertyName);
138                                logger.debug("Read dir from environment variable {}: {}", propertyName, path);
139
140                                File f = new File(path);
141                                if (!f.isDirectory()) {
142                                        logger.warn(
143                                                        "Provided path {} (with environment variable {}) is not a directory. Using system's temp directory instead {}",
144                                                        path, propertyName, System.getProperty(TMP_DIR));
145                                        path = System.getProperty(TMP_DIR);
146                                } else if (!f.canWrite()) {
147                                        logger.warn(
148                                                        "Provided path {} (with environment variable {}) is not writable. Will not be able to write cached files",
149                                                        path, propertyName);
150                                        // we don't require the PDB_DIR to be writable, so that it can be used with a pre-rsynced dir
151                                        // thus if not writable, we only warn and go ahead using it
152                                }
153
154                        } else {
155                                path = System.getProperty(TMP_DIR);
156
157                                if ( ! warningShown.get()) {
158
159                                        logger.warn("Could not read dir from system property {} or environment variable {}, "
160                                                                        + "using system's temp directory {}",
161                                                        propertyName, propertyName, path);
162
163                                        warningShown.set(true);
164                                }
165
166                                System.setProperty(propertyName,path);
167                        }
168                }
169
170                if ( ! path.endsWith(lineSplit) )
171                        path = path + lineSplit;
172
173                return path;
174
175        }
176
177        private String initCacheFilePath() {
178
179                String path = null;
180
181                String propertyName = PDB_CACHE_DIR;
182
183                String userProvidedDir = System.getProperty(propertyName);
184
185                if ( userProvidedDir != null ) {
186
187                        path = userProvidedDir;
188                        logger.debug("Read cache dir from system property {}: {}", propertyName, path);
189                        File f = new File(path);
190                        if (!f.isDirectory()) {
191                                logger.warn(
192                                                "Provided path {} (with system property {}) is not a directory. Using system's temp directory instead {}",
193                                                path, propertyName, System.getProperty(TMP_DIR));
194                                path = System.getProperty(TMP_DIR);
195                        } else if (!f.canWrite()) {
196                                logger.warn(
197                                                "Provided path {} (with system property {}) is not writable. Using system's temp directory instead {}",
198                                                path, propertyName, System.getProperty(TMP_DIR));
199                                path = System.getProperty(TMP_DIR);
200                                System.setProperty(propertyName,path);
201                        }
202
203
204                } else {
205                        Map<String,String> env = System.getenv();
206
207                        if( env.containsKey(propertyName)) {
208                                path = env.get(propertyName);
209                                logger.debug("Read dir from environment variable {}: {}", propertyName, path);
210
211                                File f = new File(path);
212                                if (!f.isDirectory()) {
213                                        logger.warn(
214                                                        "Provided path {} (with environment variable {}) is not a directory. Using system's temp directory instead {}",
215                                                        path, propertyName, System.getProperty(TMP_DIR));
216                                        path = System.getProperty(TMP_DIR);
217                                } else if (!f.canWrite()) {
218                                        logger.warn(
219                                                        "Provided path {} (with environment variable {}) is not writable. Using system's temp directory instead {}",
220                                                        path, propertyName, System.getProperty(TMP_DIR));
221                                        path = System.getProperty(TMP_DIR);
222                                }
223
224                        } else {
225                                // NOTE in case of not provided, then it is set to same as pdbFilePath
226                                // as PDB_DIR is not checked for being writable, we have to do that check here in case
227                                if (new File(pdbFilePath).canWrite()){
228                                        path = pdbFilePath;
229                                        logger.info("Could not read cache dir from system property {} or environment variable {}, "
230                                                        + "using PDB directory instead {}",
231                                                        propertyName, propertyName, path);
232                                        System.setProperty(propertyName,path);
233
234                                } else {
235                                        path = System.getProperty(TMP_DIR);
236                                        logger.warn("Could not read cache dir from system property {} or environment variable {}, "
237                                                        + "and PDB directory {} is not writable. Using system's temp directory instead {}",
238                                                        propertyName, propertyName, pdbFilePath, path);
239                                        System.setProperty(propertyName,path);
240
241                                }
242                        }
243                }
244
245                if ( ! path.endsWith(lineSplit) )
246                        path = path + lineSplit;
247
248                return path;
249
250        }
251
252        public String getPdbFilePath()
253        {
254                return pdbFilePath;
255        }
256
257        public void setPdbFilePath(String pdbFilePath)
258        {
259                this.pdbFilePath = pdbFilePath;
260        }
261
262        public String getCacheFilePath() {
263                return cacheFilePath;
264        }
265
266        public void setCacheFilePath(String cacheFilePath) {
267                this.cacheFilePath = cacheFilePath;
268        }
269
270        public FetchBehavior getFetchBehavior() {
271                return fetchBehavior;
272        }
273
274        public void setFetchBehavior(FetchBehavior fetchBehavior) {
275                this.fetchBehavior = fetchBehavior;
276        }
277
278        public ObsoleteBehavior getObsoleteBehavior() {
279                return obsoleteBehavior;
280        }
281
282        public void setObsoleteBehavior(ObsoleteBehavior obsoleteBehavior) {
283                this.obsoleteBehavior = obsoleteBehavior;
284        }
285
286        /** convert Configuration to an XML file so it can be serialized
287         *
288         * @param pw
289         * @return XMLWriter
290         * @throws IOException
291         */
292        public XMLWriter toXML(PrintWriter pw)
293                        throws IOException
294                        {
295
296                XMLWriter     xw = new PrettyXMLWriter( pw);
297
298                toXML(xw);
299                return xw ;
300                        }
301
302
303        /** convert Configuration to an XML file so it can be serialized
304         * add to an already existing xml file.
305         *
306         * @param xw the XML writer to use
307         * @return the writer again
308         * @throws IOException
309         */
310
311        public XMLWriter toXML(XMLWriter xw)
312                        throws IOException
313                        {
314                xw.printRaw("<?xml version='1.0' standalone='no' ?>");
315                //xw.printRaw("<!DOCTYPE " + XML_CONTENT_TYPE + " SYSTEM '" + XML_DTD + "' >");
316                xw.openTag("JFatCatConfig");
317
318                xw.openTag("PDBFILEPATH");
319                // we don;t serialize the tempdir...
320                String tempdir = System.getProperty(TMP_DIR);
321                if (! pdbFilePath.equals(tempdir))
322                        xw.attribute("path", pdbFilePath);
323
324                xw.attribute("fetchBehavior", fetchBehavior+"");
325                xw.attribute("obsoleteBehavior", obsoleteBehavior+"");
326                xw.attribute("fileFormat", fileFormat);
327                xw.closeTag("PDBFILEPATH");
328
329                xw.closeTag("JFatCatConfig");
330                return xw ;
331
332                        }
333
334        public static UserConfiguration fromStartupParams(StartupParameters params) {
335                UserConfiguration config = new UserConfiguration();
336                config.setPdbFilePath(params.getPdbFilePath());
337
338                if(params.isAutoFetch()) {
339                        config.setFetchBehavior(FetchBehavior.DEFAULT);
340                } else {
341                        config.setFetchBehavior(FetchBehavior.LOCAL_ONLY);
342                }
343
344                // TODO support MMCif Files
345                config.setFileFormat(UserConfiguration.PDB_FORMAT);
346                return config;
347        }
348
349        public void setFileFormat (String fileFormat){
350                this.fileFormat = fileFormat;
351        }
352
353        public String getFileFormat()
354        {
355                return fileFormat;
356        }
357
358        public StructureFiletype getStructureFiletype() {
359                switch (fileFormat) {
360                        case MMCIF_FORMAT:
361                                return StructureFiletype.CIF;
362                        case PDB_FORMAT:
363                                return StructureFiletype.PDB;
364                        case MMTF_FORMAT:
365                                return StructureFiletype.MMTF;
366                        case BCIF_FORMAT:
367                        default:
368                                return StructureFiletype.BCIF;
369                }
370        }
371}