Как вызвать Perl 6 из Java?
Регулярные выражения / грамматики Perl 6 намного лучше структурированы, более мощные и читаемые, чем Perl 5 или связанные с ним Perl-совместимые регулярные выражения везде, включая регулярные выражения в Java. Я ищу способ выполнить код Perl 6 с этим кодом регулярного выражения / грамматики из Java.
Вот типичный пример, который я хочу сделать:
grammar Calculator {
token TOP { [ <add> | <sub> ] }
rule add { <num> '+' <num> }
rule sub { <num> '-' <num> }
token num { \d+ }
}
class Calculations {
method TOP ($/) { make $<add> ?? $<add>.made !! $<sub>.made; }
method add ($/) { make [+] $<num>; }
method sub ($/) { make [-] $<num>; }
}
say Calculator.parse('2 + 3', actions => Calculations).made;
# OUTPUT: «5»
Может быть, я должен написать Class
в Perl 6 и должны скомпилировать это для байт-кода JVM, а затем я могу вызвать это. Это решение или нет? Или это невозможно?
Может быть, слишком сложно вызвать Perl 6 из Java. Есть и другое направление. В Perl 6 множество модулей Inline, таких как Inline:: Python, Inline:: Perl5 и так далее. Существует также способ запуска кода Java в Perl 6. Вот пример, который я нашел:
use java::util::zip::CRC32:from<java>;
my $crc = CRC32.new();
for 'Hello, Java'.encode('utf-8') {
$crc.'method/update/(B)V'($_);
}
say $crc.getValue();
Это возможный способ начать с Perl 6 и связать массу Java-кода с одним проектом? Но как вернуться с Java на мой код Perl 6? Для Perl 5 я могу найти модуль Inline:: Java:: Callback, но не для Perl 6.
Как мне сделать это профессионально?
1 ответ
Я делюсь результатами собственных экспериментов и наблюдений, в надежде, что они будут полезны, даже если мой вывод на данный момент не очень положительный. Мой краткий ответ на вопрос ОП: по состоянию на май 2019 года это еще невозможно.
Теперь длинный ответ: в последних выпусках Rakudo Star внутренняя поддержка JVM Perl6 еще не находится в стабильном и готовом к использованию состоянии:https://rakudo.org/post/announce-rakudo-star-release-2019-03
В любом случае, если вы хотите испытать удачу, вот пример, полученный из rakudo-star/nqp/examples (с небольшим патчем, исходный код из rakudo-star-2019.03 не скомпилируется из коробки). Улучшения исходного примера также включают документацию и базовое управление аргументами командной строки:
package examples;
import org.perl6.nqp.runtime.*;
import static org.perl6.nqp.runtime.CallSiteDescriptor.*;
import org.perl6.nqp.sixmodel.*;
public class CallFromJava {
private GlobalContext g;
private ThreadContext t;
private SixModelObject nqpComp;
private CallFromJava(String bytecode, String hll) {
g = new GlobalContext();
t = g.getCurrentThreadContext();
Ops.loadbytecode(bytecode, t);
nqpComp = Ops.getcomp(hll, t);
}
private SixModelObject eval(String nqp) {
Ops.invokeDirect(t, Ops.findmethod(nqpComp, "compile", t),
new CallSiteDescriptor(new byte[] { ARG_OBJ, ARG_STR }, null),
new Object[] { nqpComp, nqp });
Ops.invokeDirect(t, Ops.result_o(t.resultFrame()), Ops.emptyCallSite, Ops.emptyArgList);
return Ops.result_o(t.resultFrame());
}
public static void main(String[] args) {
if (args.length != 3) {
System.err.printf("usage: java CallFromJava <jarfile> <dialect> <expression>\n");
System.err.println("<jarfile>: path to nqp.jar or perl6.jar");
System.err.println("<dialect>: nqp or perl6");
System.err.println("<expression>: a nqp or perl6 expression");
System.exit(1);
}
String jarFile = args[0];
String dialect = args[1];
String expression = args[2];
CallFromJava nqp = new CallFromJava(jarFile, dialect);
nqp.eval(expression);
}
}
Если вы берете исходный код из пакета Rakudo Star (версия 2019-03 на момент написания), обязательно примените следующее исправление (уже исправленное в приведенном выше примере):
< Ops.invokeDirect(t, Ops.findmethod(t, nqpComp, "compile"),
---
> Ops.invokeDirect(t, Ops.findmethod(nqpComp, "compile", t),
Чтобы построить и протестировать пример:
С NQP (не совсем Perl):
cd rakudo-star-yyyy-mm/nqp
javac -cp bin/ examples/CallFromJava.java
java -cp nqp-runtime.jar:3rdparty/asm/asm-4.1.jar:3rdparty/asm/asm-tree-4.1.jar:. examples.CallFromJava nqp.jar nqp 'say(2+2)'
4
Проблема в том, что NQP - это только подмножество Perl6, не предназначенное для непосредственного использования разработчиком Perl6. Предположительно, с полным Perl6 можно было бы сделать что-то вроде:
export PERL6_PREFIX=/usr/local/perl6 # or whatever your perl6 installation prefix is
cd rakudo-star-yyyy-mm/nqp
javac -cp bin/ examples/CallFromJava.java
java -cp $PERL6_PREFIX/share/nqp/runtime/asm-4.1.jar:$PERL6_PREFIX/share/nqp/runtime/asm-tree-4.1.jar:$PERL6_PREFIX/share/nqp/runtime/nqp-runtime.jar:$PERL6_PREFIX/share/perl6/runtime/rakudo-runtime.jar:$PERL6_PREFIX/share/perl6/runtime/perl6.jar:. examples.CallFromJava $PERL6_PREFIX/share/perl6/runtime/perl6.jar perl6 'say 2 + 2'
но мне пока не удалось заставить его работать
Unhandled exception: java.nio.file.NoSuchFileException: Perl6/Grammar
in <anon> (src/vm/jvm/ModuleLoader.nqp:76)
in load_module (src/vm/jvm/ModuleLoader.nqp:58)
in <anon> (gen/jvm/main.nqp)
Я не думаю, что компиляция кода perl6 в байт-код JVM сразу поможет вам, но есть "Eval Server", который использует набор тестов, чтобы не запускать JVM с нуля для каждого из множества тестов. файлы в тестовом наборе спецификаций.
Вы можете найти исходный код сервера eval здесь и, вероятно, украсть у него несколько вещей: https://github.com/perl6/nqp/blob/master/src/vm/jvm/runtime/org/perl6/nqp/tools/EvalServer.java