enter_bug.cgi 19.5 KB
Newer Older
1
#!/usr/bin/perl -wT
2
# -*- Mode: perl; indent-tabs-mode: nil -*-
terry%netscape.com's avatar
terry%netscape.com committed
3
#
4 5 6 7 8 9 10 11 12 13
# The contents of this file are subject to the Mozilla Public
# License Version 1.1 (the "License"); you may not use this file
# except in compliance with the License. You may obtain a copy of
# the License at http://www.mozilla.org/MPL/
#
# Software distributed under the License is distributed on an "AS
# IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
# implied. See the License for the specific language governing
# rights and limitations under the License.
#
terry%netscape.com's avatar
terry%netscape.com committed
14
# The Original Code is the Bugzilla Bug Tracking System.
15
# 
terry%netscape.com's avatar
terry%netscape.com committed
16
# The Initial Developer of the Original Code is Netscape Communications
17 18 19
# Corporation. Portions created by Netscape are Copyright (C) 1998
# Netscape Communications Corporation. All Rights Reserved.
# 
terry%netscape.com's avatar
terry%netscape.com committed
20
# Contributor(s): Terry Weissman <terry@mozilla.org>
21
#                 Dave Miller <justdave@syndicomm.com>
22
#                 Joe Robins <jmrobins@tgix.com>
23
#                 Gervase Markham <gerv@gerv.net>
24
#                 Shane H. W. Travis <travis@sedsystems.ca>
terry%netscape.com's avatar
terry%netscape.com committed
25

26
##############################################################################
27 28 29 30
#
# enter_bug.cgi
# -------------
# Displays bug entry form. Bug fields are specified through popup menus, 
31 32
# drop-down lists, or text fields. Default for these values can be 
# passed in as parameters to the cgi.
33
#
34
##############################################################################
35

36
use strict;
terry%netscape.com's avatar
terry%netscape.com committed
37

38 39
use lib qw(.);

40
use Bugzilla;
41
use Bugzilla::Constants;
42
use Bugzilla::Bug;
43
use Bugzilla::User;
44
require "CGI.pl";
45

46 47 48
use vars qw(
  $template
  $vars
49
  @enterable_products
50 51 52 53
  @legal_opsys
  @legal_platform
  @legal_priority
  @legal_severity
54
  @legal_keywords
55
  $userid
56
  %versions
57
  %target_milestone
58
  $proddesc
59
  $classdesc
60 61 62 63
);

# If we're using bug groups to restrict bug entry, we need to know who the 
# user is right from the start. 
64
Bugzilla->login(LOGIN_REQUIRED) if AnyEntryGroups();
65

66 67 68
my $cloned_bug;
my $cloned_bug_id;

69 70
my $cgi = Bugzilla->cgi;

71 72
my $product = $cgi->param('product');

73
if (!defined $product || $product eq "") {
74
    GetVersionTable();
75
    Bugzilla->login();
76

77 78
   if ( ! Param('useclassification') ) {
      # just pick the default one
79
      $cgi->param(-name => 'classification', -value => (keys %::classdesc)[0]);
80 81
   }

82
   if (!$cgi->param('classification')) {
83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
       my %classdesc;
       my %classifications;
    
       foreach my $c (GetSelectableClassifications()) {
           $classdesc{$c} = $::classdesc{$c};
           $classifications{$c} = $::classifications{$c};
       }

       my $classification_size = scalar(keys %classdesc);
       if ($classification_size == 0) {
           ThrowUserError("no_products");
       } 
       elsif ($classification_size > 1) {
           $vars->{'classdesc'} = \%classdesc;
           $vars->{'classifications'} = \%classifications;

           $vars->{'target'} = "enter_bug.cgi";
100
           $vars->{'format'} = $cgi->param('format');
101
           
102 103
           $vars->{'cloned_bug_id'} = $cgi->param('cloned_bug_id');

104
           print $cgi->header();
105 106 107 108
           $template->process("global/choose-classification.html.tmpl", $vars)
             || ThrowTemplateError($template->error());
           exit;        
       }
109
       $cgi->param(-name => 'classification', -value => (keys %classdesc)[0]);
110
   }
111

112
    my %products;
113
    foreach my $p (@enterable_products) {
114
        if (CanEnterProduct($p)) {
115 116
            if (IsInClassification(scalar $cgi->param('classification'),$p) ||
                $cgi->param('classification') eq "__all") {
117 118
                $products{$p} = $::proddesc{$p};
            }
119
        }
120
    }
121
 
122
    my $prodsize = scalar(keys %products);
123
    if ($prodsize == 0) {
124
        ThrowUserError("no_products");
125 126
    } 
    elsif ($prodsize > 1) {
127 128 129 130
        my %classifications;
        if ( ! Param('useclassification') ) {
            @{$classifications{"all"}} = keys %products;
        }
131
        elsif ($cgi->param('classification') eq "__all") {
132 133
            %classifications = %::classifications;
        } else {
134 135
            $classifications{$cgi->param('classification')} =
                $::classifications{$cgi->param('classification')};
136
        }
137
        $vars->{'proddesc'} = \%products;
138 139
        $vars->{'classifications'} = \%classifications;
        $vars->{'classdesc'} = \%::classdesc;
140 141

        $vars->{'target'} = "enter_bug.cgi";
142
        $vars->{'format'} = $cgi->param('format');
143 144

        $vars->{'cloned_bug_id'} = $cgi->param('cloned_bug_id');
145
        
146
        print $cgi->header();
147 148
        $template->process("global/choose-product.html.tmpl", $vars)
          || ThrowTemplateError($template->error());
149
        exit;        
150 151 152
    } else {
        # Only one product exists
        $product = (keys %products)[0];
153
    }
154
}
terry%netscape.com's avatar
terry%netscape.com committed
155

156 157 158
##############################################################################
# Useful Subroutines
##############################################################################
159 160
sub formvalue {
    my ($name, $default) = (@_);
161
    return $cgi->param($name) || $default || "";
162
}
terry%netscape.com's avatar
terry%netscape.com committed
163

164
sub pickplatform {
165 166
    return formvalue("rep_platform") if formvalue("rep_platform");

167 168 169
    if (Param('defaultplatform')) {
        return Param('defaultplatform');
    } else {
170
        for ($ENV{'HTTP_USER_AGENT'}) {
171 172 173 174 175 176 177
        #PowerPC
            /\(.*PowerPC.*\)/i && do {return "Macintosh";};
            /\(.*PPC.*\)/ && do {return "Macintosh";};
            /\(.*AIX.*\)/ && do {return "Macintosh";};
        #Intel x86
            /\(.*[ix0-9]86.*\)/ && do {return "PC";};
        #Versions of Windows that only run on Intel x86
178 179
            /\(.*Win(?:dows )[39M].*\)/ && do {return "PC";};
            /\(.*Win(?:dows )16.*\)/ && do {return "PC";};
180 181 182 183
        #Sparc
            /\(.*sparc.*\)/ && do {return "Sun";};
            /\(.*sun4.*\)/ && do {return "Sun";};
        #Alpha
184 185 186
            /\(.*AXP.*\)/i && do {return "DEC";};
            /\(.*[ _]Alpha.\D/i && do {return "DEC";};
            /\(.*[ _]Alpha\)/i && do {return "DEC";};
187 188 189 190 191 192
        #MIPS
            /\(.*IRIX.*\)/i && do {return "SGI";};
            /\(.*MIPS.*\)/i && do {return "SGI";};
        #68k
            /\(.*68K.*\)/ && do {return "Macintosh";};
            /\(.*680[x0]0.*\)/ && do {return "Macintosh";};
193 194
        #HP
            /\(.*9000.*\)/ && do {return "HP";};
195 196 197 198 199 200
        #ARM
#            /\(.*ARM.*\) && do {return "ARM";};
        #Stereotypical and broken
            /\(.*Macintosh.*\)/ && do {return "Macintosh";};
            /\(.*Mac OS [89].*\)/ && do {return "Macintosh";};
            /\(Win.*\)/ && do {return "PC";};
201
            /\(.*Win(?:dows[ -])NT.*\)/ && do {return "PC";};
202 203 204 205 206 207
            /\(.*OSF.*\)/ && do {return "DEC";};
            /\(.*HP-?UX.*\)/i && do {return "HP";};
            /\(.*IRIX.*\)/i && do {return "SGI";};
            /\(.*(SunOS|Solaris).*\)/ && do {return "Sun";};
        #Braindead old browsers who didn't follow convention:
            /Amiga/ && do {return "Macintosh";};
208
            /WinMosaic/ && do {return "PC";};
209
        }
210
        return "Other";
terry%netscape.com's avatar
terry%netscape.com committed
211 212 213
    }
}

214 215 216
sub pickos {
    if (formvalue('op_sys') ne "") {
        return formvalue('op_sys');
terry%netscape.com's avatar
terry%netscape.com committed
217
    }
218 219 220
    if (Param('defaultopsys')) {
        return Param('defaultopsys');
    } else {
221
        for ($ENV{'HTTP_USER_AGENT'}) {
222 223 224
            /\(.*IRIX.*\)/ && do {return "IRIX";};
            /\(.*OSF.*\)/ && do {return "OSF/1";};
            /\(.*Linux.*\)/ && do {return "Linux";};
225
            /\(.*Solaris.*\)/ && do {return "Solaris";};
226
            /\(.*SunOS 5.*\)/ && do {return "Solaris";};
227
            /\(.*SunOS.*sun4u.*\)/ && do {return "Solaris";};
228 229
            /\(.*SunOS.*\)/ && do {return "SunOS";};
            /\(.*HP-?UX.*\)/ && do {return "HP-UX";};
230
            /\(.*BSD\/(?:OS|386).*\)/ && do {return "BSDI";};
231 232 233 234 235
            /\(.*FreeBSD.*\)/ && do {return "FreeBSD";};
            /\(.*OpenBSD.*\)/ && do {return "OpenBSD";};
            /\(.*NetBSD.*\)/ && do {return "NetBSD";};
            /\(.*BeOS.*\)/ && do {return "BeOS";};
            /\(.*AIX.*\)/ && do {return "AIX";};
236
            /\(.*OS\/2.*\)/ && do {return "OS/2";};
237 238
            /\(.*QNX.*\)/ && do {return "Neutrino";};
            /\(.*VMS.*\)/ && do {return "OpenVMS";};
239
            /\(.*Windows XP.*\)/ && do {return "Windows XP";};
240
            /\(.*Windows NT 5\.2.*\)/ && do {return "Windows Server 2003";};
241
            /\(.*Windows NT 5\.1.*\)/ && do {return "Windows XP";};
242
            /\(.*Windows 2000.*\)/ && do {return "Windows 2000";};
243
            /\(.*Windows NT 5.*\)/ && do {return "Windows 2000";};
244
            /\(.*Win.*9[8x].*4\.9.*\)/ && do {return "Windows ME";};
245 246 247 248 249 250
            /\(.*Win(?:dows )M[Ee].*\)/ && do {return "Windows ME";};
            /\(.*Win(?:dows )98.*\)/ && do {return "Windows 98";};
            /\(.*Win(?:dows )95.*\)/ && do {return "Windows 95";};
            /\(.*Win(?:dows )16.*\)/ && do {return "Windows 3.1";};
            /\(.*Win(?:dows[ -])NT.*\)/ && do {return "Windows NT";};
            /\(.*Windows.*NT.*\)/ && do {return "Windows NT";};
251 252 253 254
            /\(.*32bit.*\)/ && do {return "Windows 95";};
            /\(.*16bit.*\)/ && do {return "Windows 3.1";};
            /\(.*Mac OS 9.*\)/ && do {return "Mac System 9.x";};
            /\(.*Mac OS 8\.6.*\)/ && do {return "Mac System 8.6";};
255
            /\(.*Mac OS 8\.5.*\)/ && do {return "Mac System 8.5";};
256
        # Bugzilla doesn't have an entry for 8.1
257 258 259 260
            /\(.*Mac OS 8\.1.*\)/ && do {return "Mac System 8.0";};
            /\(.*Mac OS 8\.0.*\)/ && do {return "Mac System 8.0";};
            /\(.*Mac OS 8[^.].*\)/ && do {return "Mac System 8.0";};
            /\(.*Mac OS 8.*\)/ && do {return "Mac System 8.6";};
261 262
            /\(.*Mac OS X.*\)/ && do {return "Mac OS X 10.0";};
            /\(.*Darwin.*\)/ && do {return "Mac OS X 10.0";};
263
        # Silly
264 265 266
            /\(.*Mac.*PowerPC.*\)/ && do {return "Mac System 9.x";};
            /\(.*Mac.*PPC.*\)/ && do {return "Mac System 9.x";};
            /\(.*Mac.*68k.*\)/ && do {return "Mac System 8.0";};
267
        # Evil
268
            /Amiga/i && do {return "other";};
269
            /WinMosaic/ && do {return "Windows 95";};
270 271 272
            /\(.*PowerPC.*\)/ && do {return "Mac System 9.x";};
            /\(.*PPC.*\)/ && do {return "Mac System 9.x";};
            /\(.*68K.*\)/ && do {return "Mac System 8.0";};
273
        }
274
        return "other";
terry%netscape.com's avatar
terry%netscape.com committed
275 276
    }
}
277 278 279
##############################################################################
# End of subroutines
##############################################################################
terry%netscape.com's avatar
terry%netscape.com committed
280

281
Bugzilla->login(LOGIN_REQUIRED) if (!(AnyEntryGroups()));
terry%netscape.com's avatar
terry%netscape.com committed
282

283 284 285 286 287 288 289 290 291 292
# If a user is trying to clone a bug
#   Check that the user has authorization to view the parent bug
#   Create an instance of Bug that holds the info from the parent
$cloned_bug_id = $cgi->param('cloned_bug_id');

if ($cloned_bug_id) {
    ValidateBugID($cloned_bug_id);
    $cloned_bug = new Bugzilla::Bug($cloned_bug_id, $userid);
}

293
# We need to check and make sure
294
# that the user has permission to enter a bug against this product.
295
CanEnterProductOrWarn($product);
296

297
GetVersionTable();
terry%netscape.com's avatar
terry%netscape.com committed
298

299 300
my $product_id = get_product_id($product);

301
if (0 == @{$::components{$product}}) {        
302
    ThrowUserError("no_components", {product => $product});   
303 304
} 
elsif (1 == @{$::components{$product}}) {
305
    # Only one component; just pick it.
306
    $cgi->param('component', $::components{$product}->[0]);
307 308
}

309
my @components;
310

311 312 313 314 315 316 317 318 319 320 321 322
my $dbh = Bugzilla->dbh;
my $sth = $dbh->prepare(
       q{SELECT name, description, p1.login_name, p2.login_name 
           FROM components 
      LEFT JOIN profiles p1 ON components.initialowner = p1.userid
      LEFT JOIN profiles p2 ON components.initialqacontact = p2.userid
          WHERE product_id = ?
          ORDER BY name});

$sth->execute($product_id);
while (my ($name, $description, $owner, $qacontact)
       = $sth->fetchrow_array()) {
323 324 325
    push @components, {
        name => $name,
        description => $description,
326 327
        initialowner => $owner,
        initialqacontact => $qacontact || '',
328
    };
329 330
}

331
my %default;
terry%netscape.com's avatar
terry%netscape.com committed
332

333 334 335 336 337 338 339 340 341
$vars->{'product'}               = $product;
$vars->{'component_'}            = \@components;

$vars->{'priority'}              = \@legal_priority;
$vars->{'bug_severity'}          = \@legal_severity;
$vars->{'rep_platform'}          = \@legal_platform;
$vars->{'op_sys'}                = \@legal_opsys; 

$vars->{'use_keywords'}          = 1 if (@::legal_keywords);
terry%netscape.com's avatar
terry%netscape.com committed
342

343 344 345
$vars->{'assigned_to'}           = formvalue('assigned_to');
$vars->{'assigned_to_disabled'}  = !UserInGroup('editbugs');
$vars->{'cc_disabled'}           = 0;
346

347 348 349
$vars->{'qa_contact'}           = formvalue('qa_contact');
$vars->{'qa_contact_disabled'}  = !UserInGroup('editbugs');

350
$vars->{'cloned_bug_id'}         = $cloned_bug_id;
351

352
if ($cloned_bug_id) {
353

354 355 356 357 358
    $default{'component_'}    = $cloned_bug->{'component'};
    $default{'priority'}      = $cloned_bug->{'priority'};
    $default{'bug_severity'}  = $cloned_bug->{'bug_severity'};
    $default{'rep_platform'}  = $cloned_bug->{'rep_platform'};
    $default{'op_sys'}        = $cloned_bug->{'op_sys'};
359

360 361
    $vars->{'short_desc'}     = $cloned_bug->{'short_desc'};
    $vars->{'bug_file_loc'}   = $cloned_bug->{'bug_file_loc'};
362
    $vars->{'keywords'}       = $cloned_bug->keywords;
363 364
    $vars->{'dependson'}      = $cloned_bug_id;
    $vars->{'blocked'}        = "";
365
    $vars->{'deadline'}       = $cloned_bug->{'deadline'};
366

367 368
    if (defined $cloned_bug->cc) {
        $vars->{'cc'}         = join (" ", @{$cloned_bug->cc});
369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389
    } else {
        $vars->{'cc'}         = formvalue('cc');
    }

# We need to ensure that we respect the 'insider' status of
# the first comment, if it has one. Either way, make a note
# that this bug was cloned from another bug.

    $cloned_bug->longdescs();
    my $isprivate             = $cloned_bug->{'longdescs'}->[0]->{'isprivate'};

    $vars->{'comment'}        = "";
    $vars->{'commentprivacy'} = 0;

    if ( !($isprivate) ||
         ( ( Param("insidergroup") ) && 
           ( UserInGroup(Param("insidergroup")) ) ) 
       ) {
        $vars->{'comment'}        = $cloned_bug->{'longdescs'}->[0]->{'body'};
        $vars->{'commentprivacy'} = $isprivate;
    }
390

391 392
# Ensure that the groupset information is set up for later use.
    $cloned_bug->groups();
393

394
} # end of cloned bug entry form
395

396
else {
397

398 399 400 401 402 403 404 405 406 407 408
    $default{'component_'}    = formvalue('component');
    $default{'priority'}      = formvalue('priority', Param('defaultpriority'));
    $default{'bug_severity'}  = formvalue('bug_severity', Param('defaultseverity'));
    $default{'rep_platform'}  = pickplatform();
    $default{'op_sys'}        = pickos();

    $vars->{'short_desc'}     = formvalue('short_desc');
    $vars->{'bug_file_loc'}   = formvalue('bug_file_loc', "http://");
    $vars->{'keywords'}       = formvalue('keywords');
    $vars->{'dependson'}      = formvalue('dependson');
    $vars->{'blocked'}        = formvalue('blocked');
409
    $vars->{'deadline'}       = formvalue('deadline');
410

411
    $vars->{'cc'}             = join(', ', $cgi->param('cc'));
412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429

    $vars->{'comment'}        = formvalue('comment');
    $vars->{'commentprivacy'} = formvalue('commentprivacy');

} # end of normal/bookmarked entry form


# IF this is a cloned bug,
# AND the clone's product is the same as the parent's
#   THEN use the version from the parent bug
# ELSE IF a version is supplied in the URL
#   THEN use it
# ELSE IF there is a version in the cookie
#   THEN use it (Posting a bug sets a cookie for the current version.)
# ELSE
#   The default version is the last one in the list (which, it is
#   hoped, will be the most recent one).
#
430 431
# Eventually maybe each product should have a "current version"
# parameter.
432
$vars->{'version'} = $::versions{$product} || [];
433 434 435 436 437

if ( ($cloned_bug_id) &&
     ("$product" eq "$cloned_bug->{'product'}" ) ) {
    $default{'version'} = $cloned_bug->{'version'};
} elsif (formvalue('version')) {
438
    $default{'version'} = formvalue('version');
439 440 441
} elsif (defined $cgi->cookie("VERSION-$product") &&
    lsearch($vars->{'version'}, $cgi->cookie("VERSION-$product")) != -1) {
    $default{'version'} = $cgi->cookie("VERSION-$product");
442 443 444
} else {
    $default{'version'} = $vars->{'version'}->[$#{$vars->{'version'}}];
}
445

446 447 448 449 450 451 452 453 454 455 456 457 458
# Get list of milestones.
if ( Param('usetargetmilestone') ) {
    $vars->{'target_milestone'} = $::target_milestone{$product};
    if (formvalue('target_milestone')) {
       $default{'target_milestone'} = formvalue('target_milestone');
    } else {
       SendSQL("SELECT defaultmilestone FROM products WHERE " .
               "name = " . SqlQuote($product));
       $default{'target_milestone'} = FetchOneColumn();
    }
}


459 460
# List of status values for drop-down.
my @status;
461

462 463 464 465 466 467 468 469 470 471 472 473 474
# Construct the list of allowable values.  There are three cases:
# 
#  case                                 values
#  product does not have confirmation   NEW
#  confirmation, user cannot confirm    UNCONFIRMED
#  confirmation, user can confirm       NEW, UNCONFIRMED.

SendSQL("SELECT votestoconfirm FROM products WHERE name = " .
        SqlQuote($product));
if (FetchOneColumn()) {
    if (UserInGroup("editbugs") || UserInGroup("canconfirm")) {
        push(@status, "NEW");
    }
475
    push(@status, 'UNCONFIRMED');
476 477
} else {
    push(@status, "NEW");
478 479
}

480
$vars->{'bug_status'} = \@status; 
481

482 483 484 485 486 487 488 489 490
# Get the default from a template value if it is legitimate.
# Otherwise, set the default to the first item on the list.

if (formvalue('bug_status') && (lsearch(\@status, formvalue('bug_status')) >= 0)) {
    $default{'bug_status'} = formvalue('bug_status');
} else {
    $default{'bug_status'} = $status[0];
}
 
491 492 493 494 495
SendSQL("SELECT DISTINCT groups.id, groups.name, groups.description, " .
        "membercontrol, othercontrol " .
        "FROM groups LEFT JOIN group_control_map " .
        "ON group_id = id AND product_id = $product_id " .
        "WHERE isbuggroup != 0 AND isactive != 0 ORDER BY description");
496

497
my @groups;
498

499
while (MoreSQLData()) {
500 501 502 503 504 505 506 507 508 509
    my ($id, $groupname, $description, $membercontrol, $othercontrol) 
        = FetchSQLData();
    # Only include groups if the entering user will have an option.
    next if ((!$membercontrol) 
               || ($membercontrol == CONTROLMAPNA) 
               || ($membercontrol == CONTROLMAPMANDATORY)
               || (($othercontrol != CONTROLMAPSHOWN) 
                    && ($othercontrol != CONTROLMAPDEFAULT)
                    && (!UserInGroup($groupname)))
             );
510 511
    my $check;

512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528
    # If this is a cloned bug, 
    # AND the product for this bug is the same as for the original
    #   THEN set a group's checkbox if the original also had it on
    # ELSE IF this is a bookmarked template
    #   THEN set a group's checkbox if was set in the bookmark
    # ELSE
    #   set a groups's checkbox based on the group control map
    #
    if ( ($cloned_bug_id) &&
         ("$product" eq "$cloned_bug->{'product'}" ) ) {
        foreach my $i (0..(@{$cloned_bug->{'groups'}}-1) ) {
            if ($cloned_bug->{'groups'}->[$i]->{'bit'} == $id) {
                $check = $cloned_bug->{'groups'}->[$i]->{'ison'};
            }
        }
    }
    elsif(formvalue("maketemplate") ne "") {
529 530 531
        $check = formvalue("bit-$id", 0);
    }
    else {
532 533 534 535
        # Checkbox is checked by default if $control is a default state.
        $check = (($membercontrol == CONTROLMAPDEFAULT)
                 || (($othercontrol == CONTROLMAPDEFAULT)
                      && (!UserInGroup($groupname))));
536 537
    }

538 539 540 541 542 543 544 545
    my $group = 
    {
        'bit' => $id , 
        'checked' => $check , 
        'description' => $description 
    };

    push @groups, $group;        
546
}
terry%netscape.com's avatar
terry%netscape.com committed
547

548 549
$vars->{'group'} = \@groups;

550
$vars->{'default'} = \%default;
terry%netscape.com's avatar
terry%netscape.com committed
551

552
my $format = 
553 554
  GetFormat("bug/create/create", scalar $cgi->param('format'), 
            scalar $cgi->param('ctype'));
555

556
print $cgi->header($format->{'ctype'});
557
$template->process($format->{'template'}, $vars)
558
  || ThrowTemplateError($template->error());          
559