Statistics
| Revision:

gvsig-scripting / org.gvsig.scripting / trunk / org.gvsig.scripting / org.gvsig.scripting.lib / org.gvsig.scripting.lib.impl / src / main / java / org / gvsig / scripting / impl / DefaultScriptingManager.java @ 468

History | View | Annotate | Download (16.8 KB)

1
package org.gvsig.scripting.impl;
2

    
3
import java.io.File;
4
import java.util.ArrayList;
5
import java.util.Collections;
6
import java.util.HashMap;
7
import java.util.Iterator;
8
import java.util.List;
9
import java.util.Map;
10

    
11
import javax.script.ScriptContext;
12
import javax.script.ScriptEngine;
13
import javax.script.ScriptEngineFactory;
14
import javax.script.ScriptEngineManager;
15
import javax.script.SimpleBindings;
16
import javax.swing.ImageIcon;
17

    
18
import org.apache.commons.io.FileUtils;
19
import org.apache.commons.io.FilenameUtils;
20
import org.gvsig.scripting.ScriptingBaseScript;
21
import org.gvsig.scripting.ScriptingDialog;
22
import org.gvsig.scripting.ScriptingFolder;
23
import org.gvsig.scripting.ScriptingHelpManager;
24
import org.gvsig.scripting.ScriptingManager;
25
import org.gvsig.scripting.ScriptingScript;
26
import org.gvsig.scripting.ScriptingUnit;
27
import org.gvsig.tools.service.spi.ProviderFactory;
28
import org.slf4j.Logger;
29
import org.slf4j.LoggerFactory;
30

    
31
public class DefaultScriptingManager implements ScriptingManager {
32

    
33
    public static class RegisterSystemFolder {
34

    
35
        public String name;
36
        public File folder;
37

    
38
        public RegisterSystemFolder(String name, File folder) {
39
            this.name = name;
40
            this.folder = folder;
41
        }
42
    }
43

    
44
    private static final Logger LOG = LoggerFactory
45
            .getLogger(DefaultScriptingManager.class);
46

    
47
    protected Map<String, ImageIcon> icons;
48
    protected List<RegisterSystemFolder> systemFolders = new ArrayList<>();
49
    protected ScriptEngineManager engineManager = null;
50
    private final SimpleBindings bindings = new SimpleBindings();
51
    private ScriptingHelpManager helpManager = null;
52
    private List<String> unitTypes = null;
53
    private ClassLoader classLoader = null;
54
    private List<File> libFolders = new ArrayList<>();
55
    private File home = null;
56
    private List<ScriptingFolder> alternativeUserFolders = new ArrayList<>();
57
    private Map<String, String> extensionOfLanguage = null;
58
    private Map<String, String> languageOfExtension = null;
59
    private File packagesFolder;
60

    
61
    public DefaultScriptingManager() {
62
        this.classLoader = getClass().getClassLoader();
63
        this.setHomeFolder(null);
64
        this.bindings.put("ScriptingManager", this);
65
    }
66

    
67
    public DefaultScriptingManager(ClassLoader classLoader) {
68
        this();
69
        this.classLoader = classLoader;
70
    }
71

    
72
    private void createFolder(File f) {
73
        if (!f.exists()) {
74
            try {
75
                FileUtils.forceMkdir(f);
76
                LOG.info("Created scripting folder '" + f.getAbsolutePath() + "'");
77
            } catch (Throwable e) {
78
                LOG.warn("Can't Create scripting folder '" + f.getAbsolutePath() + "'");
79
            }
80
        }
81
    }
82

    
83
    private void createDefaultFolders(File home) {
84
        createFolder(new File(home, "scripts"));
85
        createFolder(new File(home, "help"));
86
        createFolder(new File(home, "lib"));
87
    }
88

    
89
    @Override
90
    public File getHomeFolder() {
91
        if (!this.home.exists()) {
92
            createFolder(home);
93
            createDefaultFolders(home);
94
        }
95
        return this.home;
96
    }
97

    
98
    @Override
99
    public void setHomeFolder(File home) {
100
        if (home == null) {
101
            this.home = new File(System.getProperty("user.home"), ".gvsig-scripting");
102
        } else {
103
            this.home = home;
104
        }
105
        createDefaultFolders(this.home);
106
        LOG.info("Set scripting home to '" + this.home.getAbsolutePath() + "'");
107
        this.addLibFolder(new File(this.home, "lib"));
108
    }
109

    
110
    protected ScriptEngineManager getEngineManager() {
111
        if (this.engineManager == null) {
112
            this.engineManager
113
                    = classLoader == null ? new ScriptEngineManager()
114
                            : new ScriptEngineManager(classLoader);
115
            showEnginesInfo(engineManager);
116
        }
117
        return this.engineManager;
118
    }
119

    
120
    private void showEnginesInfo(ScriptEngineManager mgr) {
121
        if (LOG.isInfoEnabled()) {
122
            List<ScriptEngineFactory> factories = mgr.getEngineFactories();
123
            StringBuilder buffer = new StringBuilder();
124
            List<Object> values = new ArrayList<>();
125
            buffer.append("Scripting engines available:");
126
            for (ScriptEngineFactory factory : factories) {
127

    
128
                // Main engine info
129
                buffer
130
                        .append("\n- {}: version = {}, language = {}, langVersion = {}");
131
                values.add(factory.getEngineName());
132
                values.add(factory.getEngineVersion());
133
                values.add(factory.getLanguageName());
134
                values.add(factory.getLanguageVersion());
135

    
136
                // Aliases
137
                buffer.append("\n\t- Aliases: ");
138
                List<String> engNames = factory.getNames();
139
                int size = engNames.size();
140
                for (String name : engNames) {
141
                    size--;
142
                    buffer.append("{}");
143
                    if (size > 0) {
144
                        buffer.append(", ");
145
                    }
146
                    values.add(name);
147
                }
148
                buffer.append("\n\t- File extensions: ");
149
                List<String> extNames = factory.getExtensions();
150
                size = extNames.size();
151
                for (String name : extNames) {
152
                    size--;
153
                    buffer.append("{}");
154
                    if (size > 0) {
155
                        buffer.append(", ");
156
                    }
157
                    values.add(name);
158
                }
159
                buffer.append("\n\t- Mime types: ");
160
                List<String> mimeNames = factory.getMimeTypes();
161
                size = mimeNames.size();
162
                for (String name : mimeNames) {
163
                    size--;
164
                    buffer.append("{}");
165
                    if (size > 0) {
166
                        buffer.append(", ");
167
                    }
168
                    values.add(name);
169
                }
170

    
171
            }
172
            LOG.info(buffer.toString(), values.toArray());
173
        }
174
    }
175

    
176
    protected ScriptEngine getEngine(String langName) {
177
        ScriptEngineManager manager = this.getEngineManager();
178
        ScriptEngine engine = manager.getEngineByName(langName);
179
        engine.getBindings(ScriptContext.ENGINE_SCOPE).putAll(bindings);
180
        return engine;
181
    }
182

    
183
    public ImageIcon getIcon(String name) {
184
        return this.icons.get(name);
185
    }
186

    
187
    @Override
188
    public String getEngineNameByLanguage(String langName) {
189
        List<ScriptEngineFactory> factories
190
                = getEngineManager().getEngineFactories();
191

    
192
        for (ScriptEngineFactory factory : factories) {
193
            if (factory.getLanguageName().equalsIgnoreCase(langName)) {
194
                return factory.getEngineName();
195
            }
196
        }
197
        return null;
198
    }
199

    
200
    public ScriptEngine getEngineByLanguage(String langName) {
201
        List<ScriptEngineFactory> factories
202
                = getEngineManager().getEngineFactories();
203

    
204
        for (ScriptEngineFactory factory : factories) {
205
            if (factory.getLanguageName().equalsIgnoreCase(langName)) {
206
                return factory.getScriptEngine();
207
            }
208
        }
209
        return null;
210
    }
211

    
212
    @Override
213
    public boolean validateUnitId(ScriptingFolder folder, String id) {
214
        List<ScriptingUnit> units = folder.getUnits();
215
        String fileName;
216
        String s[];
217
        String extension;
218
        for (ScriptingUnit unit : units) {
219
            fileName = unit.getId();
220
            fileName = FilenameUtils.getBaseName(fileName);
221
            if (fileName.equals(id)) {
222
                return false;
223
            }
224
        }
225
        return true;
226
    }
227

    
228
    public ScriptingScript createScript(ScriptingFolder folder, String id) {
229
        return this.createScript(folder, id, null);
230
    }
231

    
232
    private ScriptingScript createScript(ScriptingFolder folder, String id, String language) {
233
        DefaultScriptingScript script = new DefaultScriptingScript(folder, this, id);
234
        if (!script.getFile().exists()) {
235
            script.create(folder, id, language);
236
        } else {
237
            script.load(folder, id);
238
        }
239
        return script;
240
    }
241

    
242
    public ScriptingDialog createDialog(ScriptingFolder folder, String id) {
243
        return this.createDialog(folder, id, null);
244
    }
245

    
246
    private ScriptingDialog createDialog(ScriptingFolder folder, String id, String language) {
247
        DefaultScriptingDialog dialog = new DefaultScriptingDialog(folder, this, id);
248
        if (!dialog.getFile().exists()) {
249
            dialog.create(folder, id, language);
250
        } else {
251
            dialog.load(folder, id);
252
        }
253
        return dialog;
254
    }
255

    
256
    public ScriptingFolder createFolder(ScriptingFolder folder, String id) {
257
        DefaultScriptingFolder unit = new DefaultScriptingFolder(folder, this, new File(folder.getFile(), id));
258
        if (!unit.getFile().exists()) {
259
            unit.create(folder, id);
260
        } else {
261
            unit.load(folder, id);
262
        }
263
        return unit;
264
    }
265

    
266
    @Override
267
    public ScriptingBaseScript getScript(File file) {
268
        ScriptingBaseScript script = (ScriptingBaseScript) this.getUnit(file);
269
        if (script == null) {
270
            throw new ScriptNotFoundException(file);
271
        }
272
        return script;
273
    }
274

    
275
    @Override
276
    public ScriptingFolder getFolder(File file) {
277
        ScriptingFolder folder = (ScriptingFolder) this.getUnit(file);
278
        if (folder == null) {
279
            throw new RuntimeException(file.getAbsolutePath());
280
        }
281
        return folder;
282
    }
283

    
284
    public ScriptingUnit getUnit(File file) {
285
        ScriptingFolder folder;
286
        ScriptingUnit unit;
287

    
288
        if (file.isAbsolute()) {
289
            folder = new DefaultScriptingFolder(null, this, file.getParentFile());
290
            unit = folder.getUnit(file);
291
            return unit;
292
        } else {
293
            folder = this.getUserFolder();
294
            unit = folder.getUnit(file);
295
            if (unit != null) {
296
                return unit;
297
            }
298

    
299
            folder = this.getSystemFolder();
300
            unit = folder.getUnit(file);
301
            if (unit != null) {
302
                return unit;
303
            }
304
        }
305
        return null;
306
    }
307

    
308
    @Override
309
    public ScriptingFolder getSystemFolder() {
310
        return new SystemFolder(this);
311
    }
312

    
313
    @Override
314
    public ScriptingFolder getUserFolder() {
315
        return new UserFolder(this, this.getRootUserFolder());
316
    }
317

    
318
    @Override
319
    public ScriptingBaseScript getScript(String name) {
320
        return (ScriptingBaseScript) findScript(null, name);
321
    }
322

    
323
    private ScriptingUnit findScript(ScriptingFolder folder, String name) {
324
        if (name == null) {
325
            return null;
326
        }
327
        if (name.trim().length() == 0) {
328
            return null;
329
        }
330
        ScriptingUnit unit;
331
        if (folder == null) {
332
            unit = findScript(this.getUserFolder(), name);
333
            if (unit != null) {
334
                return unit;
335
            }
336
            unit = findScript(this.getSystemFolder(), name);
337
            return unit;
338
        }
339
        List<ScriptingUnit> units = folder.getUnits();
340
        Iterator<ScriptingUnit> it = units.iterator();
341
        while (it.hasNext()) {
342
            unit = it.next();
343
            if (unit instanceof ScriptingFolder) {
344
                unit = findScript((ScriptingFolder) unit, name);
345
                if (unit != null) {
346
                    return unit;
347
                }
348
            } else if (unit instanceof ScriptingBaseScript) {
349
                if (name.equalsIgnoreCase(unit.getId())) {
350
                    return unit;
351
                }
352
            }
353
        }
354
        return null;
355
    }
356

    
357
    @Override
358
    public File getRootUserFolder() {
359
        return new File(this.getHomeFolder(), "scripts");
360
    }
361

    
362
    @Override
363
    public void registerSystemFolder(String name, File folder) {
364
        this.systemFolders.add(new RegisterSystemFolder(name, folder));
365
        LOG.info("Register system folder name '" + name + "' folder " + folder.getAbsolutePath() + "'");
366

    
367
    }
368

    
369
    public List<RegisterSystemFolder> getSystemFolders() {
370
        return this.systemFolders;
371
    }
372

    
373
    @Override
374
    public List<ScriptingFolder> getAlternativeUserFolders() {
375
        return this.alternativeUserFolders;
376
    }
377

    
378
    @Override
379
    public void addAlternativeUserFolder(File f, String name, String description) {
380
        UserFolder x = new UserFolder(this, f);
381
        x.setName(name);
382
        x.setDescription(description);
383
        x.setId("UserFolder_" + this.alternativeUserFolders.size() + 1);
384
        this.alternativeUserFolders.add(x);
385
    }
386

    
387
    @Override
388
    public String getExtensionOfLanguage(String langName) {
389
        if (this.extensionOfLanguage == null) {
390
            Map<String, String> extensionOfLanguage = new HashMap<>();
391
            List<ScriptEngineFactory> factories = getEngineManager().getEngineFactories();
392
            for (ScriptEngineFactory factory : factories) {
393
                List<String> extensions = factory.getExtensions();
394
                if (extensions != null && !extensions.isEmpty()) {
395
                    extensionOfLanguage.put(
396
                            factory.getLanguageName().toLowerCase(),
397
                            extensions.get(0).toLowerCase()
398
                    );
399
                }
400
            }
401
            this.extensionOfLanguage = extensionOfLanguage;
402
        }
403
        if (langName == null) {
404
            return null;
405
        }
406
        langName = langName.toLowerCase();
407
        return this.extensionOfLanguage.get(langName);
408
    }
409

    
410
    @Override
411
    public List<String> getSupportedLanguages() {
412
        List<String> languages = new ArrayList<>();
413

    
414
        languages.addAll(this.extensionOfLanguage.keySet());
415
        Collections.sort(languages);
416
        return languages;
417
    }
418

    
419
    public String getLanguageOfExtension(String extension) {
420
        if (extension == null) {
421
            return null;
422
        }
423
        extension = extension.toLowerCase();
424
        if (extension.startsWith(".")) {
425
            extension = extension.substring(1);
426
        }
427
        if (this.languageOfExtension == null) {
428
            Map<String, String> languageOfExtension = new HashMap<>();
429
            List<ScriptEngineFactory> factories = getEngineManager().getEngineFactories();
430
            for (ScriptEngineFactory factory : factories) {
431
                List<String> extensions = factory.getExtensions();
432
                if (extensions != null) {
433
                    for (String extension1 : extensions) {
434
                        languageOfExtension.put(
435
                                extension1.toLowerCase(),
436
                                factory.getLanguageName().toLowerCase()
437
                        );
438
                    }
439
                }
440
            }
441
            this.languageOfExtension = languageOfExtension;
442
        }
443
        return this.languageOfExtension.get(extension);
444
    }
445

    
446
    public Object get(String key) {
447
        return this.bindings.get(key);
448
    }
449

    
450
    public void put(String key, Object value) {
451
        this.bindings.put(key, value);
452

    
453
    }
454

    
455
    @Override
456
    public ScriptingHelpManager getHelpManager() {
457
        if (this.helpManager == null) {
458
            this.helpManager = new DefaultScriptingHelpManager(this);
459
        }
460
        return this.helpManager;
461
    }
462

    
463
    @Override
464
    public ScriptingUnit createUnit(String unitType, ScriptingFolder folder, String id) {
465
        return createUnit(unitType, folder, id, null);
466
    }
467

    
468
    @Override
469
    public ScriptingUnit createUnit(String unitType, ScriptingFolder folder, String id, String language) {
470
        if (unitType.equals(UNIT_SCRIPT)) {
471
            return this.createScript(folder, id, language);
472
        }
473
        if (unitType.equals(UNIT_DIALOG)) {
474
            return this.createDialog(folder, id, language);
475
        }
476
        if (unitType.equals(UNIT_FOLDER)) {
477
            return this.createFolder(folder, id);
478
        }
479
        return null;
480
    }
481

    
482
    @Override
483
    public List<String> getUnitTypes() {
484
        if (this.unitTypes == null) {
485
            this.unitTypes = new ArrayList<>();
486
            this.unitTypes.add(UNIT_SCRIPT);
487
            this.unitTypes.add(UNIT_DIALOG);
488
            this.unitTypes.add(UNIT_FOLDER);
489
        }
490
        return this.unitTypes;
491
    }
492

    
493
    @Override
494
    public void addLibFolder(File lib) {
495
        if (lib.exists()) {
496
            LOG.info("Add scripting lib folder '" + lib.getAbsolutePath() + "'");
497
            this.libFolders.add(lib);
498
        } else {
499
            LOG.info("Skip add scripting lib folder '" + lib.getAbsolutePath() + "', folder don't exist");
500
        }
501
    }
502

    
503
    public List<File> getLibFolders() {
504
        return this.libFolders;
505
    }
506

    
507
    public ProviderFactory getInstallerFactory() {
508
        return new ScriptingInstallerProviderFactory();
509
    }
510

    
511
    @Override
512
    public File getPackagesFolder() {
513
        return this.packagesFolder;
514
    }
515

    
516
    @Override
517
    public void setPackagesFolder(File folder) {
518
        this.packagesFolder = folder;
519
    }
520

    
521
}