2020-07-09 08:21:54 +02:00
|
|
|
# File: Plang.pm
|
|
|
|
# Author: pragma-
|
|
|
|
#
|
|
|
|
# Purpose: Simplified scripting language for creating advanced PBot factoids
|
|
|
|
# and interacting with various internal PBot APIs.
|
|
|
|
|
|
|
|
# This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
|
|
|
package Plugins::Plang;
|
|
|
|
use parent 'Plugins::Plugin';
|
|
|
|
|
|
|
|
use warnings; use strict;
|
|
|
|
use feature 'unicode_strings';
|
|
|
|
|
|
|
|
use Getopt::Long qw(GetOptionsFromArray);
|
|
|
|
|
|
|
|
sub initialize {
|
|
|
|
my ($self, %conf) = @_;
|
|
|
|
|
2020-07-21 04:18:05 +02:00
|
|
|
# load Plang modules
|
2020-07-09 08:21:54 +02:00
|
|
|
my $path = $self->{pbot}->{registry}->get_value('general', 'plang_dir') // 'Plang';
|
|
|
|
unshift @INC, $path if not grep { $_ eq $path } @INC;
|
|
|
|
|
2020-07-21 04:18:05 +02:00
|
|
|
# require all the Plang .pm modules so Module::Refresh can reload them without
|
|
|
|
# needing to restart PBot
|
|
|
|
require "$path/Interpreter.pm";
|
|
|
|
require "$path/AstInterpreter.pm";
|
|
|
|
require "$path/Grammar.pm";
|
|
|
|
require "$path/Parser.pm";
|
|
|
|
require "$path/Lexer.pm";
|
2020-07-19 04:48:05 +02:00
|
|
|
|
2020-07-12 06:15:54 +02:00
|
|
|
# regset plang.debug 0-10 -- Plugin must be reloaded for this value to take effect.
|
2020-07-09 08:21:54 +02:00
|
|
|
my $debug = $self->{pbot}->{registry}->get_value('plang', 'debug') // 0;
|
2020-07-12 06:15:54 +02:00
|
|
|
|
|
|
|
# create our Plang interpreter object
|
2020-07-09 08:21:54 +02:00
|
|
|
$self->{plang} = Plang::Interpreter->new(embedded => 1, debug => $debug);
|
|
|
|
|
2020-07-19 04:48:05 +02:00
|
|
|
# register some PBot-specific built-in functions
|
2020-07-20 22:04:35 +02:00
|
|
|
$self->{plang}->{interpreter}->add_builtin_function('factset',
|
2020-07-13 02:36:32 +02:00
|
|
|
# parameters are [['param1 name', default arg], ['param2 name', default arg], ...]
|
2020-07-13 07:09:59 +02:00
|
|
|
[['namespace', undef], ['keyword', undef], ['text', undef]],
|
2020-07-13 06:18:17 +02:00
|
|
|
sub { $self->set_factoid(@_) });
|
2020-07-13 02:36:32 +02:00
|
|
|
|
2020-07-20 22:04:35 +02:00
|
|
|
$self->{plang}->{interpreter}->add_builtin_function('factget',
|
2020-07-13 07:09:59 +02:00
|
|
|
[['namespace', undef], ['keyword', undef], ['meta', ['STRING', 'action']]],
|
2020-07-13 06:18:17 +02:00
|
|
|
sub { $self->get_factoid(@_) });
|
2020-07-13 02:36:32 +02:00
|
|
|
|
2020-07-20 22:04:35 +02:00
|
|
|
$self->{plang}->{interpreter}->add_builtin_function('factappend',
|
2020-07-13 07:09:59 +02:00
|
|
|
[['namespace', undef], ['keyword', undef], ['text', undef]],
|
2020-07-13 06:18:17 +02:00
|
|
|
sub { $self->append_factoid(@_) });
|
2020-07-12 06:15:54 +02:00
|
|
|
|
2020-07-19 04:48:05 +02:00
|
|
|
# override the built-in `print` function to send to our output buffer instead
|
2020-07-20 22:04:35 +02:00
|
|
|
$self->{plang}->{interpreter}->add_builtin_function('print',
|
2020-07-22 01:56:38 +02:00
|
|
|
[['expr', undef], ['end', ['STRING', "\n"]]],
|
2020-07-19 04:48:05 +02:00
|
|
|
sub { $self->print_override(@_) });
|
|
|
|
|
2020-07-12 06:15:54 +02:00
|
|
|
# register the `plang` command
|
2020-07-09 08:21:54 +02:00
|
|
|
$self->{pbot}->{commands}->register(sub { $self->cmd_plang(@_) }, "plang", 0);
|
2020-07-20 22:37:41 +02:00
|
|
|
|
|
|
|
# register the `plangrepl` command (does not reset environment)
|
|
|
|
$self->{pbot}->{commands}->register(sub { $self->cmd_plangrepl(@_) }, "plangrepl", 0);
|
2020-07-09 08:21:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sub unload {
|
|
|
|
my $self = shift;
|
|
|
|
$self->{pbot}->{commands}->unregister("plang");
|
|
|
|
}
|
|
|
|
|
|
|
|
sub cmd_plang {
|
|
|
|
my ($self, $context) = @_;
|
|
|
|
|
2020-07-13 03:53:57 +02:00
|
|
|
my $usage = "Usage: plang <code>; see https://github.com/pragma-/Plang";
|
2020-07-12 06:15:54 +02:00
|
|
|
return $usage if not length $context->{arguments};
|
|
|
|
|
2020-07-20 11:46:01 +02:00
|
|
|
$self->{output} = ""; # collect output of the embedded Plang program
|
|
|
|
my $result = $self->{plang}->interpret_string($context->{arguments});
|
2020-07-12 06:15:54 +02:00
|
|
|
|
|
|
|
# check to see if we need to append final result to output
|
2020-07-16 06:47:03 +02:00
|
|
|
$self->{output} .= $self->{plang}->{interpreter}->output_value($result) if defined $result->[1];
|
2020-07-12 06:15:54 +02:00
|
|
|
|
2020-07-13 02:36:32 +02:00
|
|
|
# return the output
|
2020-07-12 06:15:54 +02:00
|
|
|
return length $self->{output} ? $self->{output} : "No output.";
|
|
|
|
}
|
|
|
|
|
2020-07-20 22:37:41 +02:00
|
|
|
sub cmd_plangrepl {
|
|
|
|
my ($self, $context) = @_;
|
|
|
|
|
|
|
|
my $usage = "Usage: plangrepl <code>; see https://github.com/pragma-/Plang";
|
|
|
|
return $usage if not length $context->{arguments};
|
|
|
|
|
|
|
|
$self->{output} = ""; # collect output of the embedded Plang program
|
|
|
|
my $result = $self->{plang}->interpret_string($context->{arguments}, repl => 1);
|
|
|
|
|
|
|
|
# check to see if we need to append final result to output
|
|
|
|
$self->{output} .= $self->{plang}->{interpreter}->output_value($result, repl => 1) if defined $result->[1];
|
|
|
|
|
|
|
|
# return the output
|
|
|
|
return length $self->{output} ? $self->{output} : "No output.";
|
|
|
|
}
|
|
|
|
|
2020-07-19 04:48:05 +02:00
|
|
|
# overridden `print` built-in
|
2020-07-20 11:46:01 +02:00
|
|
|
|
2020-07-19 04:48:05 +02:00
|
|
|
sub print_override {
|
2020-07-22 01:41:45 +02:00
|
|
|
my ($self, $plang, $context, $name, $arguments) = @_;
|
2020-07-22 01:56:38 +02:00
|
|
|
my ($expr, $end) = ($plang->output_value($arguments->[0]), $arguments->[1]->[1]);
|
|
|
|
$self->{output} .= "$expr$end";
|
2020-07-19 23:49:17 +02:00
|
|
|
return ['NIL', undef];
|
2020-07-19 04:48:05 +02:00
|
|
|
}
|
|
|
|
|
2020-07-12 06:15:54 +02:00
|
|
|
# our custom PBot built-in functions for Plang
|
|
|
|
|
2020-07-13 06:18:17 +02:00
|
|
|
sub is_locked {
|
|
|
|
my ($self, $channel, $keyword) = @_;
|
|
|
|
return $self->{pbot}->{factoids}->get_meta($channel, $keyword, 'locked');
|
|
|
|
}
|
|
|
|
|
2020-07-12 06:15:54 +02:00
|
|
|
sub get_factoid {
|
2020-07-22 01:41:45 +02:00
|
|
|
my ($self, $plang, $context, $name, $arguments) = @_;
|
2020-07-13 07:09:59 +02:00
|
|
|
my ($namespace, $keyword, $meta) = ($arguments->[0]->[1], $arguments->[1]->[1], $arguments->[2]->[1]);
|
|
|
|
my $result = $self->{pbot}->{factoids}->get_meta($namespace, $keyword, $meta);
|
2020-07-13 06:18:17 +02:00
|
|
|
return ['STRING', $result];
|
2020-07-12 06:15:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sub set_factoid {
|
2020-07-22 01:41:45 +02:00
|
|
|
my ($self, $plang, $context, $name, $arguments) = @_;
|
2020-07-13 07:09:59 +02:00
|
|
|
my ($namespace, $keyword, $text) = ($arguments->[0]->[1], $arguments->[1]->[1], $arguments->[2]->[1]);
|
|
|
|
return ['ERROR', "Factoid $namespace.$keyword is locked. Cannot set."] if $self->is_locked($namespace, $keyword);
|
|
|
|
$self->{pbot}->{factoids}->add_factoid('text', $namespace, 'Plang', $keyword, $text);
|
2020-07-13 06:18:17 +02:00
|
|
|
return ['STRING', $text];
|
2020-07-12 06:15:54 +02:00
|
|
|
}
|
2020-07-09 08:21:54 +02:00
|
|
|
|
2020-07-12 06:15:54 +02:00
|
|
|
sub append_factoid {
|
2020-07-22 01:41:45 +02:00
|
|
|
my ($self, $plang, $context, $name, $arguments) = @_;
|
2020-07-13 07:09:59 +02:00
|
|
|
my ($namespace, $keyword, $text) = ($arguments->[0]->[1], $arguments->[1]->[1], $arguments->[2]->[1]);
|
|
|
|
return ['ERROR', "Factoid $namespace.$keyword is locked. Cannot append."] if $self->is_locked($namespace, $keyword);
|
|
|
|
my $action = $self->{pbot}->{factoids}->get_meta($namespace, $keyword, 'action');
|
2020-07-13 06:18:17 +02:00
|
|
|
$action = "" if not defined $action;
|
|
|
|
$action .= $text;
|
2020-07-13 07:09:59 +02:00
|
|
|
$self->{pbot}->{factoids}->add_factoid('text', $namespace, 'Plang', $keyword, $action);
|
2020-07-13 06:18:17 +02:00
|
|
|
return ['STRING', $action];
|
2020-07-09 08:21:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
1;
|