GroovyScriptEngine无法加载正在运行的groovy脚本的导入

时间:2014-06-08 18:24:17

标签: java groovy groovyshell

背景:

我最近开始使用Groovy并尝试在eclipse插件中嵌入一个groovy脚本引擎,让我的客户在我的基于eclipse的产品中开发自己的GUI扩展。这与codehaus网站上发布的success story非常相似。

问题

当尝试加载一个groovy类时,GroovyScriptEngine会从eclipse插件运行groovy脚本(让我们称之为" main_eclipse.groovy")#" SwtGuiBuilder" ),出现以下错误:

  

BUG!在迭代的同时排队新的源代码。排队的来源是&f;文件:/home/nicolas/workspace/groovy-test/src/gui/SwtGuiBuilder.groovy'

问题

有没有人遇到同样的问题?怎么修好? 任何帮助将受到高度赞赏!

一些观察结果:

  • 当使用groovy解释器而不是GroovyScriptEngine java对象时,使用我的SwtGuiBuilder类没有问题(参见脚本" main_groovy"在下面)。

  • 我的问题似乎不是类路径问题,因为在抛出的异常中提到了包含我的SwtGuiBuilder类的文件。

  • 在两个报告的groovy错误GRECLIPSE-429和GRECLIPSE-1037中提到了错误消息。我没有完全掌握技术细节,但这些错误似乎与加载大量课程时的性能问题有关,这与我的情况无关......

详情

SampleView.java

public class SampleView
{
public SampleView() { super(); }

public void createPartControl(Composite parent) 
{
    String    groovyScript = null;
    String [] groovyPath   = null;

    boolean shall_exit = false;
    do
    {      // ask user for params
        GroovyLocationDialog groovyLocationDialog= new GroovyLocationDialog(parent.getShell() );
        int return_code = groovyLocationDialog.open();
        if ( return_code != Window.OK )
            shall_exit = true;
        else 
        {
            groovyScript= groovyLocationDialog.getInputScriptName();
            groovyPath  = groovyLocationDialog.getInputScriptPath();

            // run it
            ScriptConnector scriptConnector = new ScriptConnector(parent);
            try                 { scriptConnector.runGuiComponentScript( groovyPath, groovyScript); }
            catch (Exception e) { e.printStackTrace(); }
            System.out.println("script finished");
        }
    }
    while ( ! shall_exit );
}

ScriptConnector.java

public class ScriptConnector
{
    private String[]  roots;
    private Composite window;
    private Binding   binding;

    public ScriptConnector( Composite window )
    {
         this.window    = window;
         Binding  scriptenv = new Binding();    // A new Binding is created ...
         scriptenv.setVariable("SDE", this); 
         scriptenv.setVariable("WINDOW", this.window); // ref to current window

         this.binding = scriptenv;
    }

    public void runGuiComponentScript(final String[] groovyPath, final String scriptName) 
    {
        GroovyScriptEngine gse = null;
        this.roots     = groovyPath;
        try 
        {
            // instanciating the script engine with current classpath
            gse = new GroovyScriptEngine( roots, this.getClass().getClassLoader() );
            gse.run(scriptName, binding);      // ... and run specified script
        }
        catch (Exception e) { e.printStackTrace(); }
        catch (Throwable t) { t.printStackTrace(); }
    }
}

main_eclipse.groovy

package launcher;

import org.eclipse.swt.SWT
import org.eclipse.swt.widgets.*
import org.eclipse.swt.events.*;
import org.eclipse.swt.layout.RowLayout as Layout

// This import will fail...
import gui.SwtGuiBuilder;


WINDOW.layout = new Layout(SWT.VERTICAL);
def builder = new SwtGuiBuilder(WINDOW);
builder.Label ( style=SWT.NONE, text = 'Simple demo of Groovy and SWT')
builder.Button( style=SWT.PUSH, text = 'Click me' , action = { println "Click !" } )

SwtGuiBuilder.groovy

package gui;

import org.eclipse.swt.events.*
import org.eclipse.swt.widgets.Button
import org.eclipse.swt.widgets.Composite
import org.eclipse.swt.widgets.Label


class SwtGuiBuilder
{
    private Composite _parent

    public SwtGuiBuilder(Composite parent) { _parent = parent }

    public void Button( style = SWT.PUSH, text= null, action = null )
    {
        def btn = new Button(_parent, style)
        if ( text != null )
            btn.text = text
        if (action != null)
            btn.addSelectionListener( new SelectionAdapter() { void widgetSelected( SelectionEvent event ) { action(); } } );
    }

    public void Label( style = SWT.NONE, text = '' )
    {
        def lbl = new Label(_parent, style)
        lbl.text = text
    }
}

main_groovy.groovy

package launcher;

import org.eclipse.swt.SWT
import org.eclipse.swt.widgets.*
import org.eclipse.swt.events.*;
import org.eclipse.swt.layout.RowLayout as Layout

// ... But this import is handled properly !
import gui.SwtGuiBuilder;

def display = new Display()
def WINDOW = new Shell(display)
WINDOW.text = 'Groovy / SWT Test';

WINDOW.layout = new Layout(SWT.VERTICAL);
def builder = new SwtGuiBuilder(WINDOW);
builder.Label ( style=SWT.NONE, text = 'Simple demo of Groovy and SWT')
builder.Button( style=SWT.PUSH, text = 'Click me' , action = { println "Ya clicked me !" } )

WINDOW.pack();
WINDOW.open();

while (!WINDOW.disposed) {
    if (!WINDOW.display.readAndDispatch())
        WINDOW.display.sleep();
}

堆栈跟踪

BUG!在迭代的同时排队新的源代码。排队的来源是&f;文件:/home/nicolas/workspace/groovy-test/src/gui/SwtGuiBuilder.groovy'     在org.codehaus.groovy.control.CompilationUnit.addSource(CompilationUnit.java:460)     在org.codehaus.groovy.control.CompilationUnit.addSource(CompilationUnit.java:433)     at groovy.util.GroovyScriptEngine $ ScriptClassLoader $ 3.findClassNode(GroovyScriptEngine.java:195)     at org.codehaus.groovy.control.ClassNodeResolver.resolveName(ClassNodeResolver.java:124)     在org.codehaus.groovy.control.ResolveVisitor.resolveToOuter(ResolveVisitor.java:863)     在org.codehaus.groovy.control.ResolveVisitor.resolve(ResolveVisitor.java:377)     在org.codehaus.groovy.control.ResolveVisitor.visitClass(ResolveVisitor.java:1407)     在org.codehaus.groovy.control.ResolveVisitor.startResolving(ResolveVisitor.java:202)     在org.codehaus.groovy.control.CompilationUnit $ 1.call(CompilationUnit.java:713)     at org.codehaus.groovy.control.CompilationUnit.applyToSourceUnits(CompilationUnit.java:1015)     在org.codehaus.groovy.control.CompilationUnit.doPhaseOperation(CompilationUnit.java:647)     在org.codehaus.groovy.control.CompilationUnit.compile(CompilationUnit.java:596)     at groovy.lang.GroovyClassLoader.doParseClass(GroovyClassLoader.java:279)     在groovy.lang.GroovyClassLoader.parseClass(GroovyClassLoader.java:258)     at groovy.util.GroovyScriptEngine $ ScriptClassLoader.doParseClass(GroovyScriptEngine.java:247)     at groovy.util.GroovyScriptEngine $ ScriptClassLoader.parseClass(GroovyScriptEngine.java:229)     在groovy.lang.GroovyClassLoader.parseClass(GroovyClassLoader.java:244)     at groovy.lang.GroovyClassLoader.parseClass(GroovyClassLoader.java:202)     at groovy.util.GroovyScriptEngine.loadScriptByName(GroovyScriptEngine.java:514)     at groovy.util.GroovyScriptEngine.createScript(GroovyScriptEngine.java:564)     at groovy.util.GroovyScriptEngine.run(GroovyScriptEngine.java:551)

我的配置:

  • Linux Ubuntu 14.04 x86

  • Groovy版本:2.3.2

  • JVM:1.7.0_55

  • Eclipse Kepler SR2 - Build 20140224-0627

  • Eclipse Groovy插件v2.0.7

1 个答案:

答案 0 :(得分:0)

而不是GroovyScriptEngine,我使用了GroovyShell类(下面的groovy代码,但很容易改回java),CompilerConfiguration允许你指定类路径。

    def config = new CompilerConfiguration(classpath: classpath)
    def binding = new Binding()

    def result = new GroovyShell(binding, config).evaluate("""
def foo='bar'
""")