Skip to content

Commit

Permalink
Bug 1090275: WebServices modules should maintain a whitelist of metho…
Browse files Browse the repository at this point in the history
…ds that are allowed instead of allowing access to any function imported into its namespace

r=dylan,a=glob
  • Loading branch information
dklawren committed Jan 21, 2015
1 parent f5b9cba commit ce85e36
Show file tree
Hide file tree
Showing 10 changed files with 80 additions and 0 deletions.
4 changes: 4 additions & 0 deletions Bugzilla/WebService.pm
Expand Up @@ -23,6 +23,10 @@ use constant LOGIN_EXEMPT => { };
# Methods that can modify data MUST not be listed here.
use constant READ_ONLY => ();

# Whitelist of methods that a client is allowed to access when making
# an API call.
use constant PUBLIC_METHODS => ();

sub login_exempt {
my ($class, $method) = @_;
return $class->LOGIN_EXEMPT->{$method};
Expand Down
18 changes: 18 additions & 0 deletions Bugzilla/WebService/Bug.pm
Expand Up @@ -49,6 +49,24 @@ use constant READ_ONLY => qw(
search
);

use constant PUBLIC_METHODS => qw(
add_attachment
add_comment
attachments
comments
create
fields
get
history
legal_values
possible_duplicates
render_comment
search
update
update_see_also
update_tags
);

######################################################
# Add aliases here for old method name compatibility #
######################################################
Expand Down
9 changes: 9 additions & 0 deletions Bugzilla/WebService/Bugzilla.pm
Expand Up @@ -31,6 +31,15 @@ use constant READ_ONLY => qw(
version
);

use constant PUBLIC_METHODS => qw(
extensions
last_audit_time
parameters
time
timezone
version
);

# Logged-out users do not need to know more than that.
use constant PARAMETERS_LOGGED_OUT => qw(
maintainer
Expand Down
4 changes: 4 additions & 0 deletions Bugzilla/WebService/Classification.pm
Expand Up @@ -19,6 +19,10 @@ use constant READ_ONLY => qw(
get
);

use constant PUBLIC_METHODS => qw(
get
);

sub get {
my ($self, $params) = validate(@_, 'names', 'ids');

Expand Down
5 changes: 5 additions & 0 deletions Bugzilla/WebService/Group.pm
Expand Up @@ -13,6 +13,11 @@ use Bugzilla::Constants;
use Bugzilla::Error;
use Bugzilla::WebService::Util qw(validate translate params_to_objects);

use constant PUBLIC_METHODS => qw(
create
update
);

use constant MAPPED_RETURNS => {
userregexp => 'user_regexp',
isactive => 'is_active'
Expand Down
9 changes: 9 additions & 0 deletions Bugzilla/WebService/Product.pm
Expand Up @@ -23,6 +23,15 @@ use constant READ_ONLY => qw(
get_selectable_products
);

use constant PUBLIC_METHODS => qw(
create
get
get_accessible_products
get_enterable_products
get_selectable_products
update
);

use constant MAPPED_FIELDS => {
has_unconfirmed => 'allows_unconfirmed',
is_open => 'is_active',
Expand Down
6 changes: 6 additions & 0 deletions Bugzilla/WebService/Server/JSONRPC.pm
Expand Up @@ -28,6 +28,7 @@ use Bugzilla::Util qw(correct_urlbase trim disable_utf8);

use HTTP::Message;
use MIME::Base64 qw(decode_base64 encode_base64);
use List::MoreUtils qw(none);

#####################################
# Public JSON::RPC Method Overrides #
Expand Down Expand Up @@ -378,6 +379,11 @@ sub _argument_type_check {
}
}

# Only allowed methods to be used from our whitelist
if (none { $_ eq $method} $pkg->PUBLIC_METHODS) {
ThrowUserError('unknown_method', { method => $self->bz_method_name });
}

# This is the best time to do login checks.
$self->handle_login();

Expand Down
11 changes: 11 additions & 0 deletions Bugzilla/WebService/Server/XMLRPC.pm
Expand Up @@ -17,6 +17,9 @@ if ($ENV{MOD_PERL}) {
}

use Bugzilla::WebService::Constants;
use Bugzilla::Error;

use List::MoreUtils qw(none);

# Allow WebService methods to call XMLRPC::Lite's type method directly
BEGIN {
Expand Down Expand Up @@ -65,6 +68,14 @@ sub handle_login {
my ($self, $classes, $action, $uri, $method) = @_;
my $class = $classes->{$uri};
my $full_method = $uri . "." . $method;
# Only allowed methods to be used from the module's whitelist
my $file = $class;
$file =~ s{::}{/}g;
$file .= ".pm";
require $file;
if (none { $_ eq $method } $class->PUBLIC_METHODS) {
ThrowCodeError('unknown_method', { method => $full_method });
}
$self->SUPER::handle_login($class, $method, $full_method);
return;
}
Expand Down
9 changes: 9 additions & 0 deletions Bugzilla/WebService/User.pm
Expand Up @@ -32,6 +32,15 @@ use constant READ_ONLY => qw(
get
);

use constant PUBLIC_METHODS => qw(
create
get
login
logout
offer_account_by_email
update
);

use constant MAPPED_FIELDS => {
email => 'login',
full_name => 'name',
Expand Down
5 changes: 5 additions & 0 deletions extensions/Example/lib/WebService.pm
Expand Up @@ -11,6 +11,11 @@ use warnings;
use base qw(Bugzilla::WebService);
use Bugzilla::Error;

use constant PUBLIC_METHODS => qw(
hello
throw_an_error
);

# This can be called as Example.hello() from the WebService.
sub hello { return 'Hello!'; }

Expand Down

0 comments on commit ce85e36

Please sign in to comment.