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 @ 471

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
        unit.load(folder, id);
259
//        if (!unit.getFile().exists()) {
260
//            unit.create(folder, id);
261
//        } else {
262
//            unit.load(folder, id);
263
//        }
264
        return unit;
265
    }
266

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

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

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

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

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

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

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

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

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

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

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

    
368
    }
369

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

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

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

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

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

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

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

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

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

    
454
    }
455

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

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

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

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

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

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

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

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

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

    
522
}