1

我正在使用 zabbix 并编写一个与 zabbix api 交互的界面。由于 zabbix 公开了一个 jsonrpc 接口,我决定使用 MojoX::JSON::RPC::Service。我遇到的问题是我现在面临与使用 Mojolicious::Controllers 编写的其他服务交互,他们期望 Mojolicious::Controller 对象。使用 MojoX::JSON::RPC::Service 时没有可用的 Mojolicious::Controller 对象。

my $obj = $rpc_obj->register(
               'retrieve',
               sub {
                    # do stuff
               },
               { with_mojo_tx => 1 }
);

这注册了一条名为“检索”的路线。当访问路由并运行匿名子例程时,子例程只能访问 Mojo::Transaction::HTTP 对象。

因此,我无法访问该应用程序来使用插件以及 Mojolicious 提供的存储和其他东西。有没有办法将 Mojolicious::Controller 与 MojoX::JSON::RPC::Service 结合起来?

我可以重写它以使用 Mojolicious::Controller 但如果可能的话我会尽量避免这种情况。

4

1 回答 1

1

您应该考虑使用MojoX::JSON::RPC::Dispatcher,因为它继承了Mojolicious::Controller的所有属性

概要:

# lib/your-application.pm

use base 'Mojolicious';
use MojoX::JSON::RPC::Service;

sub startup {
    my $self = shift;
    my $svc = MojoX::JSON::RPC::Service->new;

    $svc->register(
        'sum',
        sub {
            my @params = @_;
            my $sum = 0;
            $sum += $_ for @params;
            return $sum;
        }
    );

    $self->plugin(
        'json_rpc_dispatcher',
        services => {
           '/jsonrpc' => $svc
        }
    );
}

[更新] 钩子示例:

package Application;
use Mojo::Base 'Mojolicious';
use Application::Firewall;

# This method will run once at server start
sub startup {
    my $app = shift;

    # Routes
    my $r = $app->routes;

    # Validation Middleware
    $app->hook(
        before_dispatch => sub {
            my $self = shift;
            my $data = $self->req->params->to_hash;
            my $vald = Application::Firewall->new($data);

            # mojolicious bug at the time of coding
            delete $data->{""} if defined $data->{""};

            $app->{input} = {};

            if ( keys %{$data} ) {

                # validation the submitted data
                unless ( $vald->validate( keys %{$data} ) ) {
                    $self->render(
                        text   => join( "", @{ $vald->errors } ),
                        status => 500
                    );
                    return 0;
                }

                # Helper (optional)
                # create a helper to get access to the transformed data
                # if your validation rules had/has filters
                # Note! due to a bug in the params function we must do this
                # (... i know, so what)
                $app->{input} = {
                    map { $_ => $vald->{fields}->{$_}->{value} }
                      keys %{ $vald->{fields} }
                };

            }

            return 1;
        }
    );

    # Normal route to controller * with auto-matic input validation *
    $r->route('/')->to(
        cb => sub {
            my $self = shift;
            $self->render(
                text => 'Hello ' . ( $app->{input}->{foobar} || 'World' ),
                status => 200
            );
        }
    );
}

1;
于 2012-05-10T20:34:16.517 回答