# 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/.
#
# This Source Code Form is "Incompatible With Secondary Licenses", as
# defined by the Mozilla Public License, v. 2.0.
package Bugzilla::Extension::Example;
use 5.10.1;
use strict;
use warnings;
use parent qw(Bugzilla::Extension);
use Bugzilla::Constants;
use Bugzilla::Error;
use Bugzilla::Group;
use Bugzilla::User;
use Bugzilla::User::Setting;
use Bugzilla::Util qw(diff_arrays html_quote remote_ip);
use Bugzilla::Status qw(is_open_state);
use Bugzilla::Install::Filesystem;
use Bugzilla::WebService::Constants;
# This is extensions/Example/lib/Util.pm. I can load this here in my
# Extension.pm only because I have a Config.pm.
use Bugzilla::Extension::Example::Util;
use Data::Dumper;
# See bugmail_relationships.
use constant REL_EXAMPLE => -127;
our $VERSION = '1.0';
sub user_can_administer {
my ($self, $args) = @_;
my $can_administer = $args->{can_administer};
# If you add an option to the admin pages (e.g. by using the Hooks in
# template/en/default/admin/admin.html.tmpl), you may want to allow
# users in another group view admin.cgi
#if (Bugzilla->user->in_group('other_group')) {
# $$can_administer = 1;
#}
}
sub admin_editusers_action {
my ($self, $args) = @_;
my ($vars, $action, $user) = @$args{qw(vars action user)};
my $template = Bugzilla->template;
if ($action eq 'my_action') {
# Allow to restrict the search to any group the user is allowed to bless.
$vars->{'restrictablegroups'} = $user->bless_groups();
$template->process('admin/users/search.html.tmpl', $vars)
|| ThrowTemplateError($template->error());
exit;
}
}
sub attachment_process_data {
my ($self, $args) = @_;
my $type = $args->{attributes}->{mimetype};
my $filename = $args->{attributes}->{filename};
# Make sure images have the correct extension.
# Uncomment the two lines below to make this check effective.
if ($type =~ /^image\/(\w+)$/) {
my $format = $1;
if ($filename =~ /^(.+)(:?\.[^\.]+)$/) {
my $name = $1;
#$args->{attributes}->{filename} = "${name}.$format";
}
else {
# The file has no extension. We append it.
#$args->{attributes}->{filename} .= ".$format";
}
}
}
sub auth_login_methods {
my ($self, $args) = @_;
my $modules = $args->{modules};
if (exists $modules->{Example}) {
$modules->{Example} = 'Bugzilla/Extension/Example/Auth/Login.pm';
}
}
sub auth_verify_methods {
my ($self, $args) = @_;
my $modules = $args->{modules};
if (exists $modules->{Example}) {
$modules->{Example} = 'Bugzilla/Extension/Example/Auth/Verify.pm';
}
}
sub bug_check_can_change_field {
my ($self, $args) = @_;
my ($bug, $field, $new_value, $old_value, $priv_results)
= @$args{qw(bug field new_value old_value priv_results)};
my $user = Bugzilla->user;
# Disallow a bug from being reopened if currently closed unless user
# is in 'admin' group
if ($field eq 'bug_status' && $bug->product_obj->name eq 'Example') {
if (!is_open_state($old_value) && is_open_state($new_value)
&& !$user->in_group('admin'))
{
push(@$priv_results, PRIVILEGES_REQUIRED_EMPOWERED);
return;
}
}
# Disallow a bug's keywords from being edited unless user is the
# reporter of the bug
if ($field eq 'keywords' && $bug->product_obj->name eq 'Example'
&& $user->login ne $bug->reporter->login)
{
push(@$priv_results, PRIVILEGES_REQUIRED_REPORTER);
return;
}
# Allow updating of priority even if user cannot normally edit the bug
# and they are in group 'engineering'
if ($field eq 'priority' && $bug->product_obj->name eq 'Example'
&& $user->in_group('engineering'))
{
push(@$priv_results, PRIVILEGES_REQUIRED_NONE);
return;
}
}
sub bug_columns {
my ($self, $args) = @_;
my $columns = $args->{'columns'};
push (@$columns, "delta_ts AS example")
}
sub bug_end_of_create {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my $bug = $args->{'bug'};
my $timestamp = $args->{'timestamp'};
my $bug_id = $bug->id;
# Uncomment this line to see a line in your webserver's error log whenever
# you file a bug.
# warn "Bug $bug_id has been filed!";
}
sub bug_end_of_create_validators {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my $bug_params = $args->{'params'};
# Uncomment this line below to see a line in your webserver's error log
# containing all validated bug field values every time you file a bug.
# warn Dumper($bug_params);
# This would remove all ccs from the bug, preventing ANY ccs from being
# added on bug creation.
# $bug_params->{cc} = [];
}
sub bug_start_of_update {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my ($bug, $old_bug, $timestamp, $changes) =
@$args{qw(bug old_bug timestamp changes)};
foreach my $field (keys %$changes) {
my $used_to_be = $changes->{$field}->[0];
my $now_it_is = $changes->{$field}->[1];
}
my $old_summary = $old_bug->short_desc;
my $status_message;
if (my $status_change = $changes->{'bug_status'}) {
my $old_status = new Bugzilla::Status({ name => $status_change->[0] });
my $new_status = new Bugzilla::Status({ name => $status_change->[1] });
if ($new_status->is_open && !$old_status->is_open) {
$status_message = "Bug re-opened!";
}
if (!$new_status->is_open && $old_status->is_open) {
$status_message = "Bug closed!";
}
}
my $bug_id = $bug->id;
my $num_changes = scalar keys %$changes;
my $result = "There were $num_changes changes to fields on bug $bug_id"
. " at $timestamp.";
# Uncomment this line to see $result in your webserver's error log whenever
# you update a bug.
# warn $result;
}
sub bug_end_of_update {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my ($bug, $old_bug, $timestamp, $changes) =
@$args{qw(bug old_bug timestamp changes)};
foreach my $field (keys %$changes) {
my $used_to_be = $changes->{$field}->[0];
my $now_it_is = $changes->{$field}->[1];
}
my $old_summary = $old_bug->short_desc;
my $status_message;
if (my $status_change = $changes->{'bug_status'}) {
my $old_status = new Bugzilla::Status({ name => $status_change->[0] });
my $new_status = new Bugzilla::Status({ name => $status_change->[1] });
if ($new_status->is_open && !$old_status->is_open) {
$status_message = "Bug re-opened!";
}
if (!$new_status->is_open && $old_status->is_open) {
$status_message = "Bug closed!";
}
}
my $bug_id = $bug->id;
my $num_changes = scalar keys %$changes;
my $result = "There were $num_changes changes to fields on bug $bug_id"
. " at $timestamp.";
# Uncomment this line to see $result in your webserver's error log whenever
# you update a bug.
# warn $result;
}
sub bug_fields {
my ($self, $args) = @_;
my $fields = $args->{'fields'};
push (@$fields, "example")
}
sub bug_format_comment {
my ($self, $args) = @_;
# This replaces every occurrence of the word "foo" with the word
# "bar"
my $regexes = $args->{'regexes'};
push(@$regexes, { match => qr/\bfoo\b/, replace => 'bar' });
# And this links every occurrence of the word "bar" to example.com,
# but it won't affect "foo"s that have already been turned into "bar"
# above (because each regex is run in order, and later regexes don't modify
# earlier matches, due to some cleverness in Bugzilla's internals).
#
# For example, the phrase "foo bar" would become:
# bar bar
my $bar_match = qr/\b(bar)\b/;
push(@$regexes, { match => $bar_match, replace => \&_replace_bar });
}
# Used by bug_format_comment--see its code for an explanation.
sub _replace_bar {
my $args = shift;
# $match is the first parentheses match in the $bar_match regex
# in bug-format_comment.pl. We get up to 10 regex matches as
# arguments to this function.
my $match = $args->{matches}->[0];
# Remember, you have to HTML-escape any data that you are returning!
$match = html_quote($match);
return qq{$match};
};
sub buglist_columns {
my ($self, $args) = @_;
my $columns = $args->{'columns'};
$columns->{'example'} = { 'name' => 'bugs.delta_ts' , 'title' => 'Example' };
$columns->{'product_desc'} = { 'name' => 'prod_desc.description',
'title' => 'Product Description' };
}
sub buglist_column_joins {
my ($self, $args) = @_;
my $joins = $args->{'column_joins'};
# This column is added using the "buglist_columns" hook
$joins->{'product_desc'} = {
from => 'product_id',
to => 'id',
table => 'products',
as => 'prod_desc',
join => 'INNER',
};
}
sub search_operator_field_override {
my ($self, $args) = @_;
my $operators = $args->{'operators'};
my $original = $operators->{component}->{_non_changed};
$operators->{component} = {
_non_changed => sub { _component_nonchanged($original, @_) }
};
}
sub _component_nonchanged {
my $original = shift;
my ($invocant, $args) = @_;
$invocant->$original($args);
# Actually, it does not change anything in the result,
# just an example.
$args->{term} = $args->{term} . " OR 1=2";
}
sub bugmail_recipients {
my ($self, $args) = @_;
my $recipients = $args->{recipients};
my $bug = $args->{bug};
my $user =
new Bugzilla::User({ name => Bugzilla->params->{'maintainer'} });
if ($bug->id == 1) {
# Uncomment the line below to add the maintainer to the recipients
# list of every bugmail from bug 1 as though that the maintainer
# were on the CC list.
#$recipients->{$user->id}->{+REL_CC} = 1;
# And this line adds the maintainer as though they had the
# "REL_EXAMPLE" relationship from the bugmail_relationships hook below.
#$recipients->{$user->id}->{+REL_EXAMPLE} = 1;
}
}
sub bugmail_relationships {
my ($self, $args) = @_;
my $relationships = $args->{relationships};
$relationships->{+REL_EXAMPLE} = 'Example';
}
sub cgi_headers {
my ($self, $args) = @_;
my $headers = $args->{'headers'};
$headers->{'-x_test_header'} = "Test header from Example extension";
}
sub config_add_panels {
my ($self, $args) = @_;
my $modules = $args->{panel_modules};
$modules->{Example} = "Bugzilla::Extension::Example::Config";
}
sub config_modify_panels {
my ($self, $args) = @_;
my $panels = $args->{panels};
# Add the "Example" auth methods.
my $auth_params = $panels->{'auth'}->{params};
my ($info_class) = grep($_->{name} eq 'user_info_class', @$auth_params);
my ($verify_class) = grep($_->{name} eq 'user_verify_class', @$auth_params);
push(@{ $info_class->{choices} }, 'CGI,Example');
push(@{ $verify_class->{choices} }, 'Example');
push(@$auth_params, { name => 'param_example',
type => 't',
default => 0,
checker => \&check_numeric });
}
sub db_schema_abstract_schema {
my ($self, $args) = @_;
# $args->{'schema'}->{'example_table'} = {
# FIELDS => [
# id => {TYPE => 'SMALLSERIAL', NOTNULL => 1,
# PRIMARYKEY => 1},
# for_key => {TYPE => 'INT3', NOTNULL => 1,
# REFERENCES => {TABLE => 'example_table2',
# COLUMN => 'id',
# DELETE => 'CASCADE'}},
# col_3 => {TYPE => 'varchar(64)', NOTNULL => 1},
# ],
# INDEXES => [
# id_index_idx => {FIELDS => ['col_3'], TYPE => 'UNIQUE'},
# for_id_idx => ['for_key'],
# ],
# };
}
sub email_in_before_parse {
my ($self, $args) = @_;
my $subject = $args->{mail}->header('Subject');
# Correctly extract the bug ID from email subjects of the form [Bug comp/NNN].
if ($subject =~ /\[.*(\d+)\].*/) {
$args->{fields}->{bug_id} = $1;
}
}
sub email_in_after_parse {
my ($self, $args) = @_;
my $reporter = $args->{fields}->{reporter};
my $dbh = Bugzilla->dbh;
# No other check needed if this is a valid regular user.
return if login_to_id($reporter);
# The reporter is not a regular user. We create an account for them,
# but they can only comment on existing bugs.
# This is useful for people who reply by email to bugmails received
# in mailing-lists.
if ($args->{fields}->{bug_id}) {
# WARNING: we return now to skip the remaining code below.
# You must understand that removing this line would make the code
# below effective! Do it only if you are OK with the behavior
# described here.
return;
Bugzilla::User->create({ login_name => $reporter, cryptpassword => '*' });
# For security reasons, delete all fields unrelated to comments.
foreach my $field (keys %{$args->{fields}}) {
next if $field =~ /^(?:bug_id|comment|reporter)$/;
delete $args->{fields}->{$field};
}
}
else {
ThrowUserError('invalid_username', { name => $reporter });
}
}
sub enter_bug_entrydefaultvars {
my ($self, $args) = @_;
my $vars = $args->{vars};
$vars->{'example'} = 1;
}
sub error_catch {
my ($self, $args) = @_;
# Customize the error message displayed when someone tries to access
# page.cgi with an invalid page ID, and keep track of this attempt
# in the web server log.
return unless Bugzilla->error_mode == ERROR_MODE_WEBPAGE;
return unless $args->{error} eq 'bad_page_cgi_id';
my $page_id = $args->{vars}->{page_id};
my $login = Bugzilla->user->identity || "Someone";
warn "$login attempted to access page.cgi with id = $page_id";
my $page = $args->{message};
my $new_error_msg = "Ah ah, you tried to access $page_id? Good try!";
$new_error_msg = html_quote($new_error_msg);
# There are better tools to parse an HTML page, but it's just an example.
# Since Perl 5.16, we can no longer write "class" inside look-behind
# assertions, because "ss" is also seen as the german ß character, which
# makes Perl 5.16 complain. The right fix is to use the /aa modifier,
# but it's only understood since Perl 5.14. So the workaround is to write
# "clas[s]" instead of "class". Stupid and ugly hack, but it works with
# all Perl versions.
$$page =~ s/(?<=
).*(?=<\/td>)/$new_error_msg/si;
}
sub flag_end_of_update {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my $flag_params = $args;
my ($object, $timestamp, $old_flags, $new_flags) =
@$flag_params{qw(object timestamp old_flags new_flags)};
my ($removed, $added) = diff_arrays($old_flags, $new_flags);
my ($granted, $denied) = (0, 0);
foreach my $new_flag (@$added) {
$granted++ if $new_flag =~ /\+$/;
$denied++ if $new_flag =~ /-$/;
}
my $bug_id = $object->isa('Bugzilla::Bug') ? $object->id
: $object->bug_id;
my $result = "$granted flags were granted and $denied flags were denied"
. " on bug $bug_id at $timestamp.";
# Uncomment this line to see $result in your webserver's error log whenever
# you update flags.
# warn $result;
}
sub group_before_delete {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my $group = $args->{'group'};
my $group_id = $group->id;
# Uncomment this line to see a line in your webserver's error log whenever
# you file a bug.
# warn "Group $group_id is about to be deleted!";
}
sub group_end_of_create {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my $group = $args->{'group'};
my $group_id = $group->id;
# Uncomment this line to see a line in your webserver's error log whenever
# you create a new group.
#warn "Group $group_id has been created!";
}
sub group_end_of_update {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my ($group, $changes) = @$args{qw(group changes)};
foreach my $field (keys %$changes) {
my $used_to_be = $changes->{$field}->[0];
my $now_it_is = $changes->{$field}->[1];
}
my $group_id = $group->id;
my $num_changes = scalar keys %$changes;
my $result =
"There were $num_changes changes to fields on group $group_id.";
# Uncomment this line to see $result in your webserver's error log whenever
# you update a group.
#warn $result;
}
sub install_before_final_checks {
my ($self, $args) = @_;
print "Install-before_final_checks hook\n" unless $args->{silent};
# Add a new user setting like this:
#
# add_setting('product_chooser', # setting name
# ['pretty', 'full', 'small'], # options
# 'pretty'); # default
#
# To add descriptions for the setting and choices, add extra values to
# the hash defined in global/setting-descs.none.tmpl. Do this in a hook:
# hook/global/setting-descs-settings.none.tmpl .
}
sub install_filesystem {
my ($self, $args) = @_;
my $create_dirs = $args->{'create_dirs'};
my $recurse_dirs = $args->{'recurse_dirs'};
my $htaccess = $args->{'htaccess'};
# Create a new directory in datadir specifically for this extension.
# The directory will need to allow files to be created by the extension
# code as well as allow the webserver to server content from it.
# my $data_path = bz_locations->{'datadir'} . "/" . __PACKAGE__->NAME;
# $create_dirs->{$data_path} = Bugzilla::Install::Filesystem::DIR_CGI_WRITE;
# Update the permissions of any files and directories that currently reside
# in the extension's directory.
# $recurse_dirs->{$data_path} = {
# files => Bugzilla::Install::Filesystem::CGI_READ,
# dirs => Bugzilla::Install::Filesystem::DIR_CGI_WRITE
# };
# Create a htaccess file that allows specific content to be served from the
# extension's directory.
# $htaccess->{"$data_path/.htaccess"} = {
# perms => Bugzilla::Install::Filesystem::WS_SERVE,
# contents => Bugzilla::Install::Filesystem::HT_DEFAULT_DENY
# };
}
sub install_update_db {
my $dbh = Bugzilla->dbh;
# $dbh->bz_add_column('example', 'new_column',
# {TYPE => 'INT2', NOTNULL => 1, DEFAULT => 0});
# $dbh->bz_add_index('example', 'example_new_column_idx', [qw(value)]);
}
sub install_update_db_fielddefs {
my $dbh = Bugzilla->dbh;
# $dbh->bz_add_column('fielddefs', 'example_column',
# {TYPE => 'MEDIUMTEXT', NOTNULL => 1, DEFAULT => ''});
}
sub job_map {
my ($self, $args) = @_;
my $job_map = $args->{job_map};
# This adds the named class (an instance of TheSchwartz::Worker) as a
# handler for when a job is added with the name "some_task".
$job_map->{'some_task'} = 'Bugzilla::Extension::Example::Job::SomeClass';
# Schedule a job like this:
# my $queue = Bugzilla->job_queue();
# $queue->insert('some_task', { some_parameter => $some_variable });
}
sub mailer_before_send {
my ($self, $args) = @_;
my $email = $args->{email};
# If you add a header to an email, it's best to start it with
# 'X-Bugzilla-' so that you don't conflict with
# other extensions.
$email->header_set('X-Bugzilla-Example-Header', 'Example');
}
sub object_before_create {
my ($self, $args) = @_;
my $class = $args->{'class'};
my $object_params = $args->{'params'};
# Note that this is a made-up class, for this example.
if ($class->isa('Bugzilla::ExampleObject')) {
warn "About to create an ExampleObject!";
warn "Got the following parameters: "
. join(', ', keys(%$object_params));
}
}
sub object_before_delete {
my ($self, $args) = @_;
my $object = $args->{'object'};
# Note that this is a made-up class, for this example.
if ($object->isa('Bugzilla::ExampleObject')) {
my $id = $object->id;
warn "An object with id $id is about to be deleted!";
}
}
sub object_before_set {
my ($self, $args) = @_;
my ($object, $field, $value) = @$args{qw(object field value)};
# Note that this is a made-up class, for this example.
if ($object->isa('Bugzilla::ExampleObject')) {
warn "The field $field is changing from " . $object->{$field}
. " to $value!";
}
}
sub object_columns {
my ($self, $args) = @_;
my ($class, $columns) = @$args{qw(class columns)};
if ($class->isa('Bugzilla::ExampleObject')) {
push(@$columns, 'example');
}
}
sub object_end_of_create {
my ($self, $args) = @_;
my $class = $args->{'class'};
my $object = $args->{'object'};
warn "Created a new $class object!";
}
sub object_end_of_create_validators {
my ($self, $args) = @_;
my $class = $args->{'class'};
my $object_params = $args->{'params'};
# Note that this is a made-up class, for this example.
if ($class->isa('Bugzilla::ExampleObject')) {
# Always set example_field to 1, even if the validators said otherwise.
$object_params->{example_field} = 1;
}
}
sub object_end_of_set {
my ($self, $args) = @_;
my ($object, $field) = @$args{qw(object field)};
# Note that this is a made-up class, for this example.
if ($object->isa('Bugzilla::ExampleObject')) {
warn "The field $field has changed to " . $object->{$field};
}
}
sub object_end_of_set_all {
my ($self, $args) = @_;
my $object = $args->{'object'};
my $object_params = $args->{'params'};
# Note that this is a made-up class, for this example.
if ($object->isa('Bugzilla::ExampleObject')) {
if ($object_params->{example_field} == 1) {
$object->{example_field} = 1;
}
}
}
sub object_end_of_update {
my ($self, $args) = @_;
my ($object, $old_object, $changes) =
@$args{qw(object old_object changes)};
# Note that this is a made-up class, for this example.
if ($object->isa('Bugzilla::ExampleObject')) {
if (defined $changes->{'name'}) {
my ($old, $new) = @{ $changes->{'name'} };
print "The name field changed from $old to $new!";
}
}
}
sub object_update_columns {
my ($self, $args) = @_;
my ($object, $columns) = @$args{qw(object columns)};
if ($object->isa('Bugzilla::ExampleObject')) {
push(@$columns, 'example');
}
}
sub object_validators {
my ($self, $args) = @_;
my ($class, $validators) = @$args{qw(class validators)};
if ($class->isa('Bugzilla::Bug')) {
# This is an example of adding a new validator.
# See the _check_example subroutine below.
$validators->{example} = \&_check_example;
# This is an example of overriding an existing validator.
# See the check_short_desc validator below.
my $original = $validators->{short_desc};
$validators->{short_desc} = sub { _check_short_desc($original, @_) };
}
}
sub _check_example {
my ($invocant, $value, $field) = @_;
warn "I was called to validate the value of $field.";
warn "The value of $field that I was passed in is: $value";
# Make the value always be 1.
my $fixed_value = 1;
return $fixed_value;
}
sub _check_short_desc {
my $original = shift;
my $invocant = shift;
my $value = $invocant->$original(@_);
if ($value !~ /example/i) {
# Use this line to make Bugzilla throw an error every time
# you try to file a bug or update a bug without the word "example"
# in the summary.
if (0) {
ThrowUserError('example_short_desc_invalid');
}
}
return $value;
}
sub page_before_template {
my ($self, $args) = @_;
my ($vars, $page) = @$args{qw(vars page_id)};
# You can see this hook in action by loading page.cgi?id=example.html
if ($page eq 'example.html') {
$vars->{cgi_variables} = { Bugzilla->cgi->Vars };
}
}
sub path_info_whitelist {
my ($self, $args) = @_;
my $whitelist = $args->{whitelist};
push(@$whitelist, "page.cgi");
}
sub post_bug_after_creation {
my ($self, $args) = @_;
my $vars = $args->{vars};
$vars->{'example'} = 1;
}
sub product_confirm_delete {
my ($self, $args) = @_;
my $vars = $args->{vars};
$vars->{'example'} = 1;
}
sub product_end_of_create {
my ($self, $args) = @_;
my $product = $args->{product};
# For this example, any lines of code that actually make changes to your
# database have been commented out.
# This section will take a group that exists in your installation
# (possible called test_group) and automatically makes the new
# product hidden to only members of the group. Just remove
# the restriction if you want the new product to be public.
my $example_group = new Bugzilla::Group({ name => 'example_group' });
if ($example_group) {
$product->set_group_controls($example_group,
{ entry => 1,
membercontrol => CONTROLMAPMANDATORY,
othercontrol => CONTROLMAPMANDATORY });
# $product->update();
}
# This section will automatically add a default component
# to the new product called 'No Component'.
my $default_assignee = new Bugzilla::User(
{ name => Bugzilla->params->{maintainer} });
if ($default_assignee) {
# Bugzilla::Component->create(
# { name => 'No Component',
# product => $product,
# description => 'Select this component if one does not ' .
# 'exist in the current list of components',
# initialowner => $default_assignee });
}
}
sub quicksearch_map {
my ($self, $args) = @_;
my $map = $args->{'map'};
# This demonstrates adding a shorter alias for a long custom field name.
$map->{'impact'} = $map->{'cf_long_field_name_for_impact_field'};
}
sub sanitycheck_check {
my ($self, $args) = @_;
my $dbh = Bugzilla->dbh;
my $sth;
my $status = $args->{'status'};
# Check that all users are Australian
$status->('example_check_au_user');
$sth = $dbh->prepare("SELECT userid, login_name
FROM profiles
WHERE login_name NOT LIKE '%.au'");
$sth->execute;
my $seen_nonau = 0;
while (my ($userid, $login, $numgroups) = $sth->fetchrow_array) {
$status->('example_check_au_user_alert',
{ userid => $userid, login => $login },
'alert');
$seen_nonau = 1;
}
$status->('example_check_au_user_prompt') if $seen_nonau;
}
sub sanitycheck_repair {
my ($self, $args) = @_;
my $cgi = Bugzilla->cgi;
my $dbh = Bugzilla->dbh;
my $status = $args->{'status'};
if ($cgi->param('example_repair_au_user')) {
$status->('example_repair_au_user_start');
#$dbh->do("UPDATE profiles
# SET login_name = CONCAT(login_name, '.au')
# WHERE login_name NOT LIKE '%.au'");
$status->('example_repair_au_user_end');
}
}
sub template_before_create {
my ($self, $args) = @_;
my $config = $args->{'config'};
# This will be accessible as "example_global_variable" in every
# template in Bugzilla. See Bugzilla/Template.pm's create() function
# for more things that you can set.
$config->{VARIABLES}->{example_global_variable} = sub { return 'value' };
}
sub template_after_create {
my ( $self, $args ) = @_;
my $context = $args->{template}->context;
# define a pluck method on template toolkit lists.
$context->define_vmethod(
list => pluck => sub {
my ( $list, $field ) = @_;
return [ map { $_->$field } @$list ];
}
);
}
sub template_before_process {
my ($self, $args) = @_;
my ($vars, $file, $context) = @$args{qw(vars file context)};
if ($file eq 'bug/edit.html.tmpl') {
$vars->{'viewing_the_bug_form'} = 1;
}
}
sub user_check_account_creation {
my ($self, $args) = @_;
my $login = $args->{login};
my $ip = remote_ip();
# Log all requests.
warn "USER ACCOUNT CREATION REQUEST FOR $login ($ip)";
# Reject requests based on their email address.
if ($login =~ /\@evil\.com$/) {
ThrowUserError('account_creation_restricted');
}
# Reject requests based on their IP address.
if ($ip =~ /^192\.168\./) {
ThrowUserError('account_creation_restricted');
}
}
sub user_preferences {
my ($self, $args) = @_;
my $tab = $args->{current_tab};
my $save = $args->{save_changes};
my $handled = $args->{handled};
return unless $tab eq 'my_tab';
my $value = Bugzilla->input_params->{'example_pref'};
if ($save) {
# Validate your data and update the DB accordingly.
$value =~ s/\s+/:/g;
}
$args->{'vars'}->{example_pref} = $value;
# Set the 'handled' scalar reference to true so that the caller
# knows the panel name is valid and that an extension took care of it.
$$handled = 1;
}
sub webservice {
my ($self, $args) = @_;
my $dispatch = $args->{dispatch};
$dispatch->{Example} = "Bugzilla::Extension::Example::WebService";
}
sub webservice_error_codes {
my ($self, $args) = @_;
my $error_map = $args->{error_map};
$error_map->{'example_my_error'} = 10001;
}
sub webservice_status_code_map {
my ($self, $args) = @_;
my $status_code_map = $args->{status_code_map};
# Uncomment this line to override the status code for the
# error 'object_does_not_exist' to STATUS_BAD_REQUEST
#$status_code_map->{51} = STATUS_BAD_REQUEST;
}
sub webservice_before_call {
my ($self, $args) = @_;
# This code doesn't actually *do* anything, it's just here to show you
# how to use this hook.
my $method = $args->{method};
my $full_method = $args->{full_method};
# Uncomment this line to see a line in your webserver's error log whenever
# a webservice call is made
#warn "RPC call $full_method made by ",
# Bugzilla->user->login || 'an anonymous user', "\n";
}
sub webservice_fix_credentials {
my ($self, $args) = @_;
my $rpc = $args->{'rpc'};
my $params = $args->{'params'};
# Allow user to pass in username=foo&password=bar
if (exists $params->{'username'} && exists $params->{'password'}) {
$params->{'Bugzilla_login'} = $params->{'username'};
$params->{'Bugzilla_password'} = $params->{'password'};
}
}
sub webservice_rest_request {
my ($self, $args) = @_;
my $rpc = $args->{'rpc'};
my $params = $args->{'params'};
# Internally we may have a field called 'cf_test_field' but we allow users
# to use the shorter 'test_field' name.
if (exists $params->{'test_field'}) {
$params->{'test_field'} = delete $params->{'cf_test_field'};
}
}
sub webservice_rest_resources {
my ($self, $args) = @_;
my $rpc = $args->{'rpc'};
my $resources = $args->{'resources'};
# Add a new resource that allows for /rest/example/hello
# to call Example.hello
$resources->{'Bugzilla::Extension::Example::WebService'} = [
qr{^/example/hello$}, {
GET => {
method => 'hello',
}
}
];
}
sub webservice_rest_response {
my ($self, $args) = @_;
my $rpc = $args->{'rpc'};
my $result = $args->{'result'};
my $response = $args->{'response'};
# Convert a list of bug hashes to a single bug hash if only one is
# being returned.
if (ref $$result eq 'HASH'
&& exists $$result->{'bugs'}
&& scalar @{ $$result->{'bugs'} } == 1)
{
$$result = $$result->{'bugs'}->[0];
}
}
# This must be the last line of your extension.
__PACKAGE__->NAME;
|