[Rt-commit] [svn] r986 - in rt/branches/rt-3.3: . bin etc
	html/Search lib lib/RT sbin
    jesse at pallas.eruditorum.org 
    jesse at pallas.eruditorum.org
       
    Mon May 31 21:54:38 EDT 2004
    
    
  
Author: jesse
Date: Mon May 31 21:54:38 2004
New Revision: 986
Added:
   rt/branches/rt-3.3/etc/acl.Sybase
   rt/branches/rt-3.3/etc/schema.Sybase
Modified:
   rt/branches/rt-3.3/   (props changed)
   rt/branches/rt-3.3/bin/rt.in
   rt/branches/rt-3.3/bin/webmux.pl.in
   rt/branches/rt-3.3/configure.ac
   rt/branches/rt-3.3/html/Search/Build.html
   rt/branches/rt-3.3/html/Search/Bulk.html
   rt/branches/rt-3.3/lib/RT.pm.in
   rt/branches/rt-3.3/lib/RT/EmailParser.pm
   rt/branches/rt-3.3/lib/RT/Handle.pm
   rt/branches/rt-3.3/lib/RT/Tickets_Overlay_SQL.pm
   rt/branches/rt-3.3/sbin/rt-setup-database.in
   rt/branches/rt-3.3/sbin/rt-test-dependencies.in
Log:
Merging forward from RT 3.1
Modified: rt/branches/rt-3.3/bin/rt.in
==============================================================================
--- rt/branches/rt-3.3/bin/rt.in	(original)
+++ rt/branches/rt-3.3/bin/rt.in	Mon May 31 21:54:38 2004
@@ -153,7 +153,6 @@
             foreach $title (grep $_, @titles) {
                 $help{$title} = $item->[2]{Text};
             }
-        }
     }
 
     # What does the user want help with?
Modified: rt/branches/rt-3.3/bin/webmux.pl.in
==============================================================================
--- rt/branches/rt-3.3/bin/webmux.pl.in	(original)
+++ rt/branches/rt-3.3/bin/webmux.pl.in	Mon May 31 21:54:38 2004
@@ -31,7 +31,6 @@
     $ENV{'SHELL'}  = '/bin/sh' if defined $ENV{'SHELL'};
     $ENV{'ENV'}    = '' if defined $ENV{'ENV'};
     $ENV{'IFS'}    = '' if defined $ENV{'IFS'};
-    @ORACLE_ENV_PREF@
 
     eval { require Apache2; require Apache::compat }
         if $ENV{'MOD_PERL'} and $ENV{'MOD_PERL'} =~ m{mod_perl/(?:2|1\.9)};
Modified: rt/branches/rt-3.3/configure.ac
==============================================================================
--- rt/branches/rt-3.3/configure.ac	(original)
+++ rt/branches/rt-3.3/configure.ac	Mon May 31 21:54:38 2004
@@ -79,17 +79,23 @@
 	    		   [sort of database RT will use (default: mysql) (mysql, Pg, Oracle and Informix are valid)]), 
             DB_TYPE=$withval,
             DB_TYPE=mysql)
-if test "$DB_TYPE" != 'mysql' -a "$DB_TYPE" != 'Pg' -a "$DB_TYPE" != 'SQLite' -a "$DB_TYPE" != 'Oracle' -a "$DB_TYPE" != 'Informix' ; then
+if test "$DB_TYPE" != 'mysql' -a "$DB_TYPE" != 'Pg' -a "$DB_TYPE" != 'SQLite' -a "$DB_TYPE" != 'Oracle' -a "$DB_TYPE" != 'Informix' -a "$DB_TYPE" != 'Sybase' ; then
 	AC_MSG_ERROR([Only Oracle, Informix, Pg and mysql are valid db types])
 fi
 AC_SUBST(DB_TYPE)
 
-dnl ORACLE_ENV_PREF
+dnl DATABASE_ENV_PREF
 if test "$DB_TYPE" = 'Oracle'; then
 	test "x$ORACLE_HOME" = 'x' && AC_MSG_ERROR([Please declare the ORACLE_HOME environment variable])
-	ORACLE_ENV_PREF="\$ENV{'ORACLE_HOME'} = '$ORACLE_HOME';"
+	DATABASE_ENV_PREF="\$ENV{'ORACLE_HOME'} = '$ORACLE_HOME';"
 fi
-AC_SUBST(ORACLE_ENV_PREF)
+
+dnl DATABASE_ENV_PREF
+if test "$DB_TYPE" = 'Sybase'; then
+	test "x$SYBASE" = 'x' && AC_MSG_ERROR([Please declare the SYBASE_HOME environment variable])
+	DATABASE_ENV_PREF="\$ENV{'SYBASE'} = '$SYBASE';"
+fi
+AC_SUBST(DATABASE_ENV_PREF)
 
 dnl DB_HOST
 AC_ARG_WITH(db-host,
Added: rt/branches/rt-3.3/etc/acl.Sybase
==============================================================================
--- (empty file)
+++ rt/branches/rt-3.3/etc/acl.Sybase	Mon May 31 21:54:38 2004
@@ -0,0 +1,6 @@
+sub acl {
+return (
+"SP_ADDLOGIN ${RT::DatabaseUser}, ${RT::DatabasePassword}, ${RT::DatabaseName} ",
+);
+}
+1;
Added: rt/branches/rt-3.3/etc/schema.Sybase
==============================================================================
--- (empty file)
+++ rt/branches/rt-3.3/etc/schema.Sybase	Mon May 31 21:54:38 2004
@@ -0,0 +1,446 @@
+NEEDS UPDATE FOR 3.3
+
+# {{{ Attachments
+
+CREATE TABLE rt3.Attachments (
+  id numeric(38,0) identity,
+  TransactionId integer NOT NULL  ,
+  Parent integer NOT NULL   ,
+  MessageId varchar(160) NULL  ,
+  Subject varchar(255) NULL  ,
+  Filename varchar(255) NULL  ,
+  ContentType varchar(80) NULL  ,
+  ContentEncoding varchar(80) NULL  ,
+  Content TEXT NULL  ,
+  Headers TEXT NULL  ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX Attachments1 ON Attachments (Parent) ;
+CREATE INDEX Attachments2 ON Attachments (TransactionId) ;
+CREATE INDEX Attachments3 ON Attachments (Parent, TransactionId) ;
+# }}}
+
+# {{{ Queues
+CREATE TABLE rt3.Queues (
+  id numeric(38,0) identity,
+  Name varchar(200) NOT NULL  ,
+  Description varchar(255) NULL  ,
+  CorrespondAddress varchar(120) NULL  ,
+  CommentAddress varchar(120) NULL  ,
+  InitialPriority integer NOT NULL   ,
+  FinalPriority integer NOT NULL   ,
+  DefaultDueIn integer NOT NULL   ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  Disabled numeric(1) NOT NULL  ,
+  PRIMARY KEY (id)
+) ;
+CREATE UNIQUE INDEX Queues1 ON Queues (Name) ;
+CREATE INDEX Queues2 ON Queues (Disabled) ;
+
+# }}}
+
+# {{{ Links
+
+CREATE TABLE rt3.Links (
+  id numeric(38,0) identity,
+  Base varchar(240) NULL  ,
+  Target varchar(240) NULL  ,
+  Type varchar(20) NOT NULL  ,
+  LocalTarget integer NOT NULL   ,
+  LocalBase integer NOT NULL   ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE UNIQUE INDEX Links1 ON Links (Base, Target, Type) ;
+CREATE INDEX Links2 ON Links (Base,  Type) ;
+CREATE INDEX Links3 ON Links (Target,  Type) ;
+CREATE INDEX Links4 ON Links(Type,LocalBase);
+
+# }}}
+
+# {{{ Principals
+
+CREATE TABLE rt3.Principals (
+      id numeric(38,0) identity,
+        PrincipalType VARCHAR(16) not null,
+        ObjectId integer,         Disabled numeric(1) NOT NULL  ,
+        PRIMARY KEY (id)
+) ;
+
+CREATE INDEX Principals2 ON Principals (ObjectId);
+
+# }}}
+
+# {{{ Groups
+
+CREATE TABLE rt3.Groups (
+  id numeric(38,0) identity,
+  Name varchar(200) NULL  ,
+  Description varchar(255) NULL  ,
+  Domain varchar(64),
+  Type varchar(64),
+  Instance integer,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX Groups1 ON Groups (Domain,Instance,Type,id);
+CREATE INDEX Groups2 On Groups  (Type, Instance, Domain);   
+
+# }}}
+
+# {{{ ScripConditions
+
+CREATE TABLE rt3.ScripConditions (
+  id numeric(38,0) identity,
+  Name varchar(200) NULL  ,
+  Description varchar(255) NULL  ,
+  ExecModule varchar(60) NULL  ,
+  Argument varchar(255) NULL  ,
+  ApplicableTransTypes varchar(60) NULL  ,
+
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+# }}}
+
+# {{{ Transactions
+CREATE TABLE rt3.Transactions (
+  id numeric(38,0) identity,
+  EffectiveTicket integer NOT NULL   ,
+  Ticket integer NOT NULL   ,
+  TimeTaken integer NOT NULL   ,
+  Type varchar(20) NULL  ,
+  Field varchar(40) NULL  ,
+  OldValue varchar(255) NULL  ,
+  NewValue varchar(255) NULL  ,
+  Data varchar(255) NULL  ,
+
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+CREATE INDEX Transactions1 ON Transactions (Ticket);
+CREATE INDEX Transactions2 ON Transactions (EffectiveTicket);
+
+# }}}
+
+# {{{ Scrips 
+
+CREATE TABLE rt3.Scrips (
+  id numeric(38,0) identity,
+  Description varchar(255),
+  ScripCondition integer NOT NULL   ,
+  ScripAction integer NOT NULL   ,
+  ConditionRules text NULL  ,
+  ActionRules text NULL  ,
+  CustomIsApplicableCode text NULL  ,
+  CustomPrepareCode text NULL  ,
+  CustomCommitCode text NULL  ,
+  Stage varchar(32) NULL  ,
+  Queue integer NOT NULL   ,
+  Template integer NOT NULL   ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+# }}}
+
+# {{{ ACL
+CREATE TABLE rt3.ACL (
+  id numeric(38,0) identity,
+  PrincipalType varchar(25) NOT NULL, #"User" "Group", "Owner", "Cc" "AdminCc", "Requestor", "Requestor" 
+
+  PrincipalId integer NOT NULL  , #Foreign key to principals
+  RightName varchar(25) NOT NULL  ,
+  ObjectType varchar(25) NOT NULL  ,
+  ObjectId integer NOT NULL ,
+  DelegatedBy integer NOT NULL , #foreign key to principals with a userid
+  DelegatedFrom integer NOT NULL , #foreign key to ACL
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX  ACL1 on ACL(RightName, ObjectType, ObjectId,PrincipalType,PrincipalId);
+
+# }}}
+
+# {{{ GroupMembers 
+
+CREATE TABLE rt3.GroupMembers (
+  id numeric(38,0) identity,
+  GroupId integer NOT NULL ,
+  MemberId integer NOT NULL ,  #Foreign key to Principals
+  PRIMARY KEY (id)
+) ;
+CREATE UNIQUE INDEX GroupMembers1 on GroupMembers (GroupId, MemberId);
+
+
+# }}}
+
+# {{{ GroupMembersCache
+
+CREATE TABLE rt3.CachedGroupMembers (
+  id numeric(38,0) identity,
+        GroupId int, # foreign key to Principals
+        MemberId int, # foreign key to Principals
+        Via int, #foreign key to CachedGroupMembers. (may point to $self->id)
+        ImmediateParentId int, #foreign key to prinicpals.         
+                               # this points to the group that the member is
+                               # a member of, for ease of deletes.
+        Disabled numeric(1) NOT NULL  , # if this cached group member is a member of this group by way of a disabled
+                                           # group or this group is disabled, this will be set to 1
+                                           # this allows us to not find members of disabled subgroups when listing off
+                                           # group members recursively.
+                                           # Also, this allows us to have the ACL system elide members of disabled groups
+        PRIMARY KEY (id)
+) ;
+
+CREATE INDEX DisGrouMem  on CachedGroupMembers (GroupId,MemberId,Disabled);
+CREATE INDEX GrouMem  on CachedGroupMembers (GroupId,MemberId);
+
+# }}}
+
+# {{{ Users
+
+CREATE TABLE rt3.Users (
+  id numeric(38,0) identity,
+  Name varchar(200) NOT NULL  ,
+  Password varchar(40) NULL  ,
+  Comments text NULL  ,
+  Signature text NULL  ,
+  EmailAddress varchar(120) NULL  ,
+  FreeformContactInfo text NULL  ,
+  Organization varchar(200) NULL  ,
+  RealName varchar(120) NULL  ,
+  NickName varchar(16) NULL  ,
+  Lang varchar(16) NULL  ,
+  EmailEncoding varchar(16) NULL  ,
+  WebEncoding varchar(16) NULL  ,
+  ExternalContactInfoId varchar(100) NULL  ,
+  ContactInfoSystem varchar(30) NULL  ,
+  ExternalAuthId varchar(100) NULL  ,
+  AuthSystem varchar(30) NULL  ,
+  Gecos varchar(16) NULL  ,
+  HomePhone varchar(30) NULL  ,
+  WorkPhone varchar(30) NULL  ,
+  MobilePhone varchar(30) NULL  ,
+  PagerPhone varchar(30) NULL  ,
+  Address1 varchar(200) NULL  ,
+  Address2 varchar(200) NULL  ,
+  City varchar(100) NULL  ,
+  State varchar(100) NULL  ,
+  Zip varchar(16) NULL  ,
+  Country varchar(50) NULL  ,
+  Timezone varchar(50) NULL  ,
+  PGPKey text NULL,
+
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+
+CREATE UNIQUE INDEX Users1 ON Users (Name) ;
+CREATE INDEX Users2 ON Users (Name);
+CREATE INDEX Users3 ON Users (id, EmailAddress);
+CREATE INDEX Users4 ON Users (EmailAddress);
+
+
+# }}}
+
+# {{{ Tickets
+
+CREATE TABLE rt3.Tickets (
+  id numeric(38,0) identity,
+  EffectiveId integer NOT NULL   ,
+  Queue integer NOT NULL   ,
+  Type varchar(16) NULL  ,
+  IssueStatement integer NOT NULL   ,
+  Resolution integer NOT NULL   ,
+  Owner integer NOT NULL   ,
+  Subject varchar(200) NULL,
+  InitialPriority integer NOT NULL   ,
+  FinalPriority integer NOT NULL   ,
+  Priority integer NOT NULL   ,
+  TimeEstimated integer NOT NULL   ,
+  TimeWorked integer NOT NULL   ,
+  Status varchar(10) NULL  ,
+  TimeLeft integer NOT NULL   ,
+  Told DATETIME NULL  ,
+  Starts DATETIME NULL  ,
+  Started DATETIME NULL  ,
+  Due DATETIME NULL  ,
+  Resolved DATETIME NULL  ,
+
+
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  Disabled numeric(1) NOT NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX Tickets1 ON Tickets (Queue, Status) ;
+CREATE INDEX Tickets2 ON Tickets (Owner) ;
+CREATE INDEX Tickets3 ON Tickets (EffectiveId) ;
+CREATE INDEX Tickets4 ON Tickets (id, Status) ;
+CREATE INDEX Tickets5 ON Tickets (id, EffectiveId) ;
+CREATE INDEX Tickets6 ON Tickets (EffectiveId, Type) ;
+
+# }}}
+
+# {{{ ScripActions
+
+CREATE TABLE rt3.ScripActions (
+  id numeric(38,0) identity,
+  Name varchar(200) NULL  ,
+  Description varchar(255) NULL  ,
+  ExecModule varchar(60) NULL  ,
+  Argument varchar(255) NULL  ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+# }}}
+
+# {{{ Templates
+
+CREATE TABLE rt3.Templates (
+  id numeric(38,0) identity,
+  Queue integer NOT NULL  ,
+  Name varchar(200) NOT NULL  ,
+  Description varchar(255) NULL  ,
+  Type varchar(16) NULL  ,
+  Language varchar(16) NULL  ,
+  TranslationOf integer NOT NULL   ,
+  Content text NULL  ,
+  LastUpdated DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+# }}}
+
+# {{{ TicketCustomFieldValues 
+
+CREATE TABLE rt3.TicketCustomFieldValues (
+  id numeric(38,0) identity,
+  Ticket int NOT NULL  ,
+  CustomField int NOT NULL  ,
+  Content varchar(255) NULL  ,
+
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX TicketCustomFieldValues1 ON TicketCustomFieldValues (CustomField,Ticket,Content); 
+CREATE INDEX TicketCustomFieldValues2 ON TicketCustomFieldValues (CustomField,Ticket); 
+
+# }}}
+
+# {{{ CustomFields
+
+CREATE TABLE rt3.CustomFields (
+  id numeric(38,0) identity,
+  Name varchar(200) NULL  ,
+  Type varchar(200) NULL  ,
+  Queue integer NOT NULL  ,
+  Description varchar(255) NULL  ,
+  SortOrder integer NOT NULL   ,
+
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  Disabled numeric(1) NOT NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX CustomFields1 on CustomFields (Disabled, Queue);
+
+
+# }}}
+
+# {{{ CustomFieldValues 
+
+CREATE TABLE rt3.CustomFieldValues (
+  id numeric(38,0) identity,
+  CustomField int NOT NULL  ,
+  Name varchar(200) NULL  ,
+  Description varchar(255) NULL  ,
+  SortOrder integer NOT NULL   ,
+
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX CustomFieldValues1 ON CustomFieldValues (CustomField);
+ 
+# }}}
+
+
+# {{{ Attributes
+
+CREATE TABLE rt3.Attributes (
+  id numeric(38,0) identity,
+  Name varchar(255) NULL  ,
+  Description varchar(255) NULL  ,
+  Content text,
+  ContentType varchar(16),
+  ObjectType varchar(64),
+  ObjectId integer, # foreign key to anything
+  Creator integer NOT NULL   ,
+  Created DATETIME NULL  ,
+  LastUpdatedBy integer NOT NULL   ,
+  LastUpdated DATETIME NULL  ,
+  PRIMARY KEY (id)
+) ;
+
+CREATE INDEX Attributes1 on Attributes(Name);
+CREATE INDEX Attributes2 on Attributes(ObjectType, ObjectId);
+
+# }}}
+
+# {{{ Sessions
+
+# sessions is used by Apache::Session to keep sessions in the database.
+# We should have a reaper script somewhere.
+
+CREATE TABLE rt3.sessions (
+    id char(32) NOT NULL,
+    a_session TEXT,
+    LastUpdated DATETIME,
+    PRIMARY KEY (id)
+);
+
+# }}}
Modified: rt/branches/rt-3.3/html/Search/Build.html
==============================================================================
--- rt/branches/rt-3.3/html/Search/Build.html	(original)
+++ rt/branches/rt-3.3/html/Search/Build.html	Mon May 31 21:54:38 2004
@@ -24,10 +24,10 @@
 <& /Elements/Header, Title => $title &>
 <& /Ticket/Elements/Tabs, 
     current_tab => "Search/Build.html".$QueryString, 
-    Title => $title,
-    Format => $Format,
+    Title => $title, Format => $Format,
     Query => $Query,
-    Rows => $ARGS{'Rows'} &>
+    Rows => $ARGS{'Rows'},
+&>
 
 <table width=100%>
 <tr>
@@ -64,7 +64,7 @@
 <input type=hidden name=Query value="<%$Query%>">
 <input type=hidden name=Format value="<%$Format%>">
 <& Elements/EditSearches, CurrentSearch => $search_hash &>
-</form>
+</FORM>
 </td>
 </tr>
 <tr>
@@ -76,13 +76,13 @@
 <input type=hidden name=SearchId value="<%$ARGS{'SearchId'}%>">
 <& Elements/DisplayOptions, Format=> ($Format||$search_hash->{'Format'}),
 AvailableColumns => $AvailableColumns,  CurrentFormat => $CurrentFormat, %ARGS &>
-</form>
+</FORM>
 </td>
 </tr>
 </table>
-</FORM>
 
 <%INIT>
+use Tree::Simple;
 
 my $search_hash = {};
 my $search;
@@ -93,6 +93,7 @@
     $Format = '';
     $Description = '';
     undef $session{'CurrentSearchHash'};
+    $session{'tickets'}->CleanSlate() if defined $session{'tickets'};
 } else {
     $search_hash = $session{'CurrentSearchHash'};
     $Query ||= $search_hash->{'Query'};
@@ -202,7 +203,8 @@
 
 
 # {{{ Parse the query
-my $items = ParseQuery( $Query, \@actions );
+my $tree;
+ParseQuery( $Query, \$tree, \@actions );
 
 # if parsing went poorly, send them to the edit page to fix it
 if ( $actions[0] ) {
@@ -211,29 +213,19 @@
 }
 
 my @options;
+my $optionlist;
 $Query  = "";
 %queues = ();
-build_array( \$Query, $items, "", "", 0, \@options, \%queues );
-my $currentkey = "";
-$currentkey = $options[$ARGS{clauses}]->{Key} if $ARGS{clauses};
+
+$optionlist = build_array( \$Query, $ARGS{clauses}, $tree, \@options, \%queues );
+
+my $currentkey;
+$currentkey = $options[$ARGS{clauses}] if defined $ARGS{clauses};
 
 # We can't check for the addclause button because hitting return in
 # a criterion will get lost otherwise
 if (1) {
 
-    my ($key, $top);
-    if ($currentkey) {
-        $key = $currentkey;
-        my ( $prefix, $depth, $num ) = parsekey($key);
-        $top = 1 if $depth == 1;  
-    }
-    else {
-        $key = "{" . '0.0' . "}{Subkey}{1.0}";
-        $top = 1;
-    }
-    while ( keyexists( $items, $key ) ) {
-        $key = nextkey($key);
-    }
     foreach my $arg ( keys %ARGS ) {
         if ( $arg =~ m/ValueOf(.+)/ && $ARGS{$arg} ) {
             my $field = $1;
@@ -252,31 +244,36 @@
                 Value => "'$ARGS{'ValueOf' . $field}'"
             };
 
-            setvalue($items, $key, $clause);
-            setaggregator( $items, $key, $ARGS{'AndOr'} );
-
-            if ($top) {
-                # if the next item is a subkey, put it there
-                my $newkey = appendsubkey( $items, $key );
-                my $oldkey = $key;
-                movecurrent( $items, $newkey, \$oldkey );
-            }
-
-            $currentkey = $key;
-            $key = nextkey($key);
-        }
+	    my $newnode = Tree::Simple->new($clause);
+	    if ($currentkey) {
+		my $newindex = $currentkey->getIndex() + 1;
+		if ($currentkey->getParent()->getParent()->isRoot()) {
+		    my $newparent = Tree::Simple->new($ARGS{'AndOr'});
+		    $newparent->addChild($newnode);
+		    $newnode = $newparent;
+		}
+		$currentkey->insertSibling($newindex, $newnode);
+		$currentkey = $newnode;
+	    } else {
+		my $newparent = Tree::Simple->new($ARGS{'AndOr'});
+		$newparent->addChild($newnode);
+		$tree->getChild(0)->addChild($newparent);
+		$currentkey = $newparent;
+	    }
+	}
     }
 }
 
 # {{{ Move things around
 if ( $ARGS{"Up"} ) {
     if ($currentkey) {
-        # we can only move it up if it's not at the top
-        my $prev = prevkey($currentkey);
-
-        if ( swap( $items, $prev, $currentkey ) ) {
-            $currentkey = "$prev";
-        }
+	my $index = $currentkey->getIndex();
+	if ( $currentkey->getIndex() > 0 ) {
+	    my $parent = $currentkey->getParent();
+	    $parent->removeChild($index);
+	    $parent->insertChild($index - 1, $currentkey);
+	    $currentkey = $parent->getChild($index - 1);
+	}
         else {
             push( @actions, [ "error: can't move up", -1 ] );
         }
@@ -287,12 +284,13 @@
 }
 elsif ( $ARGS{"Down"} ) {
     if ($currentkey) {
-        # we can only move it down if it's not at the bottom
-        my $newkey = nextkey($currentkey);
-
-        if ( swap( $items, $newkey, $currentkey ) ) {
-            $currentkey = "$newkey";
-        }
+	my $index = $currentkey->getIndex();
+	my $parent = $currentkey->getParent();
+	if ( $currentkey->getIndex() < ($parent->getChildCount - 1) ) {
+	    $parent->removeChild($index);
+	    $parent->insertChild($index + 1, $currentkey);
+	    $currentkey = $parent->getChild($index + 1);
+	}
         else {
             push( @actions, [ "error: can't move down", -1 ] );
         }
@@ -303,20 +301,15 @@
 }
 elsif ( $ARGS{"Left"} ) {
     if ($currentkey) {
-        # we can only move it left if...what?
-        my $parent = parentkey($currentkey);
-
-        if ( $parent =~ m/^{0.0}.*/ && $parent ne "{0.0}" ) {
-            my $newkey = appendkey( $items, $parent );
-            movecurrent( $items, $newkey, \$currentkey );
-
-            # if there was an empty group left behind, delete it
-            my $subhash = gethash($items, $parent . "{Subkey}");
-            if ( !( keys %{$subhash} ) ) {
-                deletevalue($items, $parent);
-                reworkkeys( $items, $parent, \$currentkey );
-            }
-
+	my $parent = $currentkey->getParent();
+	my $grandparent = $parent->getParent();
+	if (!$grandparent->isRoot) {
+	    my $index = $parent->getIndex();
+	    $parent->removeChild($currentkey);
+	    $grandparent->insertChild($index, $currentkey);
+	    if ($parent->isLeaf()) {
+		$grandparent->removeChild($parent);
+	    }
         }
         else {
             push( @actions, [ "error: can't move left", -1 ] );
@@ -328,44 +321,60 @@
 }
 elsif ( $ARGS{"Right"} ) {
     if ($currentkey) {
-        # you can't move right if you leave no siblings behind
-        if (   keyexists( $items, nextkey($currentkey) )
-            || keyexists( $items, prevkey($currentkey) ) )
-        {
-
-            # if the next item is a subkey, put it there
-            my $newkey = appendsubkey( $items, $currentkey );
-
-            movecurrent( $items, $newkey, \$currentkey );
-        }
-        else {
-            push( @actions, [ "error: can't move right", -1 ] );
-        }
-
-    }
-    else {
+	my $parent = $currentkey->getParent();
+	my $index = $currentkey->getIndex();
+	my $newparent;
+	if ($index > 0 ) {
+	    my $sibling = $parent->getChild($index - 1);
+	    if (ref($sibling)) {
+		$parent->removeChild($currentkey);
+		my $newtree = Tree::Simple->new('AND', $parent);
+		$newtree->addChild($currentkey);
+	    } else {
+		$parent->removeChild($index);
+		$sibling->addChild($currentkey);
+	    }
+	}
+	else {
+	    $parent->removeChild($currentkey);
+	    $newparent = Tree::Simple->new('AND', $parent);
+	    $newparent->addChild($currentkey);
+	}
+    } else {
         push( @actions, [ "error: nothing to move", -1 ] );
     }
 }
 elsif ( $ARGS{"DeleteClause"} ) {
     if ($currentkey) {
-        deletevalue($items, $currentkey);
-        reworkkeys( $items, $currentkey, \$currentkey );
+	$currentkey->getParent()->removeChild($currentkey);
     }
     else {
         push( @actions, [ "error: nothing to delete", -1 ] );
     }
 }
 elsif ( $ARGS{"Toggle"} ) {
-    my $ea = getaggregator( $items, $currentkey);
-    if ($ea eq 'AND') {
-        setaggregator( $items, $currentkey, 'OR' );
-    } else {
-        setaggregator( $items, $currentkey, 'AND' );
+    my $ea;
+    if ($currentkey) {
+	if (!ref($currentkey->getNodeValue())) {
+	    if ($currentkey->getNodeValue() eq 'AND') {
+		$currentkey->setNodeValue('OR');
+	    } else {
+		$currentkey->setNodeValue('AND');
+	    }
+	} else {
+	    if ($currentkey->getParent()->getNodeValue() eq 'AND') {
+		$currentkey->getParent()->setNodeValue('OR');
+	    } else {
+		$currentkey->getParent()->setNodeValue('AND');
+	    }
+	}
+    }
+    else {
+        push( @actions, [ "error: nothing to toggle", -1 ] );
     }
 }
 elsif ( $ARGS{"Clear"} ) {
-    $items = ();
+    $tree = Tree::Simple->new(Tree::Simple->ROOT);
 }
 
 # }}}
@@ -374,95 +383,81 @@
 $Query   = "";
 @options = ();
 %queues  = ();
-build_array( \$Query, $items, "", "", 0, \@options, \%queues );
-my $ea;
-my $i          = 0;
-my $optionlist = "";
-while ( my $val = shift @options ) {
-    last if ( !$val->{Value} );
-    my $key = $val->{Key};
-    my ( $prefix, $depth, $num ) = parsekey($key);
-    my $selected;
+$optionlist = build_array( \$Query, $currentkey, $tree, \@options, \%queues );
+
+sub build_array {
+    my $Query     = shift;
+    my $currentkey = shift;
+    my $tree = shift;
+    my ($keys, $queues)    = @_;
+    my $i = 0;
+    my $optionlist;
+    my $depth = 0;
+    my %parens;
 
-    next unless ( $val->{Value}->{Key} || $val->{Value}->{Subkey} );
-    $ea = $val->{EA};
+    $tree->traverse(
+    sub {
+    my ($_tree) = @_;
+
+    return if $_tree->getParent->isRoot();
+
+    push @$keys, $_tree;
+    my $clause = $_tree->getNodeValue();
+    my $str;
+    if (ref($clause)) {
+	my $ea = $_tree->getParent()->getNodeValue();
+        $str .= $ea . " " if $_tree->getIndex() > 0;
+	$str .= $clause->{Key} . " " . $clause->{Op} . " " . $clause->{Value};
 
-    if ( $depth > 0 ) {
-        if ( $key eq $currentkey ) {
-            $selected = "SELECTED";
-        }
-        else {
-            $selected = "";
-        }
-        $optionlist .= "<option value=$i $selected>";
-        $optionlist .= " " x ( 5 * ( $depth - 1 ) );
-#        $optionlist .= "$i: $key ";
+	if ( $clause->{Key} eq "Queue" ) {
+	    $queues->{ $clause->{Value} } = 1;
+	}
+    } else {
+	$str = $_tree->getNodeValue() if $_tree->getIndex() > 0;
+    }
 
-        if ( $num > 0 || $val->{Value}->{Subkey} ) {
-            $optionlist .= $ea . " ";
-        }
-        $optionlist .=
-          "$val->{Value}->{Key} $val->{Value}->{Op} $val->{Value}->{Value}";
-        $optionlist .= "</option>\n";
+    my $selected;
+    if ($_tree == $currentkey) {
+	$selected = "SELECTED";
+    }
+    else {
+	$selected = "";
+    }
 
+    foreach my $p (keys %parens) {
+	if ($p > $_tree->getDepth) {
+	    $$Query .= ')' x $parens{$p};
+	    $parens{$p}--;
+	}
     }
+
+    $optionlist .= "<option value=$i $selected>" .
+	(" " x 5 x ($_tree->getDepth() - 1)) . "$str</option>\n";
+    my $parent = $_tree->getParent();
+    if (!($parent->isRoot || $parent->getParent()->isRoot) &&
+	!ref($parent->getNodeValue())) {
+	if ( $_tree->getIndex() == 0) {
+	    $$Query .= '(';
+	    $parens{$_tree->getDepth}++;
+	}
+    }
+    $$Query .= " " . $str . " ";
+
+    if ($_tree->getDepth < $depth) {
+	$$Query .= ')';
+	$parens{$depth}--;
+    }
+
     $i++;
 }
+);
 
-sub build_array {
-    my $Query     = shift;
-    my $items     = shift;
-    my $parentkey = shift;
-    my $ea        = shift;
-    my $i         = shift;
-    my ($keys, $queues)    = @_;
+foreach my $p (keys %parens) {
+    $$Query .= ") " x $parens{$p};
+}
+
+return $optionlist;
 
-    while ( my $item = $items->{ $i + 0 } ) {
-        my $j = 0;
-        while ( my $item = $items->{ $i + ( $j / 10 ) } ) {
-
-            my $depth = $i;
-
-            my $subkey = $item->{Subkey};
-            if ( defined $subkey && scalar keys %$subkey ) {
-                my $sendkey = "";
-                if ($parentkey) {
-                    $sendkey = $parentkey . "{Subkey}{$i.$j}";
-                }
-                else {
-                    $sendkey = "{$i.$j}";
-                }
-                push @$keys, { Key => $sendkey, Value => $item, EA => $ea };
-                # no aggregator if it's first
-                $$Query .= " " . $ea . " " if ($j > 0 && $$Query); 
-                $$Query .= "("             if $depth > 0;
-                my $x = $i + 1;
-                build_array( $Query, $subkey, $sendkey, $item->{EA}, 
-			     $x, $keys, , $queues );
-                $$Query .= ")" if $depth > 0;
-            }
-            else {
-                if ( $depth >= 1 ) {
-                    my $mykey;
-                    if ($parentkey) {
-                        $mykey = $parentkey . "{Subkey}{$i.$j}";
-                    }
-                    else {
-                        $mykey = " " . $i . $j . " ";
-                    }
-                    $$Query .= " " . $ea . " " if ( $j > 0 );
-                    $$Query .= "$item->{Key} $item->{Op} $item->{Value}" if $item->{Key};
-                    push @$keys, { Key => $mykey, Value => $item, EA => $ea };
-
-                    if ( $item->{Key} eq "Queue" ) {
-                        $queues->{ $item->{Value} } = 1;
-                    }
-                }
-            }
-            $j++;
-        }
-        $i++;
-    }
 }
 
 use Regexp::Common qw /delimited/;
@@ -476,19 +471,18 @@
 
 sub ParseQuery {
     my $string = shift;
+    my $tree = shift;
     my @actions = shift;
     my $want   = KEYWORD | PAREN;
     my $last   = undef;
 
     my $depth = 1;
-    my $query = {};
-    my %depths;
-    my %items;
-
-    setvalue($query, "{0.0}", {});
-    setvalue($query, "{0.0}{Subkey}", {});
-    setvalue($query, "{0.0}{Subkey}{1.0}", {});
-    setvalue($query, "{0.0}{Subkey}{1.0}{Subkey}", {});
+
+    # make a tree root
+    $$tree = Tree::Simple->new(Tree::Simple->ROOT);
+    my $root = Tree::Simple->new('AND', $$tree);
+    my $lastnode = $root;
+    my $parentnode = $root;
 
     # get the FIELDS from Tickets_Overlay
     my $tickets = new RT::Tickets( $session{'CurrentUser'} );
@@ -504,14 +498,13 @@
     my $re_op      = qr[=|!=|>=|<=|>|<|(?i:IS NOT)|(?i:IS)|(?i:NOT LIKE)|(?i:LIKE)] ;    # long to short
     my $re_paren = qr'\(|\)';
 
-    my ( $ea, $key, $op, $value ) = ( "", "", "", "" );
+    # assume that $ea is AND if it's not set
+    my ( $ea, $key, $op, $value ) = ( "AND", "", "", "" );
 
     # order of matches in the RE is important.. op should come early,
     # because it has spaces in it.  otherwise "NOT LIKE" might be parsed
     # as a keyword or value.
 
-    my $num = 0;
-    $depths{1} = 0;
     while ( $string =~ /(
                       $re_aggreg
                       |$re_op
@@ -546,44 +539,20 @@
         # Parens are highest priority
         if ( $current & PAREN ) {
             if ( $val eq "(" ) {
-                $num = 0;
-                $depth++;
-		$depths{$depth} = 0;
-
-                my $hashkey;
-                my @keys;
-                $hashkey = "$depth" . "." . "$depths{$depth}" . "";
-                my $keystring = "";
-
-                while ( $parentdepth >= 1 ) {
-                    $hashkey = $parentdepth . ".$depths{$parentdepth}";
-                    push @keys, $hashkey;
-                    $parentdepth--;
-                }
-
-                # build up the keystring from the top
-                $keystring .= "{" . ( pop @keys ) . "}";
-                while ( my $k = pop @keys ) {
-                    $keystring .= "{Subkey}{$k}";
-                }
-                $keystring = "{0.0}{Subkey}" . $keystring;
-
-		if (!gethash($query, parentkey($keystring))) {
-		    setvalue($query, parentkey($keystring), {});
-		}
-                setaggregator( $query, $keystring, $ea );
+		$depth++;
+		# make a new node that the clauses can be children of
+		$parentnode = Tree::Simple->new($ea, $parentnode);
             }
             else {
-                $depth--;
-                $num = $depths{$depth} + 1;
+		$depth--;
+		$parentnode = $parentnode->getParent();
+		$lastnode = $parentnode;
             }
 
             $want = KEYWORD | PAREN | AGGREG;
         }
         elsif ( $current & AGGREG ) {
             $ea = $val;
-	    $depths{$depth}++ if ($last & PAREN);
-
             $want = KEYWORD | PAREN;
         }
         elsif ( $current & KEYWORD ) {
@@ -630,23 +599,6 @@
         }
 
         if ( $current & VALUE ) {
-	    $depths{$depth}++;
-
-            my $keystring = "";
-            $keystring = "{Subkey}{" . $parentdepth . ".$num" . "}";
-            $parentdepth--;
-
-            while ( $parentdepth > 0 ) {
-                $keystring = "{Subkey}{"
-                  . $parentdepth
-                  . ".$depths{$parentdepth}" . "}"
-                  . $keystring;
-                $parentdepth--;
-            }
-
-	    $ea = 'AND' if !$ea;
-            $keystring = "{0.0}" . $keystring;
-
 	    if ( $key =~ /^CF./ ) {
 	        $key = "'" . $key . "'";
 	    }
@@ -656,10 +608,9 @@
                 Value => $val
             };
 
-	    setvalue($query, $keystring, $clause);
-            setaggregator( $query, $keystring, $ea );
-
-            $num++;
+	    # explicity add a child to it
+	    $lastnode = Tree::Simple->new($clause, $parentnode);
+	    $lastnode->getParent()->setNodeValue($ea);
 
             ( $ea, $key, $op, $value ) = ( "", "", "", "" );
         }
@@ -676,8 +627,6 @@
     # This will never happen, because the parser will complain
     push @actions, [ "Mismatched parentheses", -1 ]
       unless $depth == 1;
-
-    return $query;
 }
 
 sub _match {
@@ -690,306 +639,6 @@
     return 0;
 }
 
-sub keyexists {
-    my $hash = shift;
-    my $key  = shift;
-
-    my ($prefix, $depth, $num) = parsekey($key);
-    return exists $hash->{0.0} if ($depth == 0);
-
-    my $myhash = gethash($hash, $prefix);
-
-    return exists ($myhash->{$depth + ($num/10)});
-}
-
-sub subkeyexists {
-    my $hash = shift;
-    my $key  = shift;
-
-    my $myhash = gethash($hash, $key);
-    return exists $myhash->{Subkey};
-}
-
-sub appendsubkey {
-    my $hash = shift;
-    my $key  = shift;
-
-    my $newkey;
-
-    # if the next key exists, get the last key
-    my $nextkey = nextkey($key);
-    if ( subkeyexists( $hash, $nextkey ) ) {
-        $newkey = appendkey( $hash, subkey( $nextkey ) );
-    }
-    else {
-        $newkey = subkey($key);
-    }
-
-    return $newkey;
-}
-
-sub prevkey {
-    my $key = shift;
-    my ( $prefix, $depth, $num ) = parsekey($key);
-
-    my $k = "$depth." . ( $num - 1 );
-    return $prefix . "{$k}";
-}
-
-sub nextkey {
-    my $key = shift;
-    my ( $prefix, $depth, $num ) = parsekey($key);
-
-    my $k = "$depth." . ( $num + 1 );
-    
-    return $prefix . "{$k}";
-}
-
-sub subkey {
-    my $key = shift;
-    my ( $prefix, $depth, $num ) = parsekey($key);
-
-    return $prefix . "{" . "$depth.$num" . "}{Subkey}{" . ( $depth + 1 ) . ".0}";
-}
-
-sub parentkey {
-    my $key = shift;
-    my ( $prefix, $depth, $num ) = parsekey($key);
-    if ( $depth > 0 ) {
-        $prefix =~ s/(.*){Subkey}/$1/;
-    }
-    else {
-        $prefix = "";
-    }
-
-    return $prefix;
-}
-
-sub parsekey {
-    my $key = shift;
-
-    # pull apart the key
-    $key =~ m/(.*){(.*)}$/;
-    my $prefix = $1;
-    my $depth = 0;
-    my $num = 0;
-    my $final = $2;
-    if ($final =~ m/(\d).(\d)/) {
-	$depth  = $1;
-	$num    = $2;
-    } else {
-	$depth = $final;
-    }
-    return ( $prefix, $depth, $num );
-}
-
-sub reworkkeys {
-    my $hash = shift;
-    my $key  = shift;
-    my $currentkey = shift;
-
-    # if our parent doesn't exist, return immediately
-    if ( !keyexists( $hash, parentkey($key) ) ) {
-        return;
-    }
-
-    fixsubkeys( $hash, $key );
-
-    # don't try to rework keys if the first one exists
-    if (   keyexists( $hash, $key )
-        || subkeyexists( $hash, $key ) )
-    {
-        return;
-    }
-
-    my $nextkey = nextkey($key);
-    while ( keyexists( $hash, $nextkey ) ) {
-        if ( keyexists( $hash, $nextkey ) ) {
-	    my $temp = gethash($hash, $nextkey);
-	    setvalue($hash, $key, $temp);
-            $$currentkey = $key . $2 if $$currentkey =~ m/($nextkey)(.*)/;
-        }
-
-        # set this so that we can return it;
-        $key = $nextkey;
-
-        if ( subkeyexists( $hash, $key ) ) {
-            my $subkey = subkey($key);
-            reworkkeys( $hash, subkey($key), $currentkey );
-        }
-
-        $nextkey = nextkey($key);
-    }
-    deletevalue($hash, $key);
-}
-
-sub fixsubkeys {
-    my $hash = shift;
-    my $key  = shift;
-
-    if ( subkeyexists( $hash, $key ) ) {
-        my ( $prefix, $depth, $num ) = parsekey($key);
-        my $item = gethash($hash, $key . "{Subkey}");
-	my %temp = %$item;
-        foreach my $i ( keys %temp ) {
-            my $num = $i - int($i);
-            my $new = int($depth) + 1 + $num;
-            if ( $i != $new ) {
-		setvalue($hash, $key . "{Subkey}{$new}", $temp{$i});
-		deletevalue($hash, $key . "{Subkey}{$i}");
-                fixsubkeys( $hash, $key . "{Subkey}{$new}" );
-            }
-            $num++;
-        }
-    }
-}
-
-sub getaggregator {
-    my $hash = shift;
-    my $key  = shift;
-
-    my $parent = parentkey($key);
-    my $ea = gethash($hash, $parent . "{EA}");
-
-    return $ea;
-}
-
-sub setaggregator {
-    my $hash = shift;
-    my $key  = shift;
-    my $ea   = shift;
-
-    my $parentkey = parentkey($key);
-    my $parenthash = gethash($hash, parentkey($key));
-    $parenthash->{EA} = $ea;
-
-}
-
-sub appendkey {
-    my $hash = shift;
-    my $key  = shift;
-
-    my $newkey;
-    if ($key) {
-        $newkey = $key;
-    }
-    else {
-        $newkey = "{" . "0.0" . "}{Subkey}{1.0}";
-    }
-
-    while ( keyexists( $hash, $newkey ) ) {
-        $newkey = nextkey($newkey);
-    }
-
-    return $newkey;
-}
-
-sub movecurrent {
-    my $hash   = shift;
-    my $newkey = shift;
-    my $currentkey = shift;
-
-    my $origkey = $$currentkey;
-    my $old = gethash($hash, $origkey);
-    my %temp = %$old;
-
-    deletevalue($hash, $origkey);
-    my $orignewkey = $newkey;
-
-    setvalue($hash, $newkey, \%temp);
-
-    $$currentkey = $orignewkey;
-    reworkkeys( $hash, $orignewkey, $currentkey );
-    reworkkeys( $hash, $origkey, $currentkey );
-
-    return 1;
-}
-
-sub swap {
-    my $hash = shift;
-    my $key1 = shift;
-    my $key2 = shift;
-
-    return 0 if !( keyexists( $hash, $key1 ) && keyexists( $hash, $key2 ) );
-
-    # store the value temporarily
-    my $val1 = gethash($hash, $key1);
-    my %temp = %$val1;
-
-    my $val2 = gethash($hash, $key2);
-
-    setvalue($hash, $key2, \%temp);
-    setvalue($hash, $key1, $val2);
-
-    return 1;
-}
-
-sub gethash {
-    my $hash = shift;
-    my $key = shift;
-
-    my $returnhash;
-
-    $key =~ s/^{0.0}(.*)$/$1/;
-    $key =~ s/^{(.*)}$/$1/;
-    my @keys = split(/\}\{/, $key);
-    $returnhash = $hash->{0.0};
-    foreach my $k (@keys) {
-	if (exists $returnhash->{$k}) {
-	    $returnhash = $returnhash->{$k};
-	} else {
-	    $returnhash = $returnhash->{$k+0};
-	}
-    }
-
-    return $returnhash;
-}
-
-sub setvalue {
-    my $hash = shift;
-    my $key = shift;
-    my $value = shift;
-
-    my ($prefix, $depth, $num) = parsekey($key);
-    my $newhash;
-    if ($prefix) {
-	my $parentkey = parentkey($key);
-
-	my $parent;
-	if (!gethash($hash, $parentkey)) {
-	    setvalue($hash, $parentkey, {});
-	    $parent = gethash($hash, $parentkey);
-	} else {
-	    $parent = gethash($hash, $parentkey);
-	}
-
-	if (!exists $parent->{Subkey}) {
-	    $parent->{Subkey} = {};
-	}
-	$newhash = gethash($hash, $prefix);
-    } else {
-	$newhash = $hash;
-    }
-
-    $newhash->{$depth + ($num / 10)} = $value;
-
-    # if there isn't an aggregator, set it to AND
-    my $ea = getaggregator( $hash, $key);
-    if (!$ea) {
-	setaggregator( $hash, $key, 'AND' );
-    }
-
-}
-
-sub deletevalue {
-    my $hash = shift;
-    my $key = shift;
-
-    my ($prefix, $depth, $num) = parsekey($key);
-    my $parent = gethash($hash, $prefix);
-    delete $parent->{$depth + ($num / 10)};
-}
-
 sub debug {
     my $message = shift;
     $m->print($message . "<br>");
Modified: rt/branches/rt-3.3/html/Search/Bulk.html
==============================================================================
--- rt/branches/rt-3.3/html/Search/Bulk.html	(original)
+++ rt/branches/rt-3.3/html/Search/Bulk.html	Mon May 31 21:54:38 2004
@@ -191,6 +191,11 @@
     $RT::Logger->debug( "Checking Ticket ".$Ticket->Id ."\n");
     next unless ($ARGS{"UpdateTicket".$Ticket->Id});
     $RT::Logger->debug ("Matched\n");
+    my @updateresults; 
+    if ($do_comment_reply) {
+       ProcessUpdateMessage(TicketObj => $Ticket, ARGSRef => \%ARGS, Actions => \@updateresults); 
+    }
+
     #Update the basics.
     my @basicresults = ProcessTicketBasics(TicketObj => $Ticket, ARGSRef => \%ARGS);
     my @dateresults = ProcessTicketDates(TicketObj => $Ticket, ARGSRef => \%ARGS);
@@ -216,11 +221,7 @@
     delete $ARGS{$Ticket->Id.'-RefersTo'};
     delete $ARGS{'RefersTo-'.$Ticket->Id};
     
-    my @updateresults; 
-    if ($do_comment_reply) {
-    ProcessUpdateMessage(TicketObj => $Ticket, ARGSRef => \%ARGS, Actions => \@updateresults); 
-   } 
-   my @tempresults = (@watchresults, @basicresults, @dateresults, @updateresults, @linkresults);
+    my @tempresults = (@watchresults, @basicresults, @dateresults, @updateresults, @linkresults);
     @tempresults = map { loc("Ticket [_1]: [_2]",$Ticket->Id,$_) } @tempresults;
 
     @results = (@results, @tempresults);
Modified: rt/branches/rt-3.3/lib/RT.pm.in
==============================================================================
--- rt/branches/rt-3.3/lib/RT.pm.in	(original)
+++ rt/branches/rt-3.3/lib/RT.pm.in	Mon May 31 21:54:38 2004
@@ -51,6 +51,8 @@
 $CORE_CONFIG_FILE = "@CONFIG_FILE_PATH@/RT_Config.pm";
 $SITE_CONFIG_FILE = "@CONFIG_FILE_PATH@/RT_SiteConfig.pm";
 
+ at DATABASE_ENV_PREF@
+
 $BasePath = '@RT_PATH@';
 
 $EtcPath = '@RT_ETC_PATH@';
Modified: rt/branches/rt-3.3/lib/RT/EmailParser.pm
==============================================================================
--- rt/branches/rt-3.3/lib/RT/EmailParser.pm	(original)
+++ rt/branches/rt-3.3/lib/RT/EmailParser.pm	Mon May 31 21:54:38 2004
@@ -628,7 +628,7 @@
 sub _SetupMIMEParser {
     my $self   = shift;
     my $parser = shift;
-
+	
     #$RT::Logger->crit(  $self->{'AttachmentDir'} );
     # Set up output directory for files:
 
Modified: rt/branches/rt-3.3/lib/RT/Handle.pm
==============================================================================
--- rt/branches/rt-3.3/lib/RT/Handle.pm	(original)
+++ rt/branches/rt-3.3/lib/RT/Handle.pm	Mon May 31 21:54:38 2004
@@ -48,7 +48,13 @@
 
 eval "use DBIx::SearchBuilder::Handle::$RT::DatabaseType;
 \@ISA= qw(DBIx::SearchBuilder::Handle::$RT::DatabaseType);";
-#TODO check for errors here.
+
+if ($@) {
+    die "Unable to load DBIx::SearchBuilder database handle for '$RT::DatabaseType'.".
+        "\n".
+        "Perhaps you've picked an invalid database type or spelled it incorrectly.".
+        "\n". $@;
+}
 
 =head2 Connect
 
Modified: rt/branches/rt-3.3/lib/RT/Tickets_Overlay_SQL.pm
==============================================================================
--- rt/branches/rt-3.3/lib/RT/Tickets_Overlay_SQL.pm	(original)
+++ rt/branches/rt-3.3/lib/RT/Tickets_Overlay_SQL.pm	Mon May 31 21:54:38 2004
@@ -468,7 +468,11 @@
 sub FromSQL {
   my ($self,$query) = @_;
 
-  $self->CleanSlate;
+  {
+    # preserve first_row and show_rows across the CleanSlate
+    local($self->{'first_row'}, $self->{'show_rows'});
+    $self->CleanSlate;
+  }
   $self->_InitSQL();
 
   return (1,$self->loc("No Query")) unless $query;
Modified: rt/branches/rt-3.3/sbin/rt-setup-database.in
==============================================================================
--- rt/branches/rt-3.3/sbin/rt-setup-database.in	(original)
+++ rt/branches/rt-3.3/sbin/rt-setup-database.in	Mon May 31 21:54:38 2004
@@ -277,6 +277,10 @@
         do $base_path . "/acl.mysql"
           || die "Couldn't find ACLS for mysql in " . $RT::EtcPath . "\n" . $@;
     }
+    elsif ( $RT::DatabaseType =~ /^Sybase$/i ) {
+        do $base_path . "/acl.Sybase"
+          || die "Couldn't find ACLS for Sybase in " . $RT::EtcPath . "\n" . $@;
+    }
     elsif ( $RT::DatabaseType =~ /^informix$/i ) {
         do $base_path . "/acl.Informix"
           || die "Couldn't find ACLS for Informix in " . $RT::EtcPath . "\n" . $@;
Modified: rt/branches/rt-3.3/sbin/rt-test-dependencies.in
==============================================================================
--- rt/branches/rt-3.3/sbin/rt-test-dependencies.in	(original)
+++ rt/branches/rt-3.3/sbin/rt-test-dependencies.in	Mon May 31 21:54:38 2004
@@ -116,6 +116,7 @@
 Term::ReadKey
 Text::Autoformat
 Text::Quoted 1.3
+Tree::Simple 1.04
 Scalar::Util
 Module::Versions::Report
 .
    
    
More information about the Rt-commit
mailing list