mirror of
https://github.com/pragma-/pbot.git
synced 2024-11-06 20:19:33 +01:00
130 lines
4.0 KiB
Perl
130 lines
4.0 KiB
Perl
# 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) = @_;
|
|
|
|
my $path = $self->{pbot}->{registry}->get_value('general', 'plang_dir') // 'Plang';
|
|
unshift @INC, $path if not grep { $_ eq $path } @INC;
|
|
require "$path/Interpreter.pm";
|
|
|
|
# regset plang.debug 0-10 -- Plugin must be reloaded for this value to take effect.
|
|
my $debug = $self->{pbot}->{registry}->get_value('plang', 'debug') // 0;
|
|
|
|
# create our Plang interpreter object
|
|
$self->{plang} = Plang::Interpreter->new(embedded => 1, debug => $debug);
|
|
|
|
# register some built-in functions
|
|
$self->{plang}->{interpreter}->add_function_builtin('factset',
|
|
# parameters are [['param1 name', default arg], ['param2 name', default arg], ...]
|
|
[['channel', undef], ['keyword', undef], ['text', undef]],
|
|
\&set_factoid);
|
|
|
|
$self->{plang}->{interpreter}->add_function_builtin('factget',
|
|
[['channel', undef], ['keyword', undef]],
|
|
\&get_factoid);
|
|
|
|
$self->{plang}->{interpreter}->add_function_builtin('factappend',
|
|
[['channel', undef], ['keyword', undef], ['text', undef]],
|
|
\&append_factoid);
|
|
|
|
# register the `plang` command
|
|
$self->{pbot}->{commands}->register(sub { $self->cmd_plang(@_) }, "plang", 0);
|
|
}
|
|
|
|
sub unload {
|
|
my $self = shift;
|
|
$self->{pbot}->{commands}->unregister("plang");
|
|
}
|
|
|
|
sub cmd_plang {
|
|
my ($self, $context) = @_;
|
|
|
|
my $usage = "plang <Plang code>; see https://github.com/pragma-/Plang";
|
|
|
|
return $usage if not length $context->{arguments};
|
|
|
|
# run() returns result of the final statement
|
|
my $result = $self->run($context->{arguments});
|
|
|
|
# check to see if we need to append final result to output
|
|
$self->{output} .= $result->[1] if defined $result->[1];
|
|
|
|
# return the output
|
|
return length $self->{output} ? $self->{output} : "No output.";
|
|
}
|
|
|
|
# run an embedded plang program
|
|
sub run {
|
|
my ($self, $code) = @_;
|
|
|
|
# parse the code into an ast
|
|
my $ast = $self->{plang}->parse_string($code);
|
|
return if not defined $ast;
|
|
|
|
# create a new environment for a Plang program
|
|
my $context = $self->{plang}->{interpreter}->init_program;
|
|
|
|
# grab our program's statements
|
|
my $program = $ast->[0];
|
|
my $statements = $program->[1];
|
|
|
|
$self->{output} = ""; # collect output of the embedded Plang program
|
|
my $result; # result of the final statement
|
|
|
|
eval {
|
|
# interpret the statements
|
|
foreach my $node (@$statements) {
|
|
my $ins = $node->[0];
|
|
|
|
if ($ins eq 'STMT') {
|
|
$result = $self->{plang}->{interpreter}->statement($context, $node->[1]);
|
|
}
|
|
}
|
|
};
|
|
|
|
if ($@) {
|
|
$self->{output} .= $@;
|
|
return;
|
|
}
|
|
|
|
return $result; # return result of the final statement
|
|
}
|
|
|
|
# our custom PBot built-in functions for Plang
|
|
|
|
sub get_factoid {
|
|
my ($plang, $name, $arguments) = @_;
|
|
my ($channel, $keyword) = ($arguments->[0]->[1], $arguments->[1]->[1]);
|
|
return ['STRING', "get_factoid: channel: [$channel], keyword: [$keyword]"];
|
|
}
|
|
|
|
sub set_factoid {
|
|
my ($plang, $name, $arguments) = @_;
|
|
my ($channel, $keyword, $text) = ($arguments->[0]->[1], $arguments->[1]->[1], $arguments->[2]->[1]);
|
|
return ['STRING', "set_factoid: channel: [$channel], keyword: [$keyword], text: [$text]"];
|
|
}
|
|
|
|
sub append_factoid {
|
|
my ($plang, $name, $arguments) = @_;
|
|
my ($channel, $keyword, $text) = ($arguments->[0]->[1], $arguments->[1]->[1], $arguments->[2]->[1]);
|
|
return ['STRING', "append_factoid: channel: [$channel], keyword: [$keyword], text: [$text]"];
|
|
}
|
|
|
|
1;
|