/[cvs]/nfo/patches/cpan/SQL/Parser.pm
ViewVC logotype

Annotation of /nfo/patches/cpan/SQL/Parser.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1.4 - (hide annotations)
Mon Dec 2 05:53:54 2002 UTC (22 years, 1 month ago) by joko
Branch: MAIN
CVS Tags: HEAD
Changes since 1.3: +1 -1 lines
+ re-applied modification: make dots (.) possible in table-names

1 joko 1.1 ######################################################################
2     package SQL::Parser;
3     ######################################################################
4     #
5 joko 1.3 # This module is copyright (c), 2001,2002 by Jeff Zucker.
6 joko 1.1 # All rights resered.
7 joko 1.3 #
8 joko 1.1 # It may be freely distributed under the same terms as Perl itself.
9     # See below for help and copyright information (search for SYNOPSIS).
10     #
11     ######################################################################
12    
13     use strict;
14     use vars qw($VERSION);
15 joko 1.3 use constant FUNCTION_NAMES => join '|', qw(
16     TRIM SUBSTRING UPPER LOWER TO_CHAR
17     );
18    
19     $VERSION = '1.005';
20 joko 1.1
21 joko 1.3 BEGIN { if( $ENV{SQL_USER_DEFS} ) { require SQL::UserDefs; } }
22 joko 1.1
23    
24     #############################
25     # PUBLIC METHODS
26     #############################
27    
28     sub new {
29     my $class = shift;
30     my $dialect = shift || 'ANSI';
31     $dialect = 'ANSI' if uc $dialect eq 'ANSI';
32     $dialect = 'AnyData' if uc $dialect eq 'ANYDATA' or uc $dialect eq 'CSV';
33     # $dialect = 'CSV' if uc $dialect eq 'CSV';
34     if ($dialect eq 'SQL::Eval') {
35     $dialect = 'AnyData';
36     }
37     my $flags = shift || {};
38     $flags->{"dialect"} = $dialect;
39     $flags->{"PrintError"} = 1 unless defined $flags->{"PrintError"};
40     my $self = bless_me($class,$flags);
41     $self->dialect( $self->{"dialect"} );
42     $self->set_feature_flags($self->{"select"},$self->{"create"});
43     return bless $self,$class;
44     }
45    
46     sub parse {
47     my $self = shift;
48     my $sql = shift;
49     #printf "<%s>", $self->{dialect_set};
50     $self->dialect( $self->{"dialect"} ) unless $self->{"dialect_set"};
51     $sql =~ s/^\s+//;
52     $sql =~ s/\s+$//;
53     $self->{"struct"} = {};
54     $self->{"tmp"} = {};
55     $self->{"original_string"} = $sql;
56    
57     ################################################################
58     #
59     # COMMENTS
60    
61     # C-STYLE
62     #
63     my $comment_re = $self->{"comment_re"} || '(\/\*.*?\*\/)';
64     $self->{"comment_re"} = $comment_re;
65     my $starts_with_comment;
66     if ($sql =~ /^\s*$comment_re(.*)$/s) {
67     $self->{"comment"} = $1;
68     $sql = $2;
69     $starts_with_comment=1;
70     }
71     # SQL STYLE
72     #
73     if ($sql =~ /^\s*--(.*)(\n|$)/) {
74     $self->{"comment"} = $1;
75     return 1;
76     }
77     ################################################################
78    
79     $sql = $self->clean_sql($sql);
80     my($com) = $sql =~ /^\s*(\S+)\s+/s ;
81     if (!$com) {
82     return 1 if $starts_with_comment;
83     return $self->do_err("Incomplete statement!");
84     }
85     $com = uc $com;
86     if ($self->{"opts"}->{"valid_commands"}->{$com}) {
87     my $rv = $self->$com($sql);
88     delete $self->{"struct"}->{"literals"};
89     # return $self->do_err("No table names found!")
90     # unless $self->{"struct"}->{"table_names"};
91     return $self->do_err("No command found!")
92     unless $self->{"struct"}->{"command"};
93     if ( $self->{"struct"}->{"join"}
94     and scalar keys %{$self->{"struct"}->{"join"}}==0
95     ) {
96     delete $self->{"struct"}->{"join"};
97     }
98     $self->replace_quoted_ids();
99 joko 1.3 $self->{struct}->{org_table_names} = $self->{struct}->{table_names};
100     my @uTables = map {uc $_ } @{$self->{struct}->{table_names}};
101     $self->{struct}->{table_names} = \@uTables;
102     $self->{struct}->{org_col_names} = $self->{struct}->{column_names};
103     my @uCols = map {uc $_ } @{$self->{struct}->{column_names}};
104     $self->{struct}->{column_names} = \@uCols;
105     if ($self->{original_string} =~ /Y\.\*/) {
106     #use mylibs; zwarn $self; exit;
107     }
108 joko 1.1 return $rv;
109     }
110     else {
111 joko 1.3 $self->{struct}={};
112     if ($ENV{SQL_USER_DEFS}) {
113     return SQL::UserDefs::user_parse($self,$sql);
114     }
115 joko 1.1 return $self->do_err("Command '$com' not recognized or not supported!");
116     }
117     }
118    
119     sub replace_quoted_ids {
120     my $self = shift;
121     my $id = shift;
122     return $id unless $self->{struct}->{quoted_ids};
123     if ($id) {
124     if ($id =~ /^\?QI(\d+)\?$/) {
125     return '"'.$self->{struct}->{quoted_ids}->[$1].'"';
126     }
127     else {
128     return $id;
129     }
130     }
131     my @tables = @{$self->{struct}->{table_names}};
132     for my $t(@tables) {
133     if ($t =~ /^\?QI(.+)\?$/ ) {
134     $t = '"'.$self->{struct}->{quoted_ids}->[$1].'"';
135     # $t = $self->{struct}->{quoted_ids}->[$1];
136     }
137     }
138     $self->{struct}->{table_names} = \@tables;
139     delete $self->{struct}->{quoted_ids};
140     }
141    
142     sub structure { shift->{"struct"} }
143     sub command { my $x = shift->{"struct"}->{command} || '' }
144    
145     sub feature {
146     my($self,$opt_class,$opt_name,$opt_value) = @_;
147     if (defined $opt_value) {
148     if ( $opt_class eq 'select' ) {
149     $self->set_feature_flags( {"join"=>$opt_value} );
150     }
151     elsif ( $opt_class eq 'create' ) {
152     $self->set_feature_flags( undef, {$opt_name=>$opt_value} );
153     }
154     else {
155     $self->{$opt_class}->{$opt_name} = $opt_value;
156     }
157     }
158     else {
159     return $self->{"opts"}->{$opt_class}->{$opt_name};
160     }
161     }
162    
163     sub errstr { shift->{"struct"}->{"errstr"} }
164    
165     sub list {
166     my $self = shift;
167     my $com = uc shift;
168     return () if $com !~ /COMMANDS|RESERVED|TYPES|OPS|OPTIONS|DIALECTS/i;
169     $com = 'valid_commands' if $com eq 'COMMANDS';
170     $com = 'valid_comparison_operators' if $com eq 'OPS';
171     $com = 'valid_data_types' if $com eq 'TYPES';
172     $com = 'valid_options' if $com eq 'OPTIONS';
173     $com = 'reserved_words' if $com eq 'RESERVED';
174     $self->dialect( $self->{"dialect"} ) unless $self->{"dialect_set"};
175    
176     return sort keys %{ $self->{"opts"}->{$com} } unless $com eq 'DIALECTS';
177     my $dDir = "SQL/Dialects";
178     my @dialects;
179     for my $dir(@INC) {
180     local *D;
181    
182     if ( opendir(D,"$dir/$dDir") ) {
183     @dialects = grep /.*\.pm$/, readdir(D);
184     last;
185     }
186     }
187     @dialects = map { s/\.pm$//; $_} @dialects;
188     return @dialects;
189     }
190    
191     sub dialect {
192     my($self,$dialect) = @_;
193     return $self->{"dialect"} unless $dialect;
194     return $self->{"dialect"} if $self->{dialect_set};
195     $self->{"opts"} = {};
196     my $mod = "SQL/Dialects/$dialect.pm";
197     undef $@;
198     eval {
199     require "$mod";
200     };
201     return $self->do_err($@) if $@;
202     $mod =~ s/\.pm//;
203     $mod =~ s"/"::"g;
204     my @data = split /\n/, $mod->get_config;
205     my $feature;
206     for (@data) {
207     chomp;
208     s/^\s+//;
209     s/\s+$//;
210     next unless $_;
211     if (/^\[(.*)\]$/i) {
212     $feature = lc $1;
213     $feature =~ s/\s+/_/g;
214     next;
215     }
216     my $newopt = uc $_;
217     $newopt =~ s/\s+/ /g;
218     $self->{"opts"}->{$feature}->{$newopt} = 1;
219     }
220     $self->{"dialect"} = $dialect;
221     $self->{"dialect_set"}++;
222     }
223    
224     ##################################################################
225     # SQL COMMANDS
226     ##################################################################
227    
228     ####################################################
229     # DROP TABLE <table_name>
230     ####################################################
231     sub DROP {
232     my $self = shift;
233     my $stmt = shift;
234     my $table_name;
235     $self->{"struct"}->{"command"} = 'DROP';
236 joko 1.3 if ($stmt =~ /^\s*DROP\s+TABLE\s+IF\s+EXISTS\s+(.*)$/si ) {
237     $stmt = "DROP TABLE $1";
238     $self->{"struct"}->{ignore_missing_table}=1;
239     }
240 joko 1.1 if ($stmt =~ /^\s*DROP\s+(\S+)\s+(.+)$/si ) {
241     my $com2 = $1 || '';
242     $table_name = $2;
243     if ($com2 !~ /^TABLE$/i) {
244     return $self->do_err(
245     "The command 'DROP $com2' is not recognized or not supported!"
246     );
247     }
248     $table_name =~ s/^\s+//;
249     $table_name =~ s/\s+$//;
250     if ( $table_name =~ /(\S+) (RESTRICT|CASCADE)/i) {
251     $table_name = $1;
252     $self->{"struct"}->{"drop_behavior"} = uc $2;
253     }
254     }
255     else {
256     return $self->do_err( "Incomplete DROP statement!" );
257    
258     }
259     return undef unless $self->TABLE_NAME($table_name);
260     $table_name = $self->replace_quoted_ids($table_name);
261     $self->{"tmp"}->{"is_table_name"} = {$table_name => 1};
262     $self->{"struct"}->{"table_names"} = [$table_name];
263     return 1;
264     }
265    
266     ####################################################
267     # DELETE FROM <table_name> WHERE <search_condition>
268     ####################################################
269     sub DELETE {
270     my($self,$str) = @_;
271     $self->{"struct"}->{"command"} = 'DELETE';
272     my($table_name,$where_clause) = $str =~
273     /^DELETE FROM (\S+)(.*)$/i;
274     return $self->do_err(
275     'Incomplete DELETE statement!'
276     ) if !$table_name;
277     return undef unless $self->TABLE_NAME($table_name);
278     $self->{"tmp"}->{"is_table_name"} = {$table_name => 1};
279     $self->{"struct"}->{"table_names"} = [$table_name];
280     $self->{"struct"}->{"column_names"} = ['*'];
281     $where_clause =~ s/^\s+//;
282     $where_clause =~ s/\s+$//;
283     if ($where_clause) {
284     $where_clause =~ s/^WHERE\s*(.*)$/$1/i;
285     return undef unless $self->SEARCH_CONDITION($where_clause);
286     }
287     return 1;
288     }
289    
290     ##############################################################
291     # SELECT
292     ##############################################################
293     # SELECT [<set_quantifier>] <select_list>
294     # | <set_function_specification>
295     # FROM <from_clause>
296     # [WHERE <search_condition>]
297     # [ORDER BY <order_by_clause>]
298     # [LIMIT <limit_clause>]
299     ##############################################################
300    
301     sub SELECT {
302     my($self,$str) = @_;
303     $self->{"struct"}->{"command"} = 'SELECT';
304     my($from_clause,$where_clause,$order_clause,$limit_clause);
305     $str =~ s/^SELECT (.+)$/$1/i;
306     if ( $str =~ s/^(.+) LIMIT (.+)$/$1/i ) { $limit_clause = $2; }
307     if ( $str =~ s/^(.+) ORDER BY (.+)$/$1/i ) { $order_clause = $2; }
308     if ( $str =~ s/^(.+?) WHERE (.+)$/$1/i ) { $where_clause = $2; }
309     if ( $str =~ s/^(.+?) FROM (.+)$/$1/i ) { $from_clause = $2; }
310     else {
311     return $self->do_err("Couldn't find FROM clause in SELECT!");
312     }
313     return undef unless $self->FROM_CLAUSE($from_clause);
314     return undef unless $self->SELECT_CLAUSE($str);
315     if ($where_clause) {
316     return undef unless $self->SEARCH_CONDITION($where_clause);
317     }
318     if ($order_clause) {
319     return undef unless $self->SORT_SPEC_LIST($order_clause);
320     }
321     if ($limit_clause) {
322     return undef unless $self->LIMIT_CLAUSE($limit_clause);
323     }
324     if ( ( $self->{"struct"}->{"join"}->{"clause"}
325     and $self->{"struct"}->{"join"}->{"clause"} eq 'ON'
326     )
327     or ( $self->{"struct"}->{"multiple_tables"}
328     ###new
329     and !(scalar keys %{$self->{"struct"}->{"join"}})
330     # and !$self->{"struct"}->{"join"}
331     ###
332     ) ) {
333     return undef unless $self->IMPLICIT_JOIN();
334     }
335     return 1;
336     }
337    
338     sub IMPLICIT_JOIN {
339     my $self = shift;
340     delete $self->{"struct"}->{"multiple_tables"};
341     if ( !$self->{"struct"}->{"join"}->{"clause"}
342     or $self->{"struct"}->{"join"}->{"clause"} ne 'ON'
343     ) {
344     $self->{"struct"}->{"join"}->{"type"} = 'INNER';
345     $self->{"struct"}->{"join"}->{"clause"} = 'IMPLICIT';
346     }
347     if (defined $self->{"struct"}->{"keycols"} ) {
348     my @keys;
349     my @keys2 = @keys = @{ $self->{"struct"}->{"keycols"} };
350     $self->{"struct"}->{"join"}->{"table_order"} = $self->order_joins(\@keys2);
351     @{$self->{"struct"}->{"join"}->{"keycols"}} = @keys;
352     delete $self->{"struct"}->{"keycols"};
353     }
354     else {
355     return $self->do_err("No equijoin condition in WHERE or ON clause");
356     }
357     return 1;
358     }
359    
360     sub EXPLICIT_JOIN {
361     my $self = shift;
362     my $remainder = shift;
363     return undef unless $remainder;
364     my($tableA,$tableB,$keycols,$jtype,$natural);
365     if ($remainder =~ /^(.+?) (NATURAL|INNER|LEFT|RIGHT|FULL|UNION|JOIN)(.+)$/s){
366     $tableA = $1;
367     $remainder = $2.$3;
368     }
369     else {
370     ($tableA,$remainder) = $remainder =~ /^(\S+) (.*)/;
371     }
372     if ( $remainder =~ /^NATURAL (.+)/) {
373     $self->{"struct"}->{"join"}->{"clause"} = 'NATURAL';
374     $natural++;
375     $remainder = $1;
376     }
377     if ( $remainder =~
378     /^(INNER|LEFT|RIGHT|FULL|UNION) JOIN (.+)/
379     ) {
380     $jtype = $self->{"struct"}->{"join"}->{"clause"} = $1;
381     $remainder = $2;
382     $jtype = "$jtype OUTER" if $jtype !~ /INNER|UNION/;
383     }
384     if ( $remainder =~
385     /^(LEFT|RIGHT|FULL) OUTER JOIN (.+)/
386     ) {
387     $jtype = $self->{"struct"}->{"join"}->{"clause"} = $1 . " OUTER";
388     $remainder = $2;
389     }
390     if ( $remainder =~ /^JOIN (.+)/) {
391     $jtype = 'INNER';
392     $self->{"struct"}->{"join"}->{"clause"} = 'DEFAULT INNER';
393     $remainder = $1;
394     }
395     if ( $self->{"struct"}->{"join"} ) {
396     if ( $remainder && $remainder =~ /^(.+?) USING \(([^\)]+)\)(.*)/) {
397     $self->{"struct"}->{"join"}->{"clause"} = 'USING';
398     $tableB = $1;
399     my $keycolstr = $2;
400     $remainder = $3;
401     @$keycols = split /,/,$keycolstr;
402     }
403     if ( $remainder && $remainder =~ /^(.+?) ON (.+)/) {
404     $self->{"struct"}->{"join"}->{"clause"} = 'ON';
405     $tableB = $1;
406     my $keycolstr = $2;
407     $remainder = $3;
408     if ($keycolstr =~ / OR /i ) {
409     return $self->do_err(qq~Can't use OR in an ON clause!~,1);
410     }
411     @$keycols = split / AND /i,$keycolstr;
412     $self->{"tmp"}->{"is_table_name"}->{"$tableA"} = 1;
413     $self->{"tmp"}->{"is_table_name"}->{"$tableB"} = 1;
414     for (@$keycols) {
415     my($arg1,$arg2) = split / = /;
416     return undef unless $arg1 = $self->ROW_VALUE($arg1);
417     return undef unless $arg2 = $self->ROW_VALUE($arg2);
418     if ( $arg1->{"type"}eq 'column' and $arg2->{"type"}eq 'column'){
419     push @{ $self->{"struct"}->{"keycols"} }, $arg1->{"value"};
420     push @{ $self->{"struct"}->{"keycols"} }, $arg2->{"value"};
421     delete $self->{"struct"}->{"where_clause"};
422     }
423     }
424     }
425     elsif ($remainder =~ /^(.+?)$/i) {
426     $tableB = $1;
427     $remainder = $2;
428     }
429     $remainder =~ s/^\s+// if $remainder;
430     }
431     if ($jtype) {
432     $jtype = "NATURAL $jtype" if $natural;
433     if ($natural and $keycols) {
434     return $self->do_err(
435     qq~Can't use NATURAL with a USING or ON clause!~
436     );
437     }
438     return undef unless $self->TABLE_NAME_LIST("$tableA,$tableB");
439     $self->{"struct"}->{"join"}->{"type"} = $jtype;
440     $self->{"struct"}->{"join"}->{"keycols"} = $keycols if $keycols;
441     return 1;
442     }
443     return $self->do_err("Couldn't parse explicit JOIN!");
444     }
445    
446     sub SELECT_CLAUSE {
447     my($self,$str) = @_;
448     return undef unless $str;
449     if ($str =~ s/^(DISTINCT|ALL) (.+)$/$2/i) {
450     $self->{"struct"}->{"set_quantifier"} = uc $1;
451     }
452     if ($str =~ /[()]/) {
453     return undef unless $self->SET_FUNCTION_SPEC($str);
454     }
455     else {
456     return undef unless $self->SELECT_LIST($str);
457     }
458     }
459    
460     sub FROM_CLAUSE {
461     my($self,$str) = @_;
462     return undef unless $str;
463     if ($str =~ / JOIN /i ) {
464     return undef unless $self->EXPLICIT_JOIN($str);
465     }
466     else {
467     return undef unless $self->TABLE_NAME_LIST($str);
468     }
469     }
470    
471     sub INSERT {
472     my($self,$str) = @_;
473     my $col_str;
474     my($table_name,$val_str) = $str =~
475     /^INSERT\s+INTO\s+(.+?)\s+VALUES\s+\((.+?)\)$/i;
476 joko 1.3 if ($table_name and $table_name =~ /[()]/ ) {
477 joko 1.1 ($table_name,$col_str,$val_str) = $str =~
478     /^INSERT\s+INTO\s+(.+?)\s+\((.+?)\)\s+VALUES\s+\((.+?)\)$/i;
479     }
480     return $self->do_err('No table name specified!') unless $table_name;
481 joko 1.3 return $self->do_err('Missing values list!') unless defined $val_str;
482 joko 1.1 return undef unless $self->TABLE_NAME($table_name);
483     $self->{"struct"}->{"command"} = 'INSERT';
484     $self->{"struct"}->{"table_names"} = [$table_name];
485     if ($col_str) {
486     return undef unless $self->COLUMN_NAME_LIST($col_str);
487     }
488     else {
489     $self->{"struct"}->{"column_names"} = ['*'];
490     }
491     return undef unless $self->LITERAL_LIST($val_str);
492     return 1;
493     }
494    
495     ###################################################################
496     # UPDATE ::=
497     #
498     # UPDATE <table> SET <set_clause_list> [ WHERE <search_condition>]
499     #
500     ###################################################################
501     sub UPDATE {
502     my($self,$str) = @_;
503     $self->{"struct"}->{"command"} = 'UPDATE';
504     my($table_name,$remainder) = $str =~
505     /^UPDATE (.+?) SET (.+)$/i;
506     return $self->do_err(
507     'Incomplete UPDATE clause'
508     ) if !$table_name or !$remainder;
509     return undef unless $self->TABLE_NAME($table_name);
510     $self->{"tmp"}->{"is_table_name"} = {$table_name => 1};
511     $self->{"struct"}->{"table_names"} = [$table_name];
512     my($set_clause,$where_clause) = $remainder =~
513     /(.*?) WHERE (.*)$/i;
514     $set_clause = $remainder if !$set_clause;
515     return undef unless $self->SET_CLAUSE_LIST($set_clause);
516     if ($where_clause) {
517     return undef unless $self->SEARCH_CONDITION($where_clause);
518     }
519     my @vals = @{$self->{"struct"}->{"values"}};
520     my $num_val_placeholders=0;
521     for my $v(@vals) {
522     $num_val_placeholders++ if $v->{"type"} eq 'placeholder';
523     }
524     $self->{"struct"}->{"num_val_placeholders"}=$num_val_placeholders;
525     return 1;
526     }
527    
528     #########
529     # CREATE
530     #########
531    
532     sub CREATE {
533     my $self = shift;
534     my $stmt = shift;
535     $self->{"struct"}->{"command"} = 'CREATE';
536 joko 1.3 my($table_name,$table_element_def,%is_col_name);
537 joko 1.1 if ($stmt =~ /^CREATE (LOCAL|GLOBAL) TEMPORARY TABLE(.*)$/si ) {
538     $self->{"struct"}->{"table_type"} = "$1 TEMPORARY";
539     $stmt = "CREATE TABLE$2";
540     }
541     if ($stmt =~ /^(.*) ON COMMIT (DELETE|PRESERVE) ROWS\s*$/si ) {
542     $stmt = $1;
543     $self->{"struct"}->{"commit_behaviour"} = $2;
544     return $self->do_err(
545     "Can't specify commit behaviour for permanent tables."
546     )
547     if !defined $self->{"struct"}->{"table_type"}
548     or $self->{"struct"}->{"table_type"} !~ /TEMPORARY/;
549     }
550     if ($stmt =~ /^CREATE TABLE (\S+) \((.*)\)$/si ) {
551     $table_name = $1;
552     $table_element_def = $2;
553     }
554     else {
555     return $self->do_err( "Can't find column definitions!" );
556     }
557 joko 1.3 return undef unless $self->TABLE_NAME($table_name);
558 joko 1.1 $table_element_def =~ s/\s+\(/(/g;
559     my $primary_defined;
560     for my $col(split ',',$table_element_def) {
561     my($name,$type,$constraints)=($col =~/\s*(\S+)\s+(\S+)\s*(.*)/);
562     if (!$type) {
563     return $self->do_err( "Column definition is missing a data type!" );
564     }
565     return undef if !($self->IDENTIFIER($name));
566     # if ($name =~ /^\?QI(.+)\?$/ ) {
567     $name = $self->replace_quoted_ids($name);
568     # }
569     $constraints =~ s/^\s+//;
570     $constraints =~ s/\s+$//;
571     if ($constraints) {
572     $constraints =~ s/PRIMARY KEY/PRIMARY_KEY/i;
573     $constraints =~ s/NOT NULL/NOT_NULL/i;
574     my @c = split /\s+/, $constraints;
575     my %has_c;
576     for my $constr(@c) {
577     if ( $constr =~ /^\s*(UNIQUE|NOT_NULL|PRIMARY_KEY)\s*$/i ) {
578     my $cur_c = uc $1;
579 joko 1.3 if ($has_c{$cur_c}++) {
580 joko 1.1 return $self->do_err(
581 joko 1.3 qq~Duplicate column constraint: '$constr'!~
582 joko 1.1 );
583     }
584     if ($cur_c eq 'PRIMARY_KEY' and $primary_defined++ ) {
585     return $self->do_err(
586 joko 1.3 qq~Can't have two PRIMARY KEYs in a table!~
587 joko 1.1 );
588     }
589     $constr =~ s/_/ /g;
590     push @{$self->{"struct"}->{"column_defs"}->{"$name"}->{"constraints"} }, $constr;
591    
592     }
593     else {
594 joko 1.3 return $self->do_err("Unknown column constraint: '$constr'!");
595 joko 1.1 }
596     }
597     }
598     $type = uc $type;
599     my $length;
600     if ( $type =~ /(.+)\((.+)\)/ ) {
601     $type = $1;
602     $length = $2;
603     }
604     if (!$self->{"opts"}->{"valid_data_types"}->{"$type"}) {
605     return $self->do_err("'$type' is not a recognized data type!");
606     }
607     $self->{"struct"}->{"column_defs"}->{"$name"}->{"data_type"} = $type;
608     $self->{"struct"}->{"column_defs"}->{"$name"}->{"data_length"} = $length;
609     push @{$self->{"struct"}->{"column_names"}},$name;
610 joko 1.3 my $tmpname = $name;
611     $tmpname = uc $tmpname unless $tmpname =~ /^"/;
612     return $self->do_err("Duplicate column names!")
613     if $is_col_name{$tmpname}++;
614 joko 1.1
615     }
616     $self->{"struct"}->{"table_names"} = [$table_name];
617     return 1;
618     }
619    
620    
621     ###############
622     # SQL SUBRULES
623     ###############
624    
625     sub SET_CLAUSE_LIST {
626     my $self = shift;
627     my $set_string = shift;
628     my @sets = split /,/,$set_string;
629     my(@cols,@vals);
630     for(@sets) {
631     my($col,$val) = split / = /,$_;
632 joko 1.3 return $self->do_err('Incomplete SET clause!') if !defined $col or !defined $val;
633 joko 1.1 push @cols, $col;
634     push @vals, $val;
635     }
636     return undef unless $self->COLUMN_NAME_LIST(join ',',@cols);
637     return undef unless $self->LITERAL_LIST(join ',',@vals);
638     return 1;
639     }
640    
641     sub SET_QUANTIFIER {
642     my($self,$str) = @_;
643     if ($str =~ /^(DISTINCT|ALL)\s+(.*)$/si) {
644     $self->{"struct"}->{"set_quantifier"} = uc $1;
645     $str = $2;
646     }
647     return $str;
648     }
649    
650     sub SELECT_LIST {
651     my $self = shift;
652     my $col_str = shift;
653     if ( $col_str =~ /^\s*\*\s*$/ ) {
654     $self->{"struct"}->{"column_names"} = ['*'];
655     return 1;
656     }
657     my @col_list = split ',',$col_str;
658     if (!(scalar @col_list)) {
659     return $self->do_err('Missing column name list!');
660     }
661     my(@newcols,$newcol);
662     for my $col(@col_list) {
663     # $col = trim($col);
664     $col =~ s/^\s+//;
665     $col =~ s/\s+$//;
666     if ($col =~ /^(\S+)\.\*$/) {
667 joko 1.3 my $table = $1;
668     my %is_table_alias = %{$self->{"tmp"}->{"is_table_alias"}};
669     $table = $is_table_alias{$table} if $is_table_alias{$table};
670     $table = $is_table_alias{"\L$table"} if $is_table_alias{"\L$table"};
671     # $table = uc $table unless $table =~ /^"/;
672     #use mylibs; zwarn \%is_table_alias;
673     #print "\n<<$table>>\n";
674 joko 1.1 return undef unless $self->TABLE_NAME($table);
675     $table = $self->replace_quoted_ids($table);
676     push @newcols, "$table.*";
677     }
678     else {
679     return undef unless $newcol = $self->COLUMN_NAME($col);
680     push @newcols, $newcol;
681     }
682     }
683     $self->{"struct"}->{"column_names"} = \@newcols;
684     return 1;
685     }
686    
687     sub SET_FUNCTION_SPEC {
688     my($self,$col_str) = @_;
689     my @funcs = split /,/, $col_str;
690     my %iscol;
691     for my $func(@funcs) {
692     if ($func =~ /^(COUNT|AVG|SUM|MAX|MIN) \((.*)\)\s*$/i ) {
693     my $set_function_name = uc $1;
694     my $set_function_arg = $2;
695     my $distinct;
696     if ( $set_function_arg =~ s/(DISTINCT|ALL) (.+)$/$2/i ) {
697     $distinct = uc $1;
698 joko 1.3 $self->{"struct"}->{"set_quantifier"} = $distinct;
699 joko 1.1 }
700     my $count_star = 1 if $set_function_name eq 'COUNT'
701     and $set_function_arg eq '*';
702     my $ok = $self->COLUMN_NAME($set_function_arg)
703     if !$count_star;
704     return undef if !$count_star and !$ok;
705     if ($set_function_arg !~ /^"/) {
706     $set_function_arg = uc $set_function_arg;
707     }
708     push @{ $self->{"struct"}->{'set_function'}}, {
709     name => $set_function_name,
710     arg => $set_function_arg,
711     distinct => $distinct,
712     };
713     push( @{ $self->{"struct"}->{"column_names"} }, $set_function_arg)
714     if !$iscol{$set_function_arg}++
715     and ($set_function_arg ne '*');
716     }
717     else {
718     return $self->do_err("Bad set function before FROM clause.");
719     }
720     }
721     my $cname = $self->{"struct"}->{"column_names"};
722     if ( !$cname or not scalar @$cname ) {
723     $self->{"struct"}->{"column_names"} = ['*'];
724     }
725     return 1;
726     }
727    
728     sub LIMIT_CLAUSE {
729     my($self,$limit_clause) = @_;
730     # $limit_clause = trim($limit_clause);
731     $limit_clause =~ s/^\s+//;
732     $limit_clause =~ s/\s+$//;
733    
734     return 1 if !$limit_clause;
735     my($offset,$limit,$junk) = split /,/, $limit_clause;
736     return $self->do_err('Bad limit clause!')
737     if (defined $limit and $limit =~ /[^\d]/)
738     or ( defined $offset and $offset =~ /[^\d]/ )
739     or defined $junk;
740     if (defined $offset and !defined $limit) {
741     $limit = $offset;
742     undef $offset;
743     }
744     $self->{"struct"}->{"limit_clause"} = {
745     limit => $limit,
746     offset => $offset,
747     };
748     return 1;
749     }
750    
751     sub is_number {
752     my $x=shift;
753     return 0 if !defined $x;
754     return 1 if $x =~ /^([+-]?)(?=\d|\.\d)\d*(\.\d*)?([Ee]([+-]?\d+))?$/;
755     return 0;
756     }
757    
758     sub SORT_SPEC_LIST {
759     my($self,$order_clause) = @_;
760     return 1 if !$order_clause;
761     my %is_table_name = %{$self->{"tmp"}->{"is_table_name"}};
762     my %is_table_alias = %{$self->{"tmp"}->{"is_table_alias"}};
763     my @ocols;
764     my @order_columns = split ',',$order_clause;
765     for my $col(@order_columns) {
766     my $newcol;
767     my $newarg;
768     if ($col =~ /\s*(\S+)\s+(ASC|DESC)/si ) {
769     $newcol = $1;
770     $newarg = uc $2;
771     }
772     elsif ($col =~ /^\s*(\S+)\s*$/si ) {
773     $newcol = $1;
774     }
775     else {
776     return $self->do_err(
777     'Junk after column name in ORDER BY clause!'
778     );
779     }
780     return undef if !($newcol = $self->COLUMN_NAME($newcol));
781     if ($newcol =~ /^(.+)\..+$/s ) {
782     my $table = $1;
783     if ($table =~ /^'/) {
784     if (!$is_table_name{"$table"} and !$is_table_alias{"$table"} ) {
785     return $self->do_err( "Table '$table' in ORDER BY clause "
786     . "not in FROM clause."
787     );
788     }}
789     elsif (!$is_table_name{"\L$table"} and !$is_table_alias{"\L$table"} ) {
790     return $self->do_err( "Table '$table' in ORDER BY clause "
791     . "not in FROM clause."
792     );
793     }
794     }
795     push @ocols, {$newcol => $newarg};
796     }
797     $self->{"struct"}->{"sort_spec_list"} = \@ocols;
798     return 1;
799     }
800    
801     sub SEARCH_CONDITION {
802     my $self = shift;
803     my $str = shift;
804     $str =~ s/^\s*WHERE (.+)/$1/;
805     $str =~ s/^\s+//;
806     $str =~ s/\s+$//;
807     return $self->do_err("Couldn't find WHERE clause!") unless $str;
808     $str = get_btwn( $str );
809     $str = get_in( $str );
810     my $open_parens = $str =~ tr/\(//;
811     my $close_parens = $str =~ tr/\)//;
812     if ($open_parens != $close_parens) {
813     return $self->do_err("Mismatched parentheses in WHERE clause!");
814     }
815     $str = nongroup_numeric( nongroup_string( $str ) );
816     my $pred = $open_parens
817     ? $self->parens_search($str,[])
818     : $self->non_parens_search($str,[]);
819     return $self->do_err("Couldn't find predicate!") unless $pred;
820     $self->{"struct"}->{"where_clause"} = $pred;
821     return 1;
822     }
823    
824     ############################################################
825     # UTILITY FUNCTIONS CALLED TO PARSE PARENS IN WHERE CLAUSE
826     ############################################################
827    
828     # get BETWEEN clause
829     #
830     sub get_btwn {
831     my $str = shift;
832     if ($str =~ /^(.+?) BETWEEN (.+)$/i ) {
833     my($col,$in,$out,$contents);
834     my $front = $1;
835     my $back = $2;
836     my $not = 1 if $front =~ s/^(.+) NOT$/$1/i;
837     if ($front =~ s/^(.+? )(AND|OR|\() (.+)$/$1$2/i) {
838     $col = $3;
839     }
840     else {
841     $col = $front;
842     $front = '';
843     }
844     $front .= " NOT" if $not;
845     my($val1,$val2);
846     if ($back =~ s/^(.+?) AND (.+)$/$2/) {
847     $val1 = $1;
848     }
849     if ($back =~ s/^(.+?) (AND|OR)(.+)$/$2$3/i) {
850     $val2 = $1;
851     }
852     else {
853     $val2 = $back;
854     $back = '';
855     }
856     $str = "$front ($col > $val1 AND $col < $val2) $back";
857     return get_btwn($str);
858     }
859     return $str;
860     }
861    
862     # get IN clause
863     #
864     # a IN (b,c) -> (a=b OR a=c)
865     # a NOT IN (b,c) -> (a<>b AND a<>c)
866     #
867     sub get_in {
868     my $str = shift;
869     my $in_inside_parens;
870     if ($str =~ /^(.+?) IN (\(.+)$/i ) {
871     my($col,$in,$out,$contents);
872     my $front = $1;
873     my $back = $2;
874     my $not;
875     $not++ if $front =~ s/^(.+) NOT$/$1/i;
876     if ($front =~ s/^(.+? )(AND|OR|\() (.+)$/$1$2/i) {
877     $col = $3;
878     }
879     else {
880     $col = $front;
881     $not++ if $col =~ s/^NOT (.+)/$1/i;
882     $front = '';
883     }
884     if ( $col =~ s/^\(// ) {
885     $in_inside_parens++;
886     }
887     #print "~$not~\n";
888     # $front .= " NOT" if $not;
889     # $not++ if $front =~ s/^(.+) NOT$/$1/i;
890     my @chars = split '', $back;
891     for (0..$#chars) {
892     my $char = shift @chars;
893     $contents .= $char;
894     $in++ if $char eq '(';
895     if ( $char eq ')' ) {
896     $out++;
897     last if $in == $out;
898     }
899     }
900     $back = join '', @chars;
901     $back =~ s/\)$// if $in_inside_parens;
902     # print "\n[$front][$col][$contents][$back]\n";
903     #die "\n[$contents]\n";
904     $contents =~ s/^\(//;
905     $contents =~ s/\)$//;
906     my @vals = split /,/, $contents;
907     my $op = '=';
908     my $combiner = 'OR';
909     if ($not) {
910     $op = '<>';
911     $combiner = 'AND';
912     }
913     @vals = map { "$col $op $_" } @vals;
914     my $valStr = join " $combiner ", @vals;
915     $str = "$front ($valStr) $back";
916     $str =~ s/\s+/ /g;
917     return get_in($str);
918     }
919     $str =~ s/^\s+//;
920     $str =~ s/\s+$//;
921     $str =~ s/\(\s+/(/;
922     $str =~ s/\s+\)/)/;
923     #print "$str:\n";
924     return $str;
925     }
926    
927     # groups clauses by nested parens
928     #
929     sub parens_search {
930     my $self = shift;
931     my $str = shift;
932     my $predicates = shift;
933     my $index = scalar @$predicates;
934    
935     # to handle WHERE (a=b) AND (c=d)
936     if ($str =~ /\(([^()]+?)\)/ ) {
937     my $pred = quotemeta $1;
938     if ($pred !~ / (AND|OR) / ) {
939     $str =~ s/\(($pred)\)/$1/;
940     }
941     }
942     #
943    
944     if ($str =~ s/\(([^()]+)\)/^$index^/ ) {
945     push @$predicates, $1;
946     }
947     if ($str =~ /\(/ ) {
948     return $self->parens_search($str,$predicates);
949     }
950     else {
951     return $self->non_parens_search($str,$predicates);
952     }
953     }
954    
955     # creates predicates from clauses that either have no parens
956     # or ANDs or have been previously grouped by parens and ANDs
957     #
958     sub non_parens_search {
959     my $self = shift;
960     my $str = shift;
961     my $predicates = shift;
962     my $neg = 0;
963     my $nots = {};
964     if ( $str =~ s/^NOT (\^.+)$/$1/i ) {
965     $neg = 1;
966     $nots = {pred=>1};
967     }
968     my( $pred1, $pred2, $op );
969     my $and_preds =[];
970     ($str,$and_preds) = group_ands($str);
971     $str =~ s/^\s*\^0\^\s*$/$predicates->[0]/;
972     return if $str =~ /^\s*~0~\s*$/;
973     if ( ($pred1, $op, $pred2) = $str =~ /^(.+) (AND|OR) (.+)$/i ) {
974     $pred1 =~ s/\~(\d+)\~$/$and_preds->[$1]/;
975     $pred2 =~ s/\~(\d+)\~$/$and_preds->[$1]/;
976     $pred1 = $self->non_parens_search($pred1,$predicates);
977     $pred2 = $self->non_parens_search($pred2,$predicates);
978     return {
979     neg => $neg,
980     nots => $nots,
981     arg1 => $pred1,
982     op => uc $op,
983     arg2 => $pred2,
984     };
985     }
986     else {
987     my $xstr = $str;
988     $xstr =~ s/\?(\d+)\?/$self->{"struct"}->{"literals"}->[$1]/g;
989     my($k,$v) = $xstr =~ /^(\S+?)\s+\S+\s*(.+)\s*$/;
990     #print "$k,$v\n" if defined $k;
991     push @{ $self->{struct}->{where_cols}->{$k}}, $v if defined $k;
992     return $self->PREDICATE($str);
993     }
994     }
995    
996     # groups AND clauses that aren't already grouped by parens
997     #
998     sub group_ands{
999     my $str = shift;
1000     my $and_preds = shift || [];
1001     return($str,$and_preds) unless $str =~ / AND / and $str =~ / OR /;
1002     if ($str =~ /^(.*?) AND (.*)$/i ) {
1003     my $index = scalar @$and_preds;
1004     my($front, $back)=($1,$2);
1005     if ($front =~ /^.* OR (.*)$/i ) {
1006     $front = $1;
1007     }
1008     if ($back =~ /^(.*?) (OR|AND) (.*)$/i ) {
1009     $back = $1;
1010     }
1011     my $newpred = "$front AND $back";
1012     push @$and_preds, $newpred;
1013     $str =~ s/\Q$newpred/~$index~/i;
1014     return group_ands($str,$and_preds);
1015     }
1016     else {
1017     return $str,$and_preds;
1018     }
1019     }
1020    
1021     # replaces string function parens with square brackets
1022     # e.g TRIM (foo) -> TRIM[foo]
1023     #
1024 joko 1.3
1025 joko 1.1 sub nongroup_string {
1026 joko 1.3 my $f= FUNCTION_NAMES;
1027 joko 1.1 my $str = shift;
1028 joko 1.3 # $str =~ s/(TRIM|SUBSTRING|UPPER|LOWER) \(([^()]+)\)/$1\[$2\]/gi;
1029     $str =~ s/($f) \(([^()]+)\)/$1\[$2\]/gi;
1030     # if ( $str =~ /(TRIM|SUBSTRING|UPPER|LOWER) \(/i ) {
1031     if ( $str =~ /($f) \(/i ) {
1032 joko 1.1 return nongroup_string($str);
1033     }
1034     else {
1035     return $str;
1036     }
1037     }
1038    
1039     # replaces math parens with square brackets
1040     # e.g (4-(6+7)*9) -> MATH[4-MATH[6+7]*9]
1041     #
1042     sub nongroup_numeric {
1043     my $str = shift;
1044     my $has_op;
1045     if ( $str =~ /\(([0-9 \*\/\+\-_a-zA-Z\[\]\?]+)\)/ ) {
1046     my $match = $1;
1047     if ($match !~ /(LIKE |IS|BETWEEN|IN)/ ) {
1048     my $re = quotemeta($match);
1049     $str =~ s/\($re\)/MATH\[$match\]/;
1050     }
1051     else {
1052     $has_op++;
1053     }
1054     }
1055     if ( !$has_op and $str =~ /\(([0-9 \*\/\+\-_a-zA-Z\[\]\?]+)\)/ ) {
1056     return nongroup_numeric($str);
1057     }
1058     else {
1059     return $str;
1060     }
1061     }
1062     ############################################################
1063    
1064    
1065     #########################################################
1066     # LITERAL_LIST ::= <literal> [,<literal>]
1067     #########################################################
1068     sub LITERAL_LIST {
1069     my $self = shift;
1070     my $str = shift;
1071     my @tokens = split /,/, $str;
1072     my @values;
1073     for my $tok(@tokens) {
1074     my $val = $self->ROW_VALUE($tok);
1075     return $self->do_err(
1076     qq('$tok' is not a valid value or is not quoted!)
1077     ) unless $val;
1078     push @values, $val;
1079     }
1080     $self->{"struct"}->{"values"} = \@values;
1081     return 1;
1082     }
1083    
1084    
1085     ###################################################################
1086     # LITERAL ::= <quoted_string> | <question mark> | <number> | NULL
1087     ###################################################################
1088     sub LITERAL {
1089     my $self = shift;
1090     my $str = shift;
1091     return 'null' if $str =~ /^NULL$/i; # NULL
1092     # return 'empty_string' if $str =~ /^~E~$/i; # NULL
1093     if ($str eq '?') {
1094     $self->{struct}->{num_placeholders}++;
1095     return 'placeholder';
1096     }
1097     # return 'placeholder' if $str eq '?'; # placeholder question mark
1098     return 'string' if $str =~ /^'.*'$/s; # quoted string
1099     return 'number' if $str =~ # number
1100     /^([+-]?)(?=\d|\.\d)\d*(\.\d*)?([Ee]([+-]?\d+))?$/;
1101     return undef;
1102     }
1103     ###################################################################
1104     # PREDICATE
1105     ###################################################################
1106     sub PREDICATE {
1107     my $self = shift;
1108     my $str = shift;
1109     my @allops = keys %{ $self->{"opts"}->{"valid_comparison_operators"} };
1110     my @notops;
1111     for (@allops) { push (@notops, $_) if /NOT/i };
1112     my $ops = join '|', @notops;
1113     my $opexp = "^\\s*(.+)\\s+($ops)\\s+(.*)\\s*\$";
1114     my($arg1,$op,$arg2) = $str =~ /$opexp/i;
1115     if (!defined $op) {
1116     my @compops;
1117     for (@allops) { push (@compops, $_) if /<=|>=|<>/ };
1118     $ops = join '|', @compops;
1119     $opexp = "^\\s*(.+)\\s+($ops)\\s+(.*)\\s*\$";
1120     ($arg1,$op,$arg2) = $str =~ /$opexp/i;
1121     }
1122     if (!defined $op) {
1123     $ops = join '|', @allops;
1124     $opexp = "^\\s*(.+)\\s+($ops)\\s+(.*)\\s*\$";
1125     ($arg1,$op,$arg2) = $str =~ /$opexp/i;
1126     }
1127     $op = uc $op;
1128     if (!defined $arg1 || !defined $op || !defined $arg2) {
1129     return $self->do_err("Bad predicate: '$str'!");
1130     }
1131     my $negated = 0; # boolean value showing if predicate is negated
1132     my %not; # hash showing elements modified by NOT
1133     #
1134     # e.g. "NOT bar = foo" -> %not = (arg1=>1)
1135     # "bar NOT LIKE foo" -> %not = (op=>1)
1136     # "NOT bar NOT LIKE foo" -> %not = (arg1=>1,op=>1);
1137     # "NOT bar IS NOT NULL" -> %not = (arg1=>1,op=>1);
1138     # "bar = foo" -> %not = undef;
1139     #
1140     if ( $arg1 =~ s/^NOT (.+)$/$1/i ) {
1141     $not{arg1}++;
1142     }
1143     if ( $op =~ s/^(.+) NOT$/$1/i
1144     || $op =~ s/^NOT (.+)$/$1/i ) {
1145     $not{op}++;
1146     }
1147     $negated = 1 if %not and scalar keys %not == 1;
1148     return undef unless $arg1 = $self->ROW_VALUE($arg1);
1149     return undef unless $arg2 = $self->ROW_VALUE($arg2);
1150     if ( $arg1->{"type"}eq 'column'
1151     and $arg2->{"type"}eq 'column'
1152     and $op eq '='
1153     ) {
1154     push @{ $self->{"struct"}->{"keycols"} }, $arg1->{"value"};
1155     push @{ $self->{"struct"}->{"keycols"} }, $arg2->{"value"};
1156     }
1157     return {
1158     neg => $negated,
1159     nots => \%not,
1160     arg1 => $arg1,
1161     op => $op,
1162     arg2 => $arg2,
1163     };
1164     }
1165    
1166     sub undo_string_funcs {
1167     my $str = shift;
1168 joko 1.3 my $f= FUNCTION_NAMES;
1169     # $str =~ s/(TRIM|UPPER|LOWER|SUBSTRING)\[([^\]\[]+?)\]/$1 ($2)/;
1170     # if ($str =~ /(TRIM|UPPER|LOWER|SUBSTRING)\[/) {
1171     $str =~ s/($f)\[([^\]\[]+?)\]/$1 ($2)/;
1172     if ($str =~ /($f)\[/) {
1173 joko 1.1 return undo_string_funcs($str);
1174     }
1175     return $str;
1176     }
1177    
1178     sub undo_math_funcs {
1179     my $str = shift;
1180     $str =~ s/MATH\[([^\]\[]+?)\]/($1)/;
1181     if ($str =~ /MATH\[/) {
1182     return undo_math_funcs($str);
1183     }
1184     return $str;
1185     }
1186    
1187    
1188     ###################################################################
1189     # ROW_VALUE ::= <literal> | <column_name>
1190     ###################################################################
1191     sub ROW_VALUE {
1192     my $self = shift;
1193     my $str = shift;
1194     $str = undo_string_funcs($str);
1195     $str = undo_math_funcs($str);
1196     my $type;
1197    
1198     # MATH
1199     #
1200     if ($str =~ /[\*\+\-\/]/ ) {
1201     my @vals;
1202     my $i=-1;
1203     $str =~ s/([^\s\*\+\-\/\)\(]+)/push @vals,$1;$i++;"?$i?"/ge;
1204     my @newvalues;
1205     for (@vals) {
1206     my $val = $self->ROW_VALUE($_);
1207     if ($val && $val->{"type"} !~ /number|column|placeholder/) {
1208     return $self->do_err(qq[
1209     String '$val' not allowed in Numeric expression!
1210     ]);
1211     }
1212     push @newvalues,$val;
1213     }
1214     return {
1215     type => 'function',
1216     name => 'numeric_exp',
1217     str => $str,
1218     vals => \@newvalues,
1219     }
1220     }
1221    
1222     # SUBSTRING (value FROM start [FOR length])
1223     #
1224     if ($str =~ /^SUBSTRING \((.+?) FROM (.+)\)\s*$/i ) {
1225     my $name = 'SUBSTRING';
1226     my $start = $2;
1227     my $value = $self->ROW_VALUE($1);
1228     my $length;
1229     if ($start =~ /^(.+?) FOR (.+)$/i) {
1230     $start = $1;
1231     $length = $2;
1232     $length = $self->ROW_VALUE($length);
1233     }
1234     $start = $self->ROW_VALUE($start);
1235     $str =~ s/\?(\d+)\?/$self->{"struct"}->{"literals"}->[$1]/g;
1236     return $self->do_err(
1237     "Can't use a string as a SUBSTRING position: '$str'!")
1238     if $start->{"type"} eq 'string'
1239     or ($start->{"length"} and $start->{"length"}->{"type"} eq 'string');
1240     return undef unless $value;
1241     return $self->do_err(
1242     "Can't use a number in SUBSTRING: '$str'!")
1243     if $value->{"type"} eq 'number';
1244     return {
1245     "type" => 'function',
1246     "name" => $name,
1247     "value" => $value,
1248     "start" => $start,
1249     "length" => $length,
1250     };
1251     }
1252    
1253 joko 1.3 # TO_CHAR (value)
1254     #
1255     if ($str =~ /^TO_CHAR \((.+)\)\s*$/i ) {
1256     my $name = 'TO_CHAR';
1257     my $value = $self->ROW_VALUE($1);
1258     return undef unless $value;
1259     return {
1260     type => 'function',
1261     name => $name,
1262     value => $value,
1263     };
1264     }
1265 joko 1.1
1266     # UPPER (value) and LOWER (value)
1267     #
1268     if ($str =~ /^(UPPER|LOWER) \((.+)\)\s*$/i ) {
1269     my $name = uc $1;
1270     my $value = $self->ROW_VALUE($2);
1271     return undef unless $value;
1272     $str =~ s/\?(\d+)\?/$self->{"struct"}->{"literals"}->[$1]/g;
1273     return $self->do_err(
1274     "Can't use a number in UPPER/LOWER: '$str'!")
1275     if $value->{"type"} eq 'number';
1276     return {
1277     type => 'function',
1278     name => $name,
1279     value => $value,
1280     };
1281     }
1282    
1283     # TRIM ( [ [TRAILING|LEADING|BOTH] ['char'] FROM ] value )
1284     #
1285     if ($str =~ /^(TRIM) \((.+)\)\s*$/i ) {
1286     my $name = uc $1;
1287     my $value = $2;
1288     my($trim_spec,$trim_char);
1289     if ($value =~ /^(.+) FROM ([^\(\)]+)$/i ) {
1290     my $front = $1;
1291     $value = $2;
1292     if ($front =~ /^\s*(TRAILING|LEADING|BOTH)(.*)$/i ) {
1293     $trim_spec = uc $1;
1294     # $trim_char = trim($2);
1295     $trim_char = $2;
1296     $trim_char =~ s/^\s+//;
1297     $trim_char =~ s/\s+$//;
1298     undef $trim_char if length($trim_char)==0;
1299     }
1300     else {
1301     # $trim_char = trim($front);
1302     $trim_char = $front;
1303     $trim_char =~ s/^\s+//;
1304     $trim_char =~ s/\s+$//;
1305     }
1306     }
1307     $trim_char =~ s/\?(\d+)\?/$self->{"struct"}->{"literals"}->[$1]/g if $trim_char;
1308     $value = $self->ROW_VALUE($value);
1309     return undef unless $value;
1310     $str =~ s/\?(\d+)\?/$self->{"struct"}->{"literals"}->[$1]/g;
1311     return $self->do_err(
1312     "Can't use a number in TRIM: '$str'!")
1313     if $value->{"type"} eq 'number';
1314     return {
1315     type => 'function',
1316     name => $name,
1317     value => $value,
1318     trim_spec => $trim_spec,
1319     trim_char => $trim_char,
1320     };
1321     }
1322    
1323     # STRING CONCATENATION
1324     #
1325     if ($str =~ /\|\|/ ) {
1326     my @vals = split / \|\| /,$str;
1327     my @newvals;
1328     for my $val(@vals) {
1329     my $newval = $self->ROW_VALUE($val);
1330     return undef unless $newval;
1331     return $self->do_err(
1332     "Can't use a number in string concatenation: '$str'!")
1333     if $newval->{"type"} eq 'number';
1334     push @newvals,$newval;
1335     }
1336     return {
1337     type => 'function',
1338     name => 'str_concat',
1339     value => \@newvals,
1340     };
1341     }
1342    
1343     # NULL, PLACEHOLDER, NUMBER
1344     #
1345     if ( $type = $self->LITERAL($str) ) {
1346     undef $str if $type eq 'null';
1347     # if ($type eq 'empty_string') {
1348     # $str = '';
1349     # $type = 'string';
1350     # }
1351     $str = '' if $str and $str eq q('');
1352     return { type => $type, value => $str };
1353     }
1354    
1355     # QUOTED STRING LITERAL
1356     #
1357     if ($str =~ /\?(\d+)\?/) {
1358     return { type =>'string',
1359     value => $self->{"struct"}->{"literals"}->[$1] };
1360     }
1361     # COLUMN NAME
1362     #
1363     return undef unless $str = $self->COLUMN_NAME($str);
1364     if ( $str =~ /^(.*)\./ && !$self->{"tmp"}->{"is_table_name"}->{"\L$1"}
1365     and !$self->{"tmp"}->{"is_table_alias"}->{"\L$1"} ) {
1366     return $self->do_err(
1367     "Table '$1' in WHERE clause not in FROM clause!"
1368     );
1369     }
1370     # push @{ $self->{"struct"}->{"where_cols"}},$str
1371     # unless $self->{"tmp"}->{"where_cols"}->{"$str"};
1372     $self->{"tmp"}->{"where_cols"}->{"$str"}++;
1373     return { type => 'column', value => $str };
1374     }
1375    
1376     ###############################################
1377     # COLUMN NAME ::= [<table_name>.] <identifier>
1378     ###############################################
1379    
1380     sub COLUMN_NAME {
1381     my $self = shift;
1382     my $str = shift;
1383     my($table_name,$col_name);
1384     if ( $str =~ /^\s*(\S+)\.(\S+)$/s ) {
1385     if (!$self->{"opts"}->{"valid_options"}->{"SELECT_MULTIPLE_TABLES"}) {
1386     return $self->do_err('Dialect does not support multiple tables!');
1387     }
1388     $table_name = $1;
1389     $col_name = $2;
1390     # my $alias = $self->{struct}->{table_alias} || [];
1391     # $table_name = shift @$alias if $alias;
1392     #print "$table_name : $alias";
1393     return undef unless $self->TABLE_NAME($table_name);
1394     $table_name = $self->replace_quoted_ids($table_name);
1395     my $ref;
1396     if ($table_name =~ /^"/) { #"
1397     if (!$self->{"tmp"}->{"is_table_name"}->{"$table_name"}
1398     and !$self->{"tmp"}->{"is_table_alias"}->{"$table_name"}
1399     ) {
1400     $self->do_err(
1401     "Table '$table_name' referenced but not found in FROM list!"
1402     );
1403     return undef;
1404     } }
1405     elsif (!$self->{"tmp"}->{"is_table_name"}->{"\L$table_name"}
1406     and !$self->{"tmp"}->{"is_table_alias"}->{"\L$table_name"}
1407     ) {
1408     $self->do_err(
1409     "Table '$table_name' referenced but not found in FROM list!"
1410     );
1411     return undef;
1412     }
1413     }
1414     else {
1415     $col_name = $str;
1416     }
1417     # $col_name = trim($col_name);
1418     $col_name =~ s/^\s+//;
1419     $col_name =~ s/\s+$//;
1420     return undef unless $col_name eq '*' or $self->IDENTIFIER($col_name);
1421     #
1422 joko 1.3 # MAKE COL NAMES ALL UPPER CASE
1423 joko 1.1 my $orgcol = $col_name;
1424     if ($col_name =~ /^\?QI(\d+)\?$/) {
1425     $col_name = $self->replace_quoted_ids($col_name);
1426     }
1427     else {
1428     # $col_name = lc $col_name;
1429     $col_name = uc $col_name;
1430     }
1431     $self->{struct}->{ORG_NAME}->{$col_name} = $orgcol;
1432     #
1433     #
1434     if ($table_name) {
1435     my $alias = $self->{tmp}->{is_table_alias}->{"\L$table_name"};
1436 joko 1.3 #use mylibs; print "$table_name"; zwarn $self->{tmp};
1437 joko 1.1 $table_name = $alias if defined $alias;
1438     $table_name = uc $table_name;
1439     $col_name = "$table_name.$col_name";
1440 joko 1.3 #print "<<$col_name>>";
1441 joko 1.1 }
1442     return $col_name;
1443     }
1444    
1445     #########################################################
1446     # COLUMN NAME_LIST ::= <column_name> [,<column_name>...]
1447     #########################################################
1448     sub COLUMN_NAME_LIST {
1449     my $self = shift;
1450     my $col_str = shift;
1451     my @col_list = split ',',$col_str;
1452     if (!(scalar @col_list)) {
1453     return $self->do_err('Missing column name list!');
1454     }
1455     my @newcols;
1456     my $newcol;
1457     for my $col(@col_list) {
1458     $col =~ s/^\s+//;
1459     $col =~ s/\s+$//;
1460     # return undef if !($newcol = $self->COLUMN_NAME(trim($col)));
1461     return undef if !($newcol = $self->COLUMN_NAME($col));
1462     push @newcols, $newcol;
1463     }
1464     $self->{"struct"}->{"column_names"} = \@newcols;
1465     return 1;
1466     }
1467    
1468    
1469     #####################################################
1470     # TABLE_NAME_LIST := <table_name> [,<table_name>...]
1471     #####################################################
1472     sub TABLE_NAME_LIST {
1473     my $self = shift;
1474     my $table_name_str = shift;
1475     my %aliases = ();
1476     my @tables;
1477     my @table_names = split ',', $table_name_str;
1478     if ( scalar @table_names > 1
1479     and !$self->{"opts"}->{"valid_options"}->{'SELECT_MULTIPLE_TABLES'}
1480     ) {
1481     return $self->do_err('Dialect does not support multiple tables!');
1482     }
1483     my %is_table_alias;
1484     for my $table_str(@table_names) {
1485     my($table,$alias);
1486     my(@tstr) = split / /,$table_str;
1487     if (@tstr == 1) { $table = $tstr[0]; }
1488     elsif (@tstr == 2) { $table = $tstr[0]; $alias = $tstr[1]; }
1489     # elsif (@tstr == 2) { $table = $tstr[1]; $alias = $tstr[0]; }
1490     elsif (@tstr == 3) {
1491     return $self->do_err("Can't find alias in FROM clause!")
1492     unless uc($tstr[1]) eq 'AS';
1493     $table = $tstr[0]; $alias = $tstr[2];
1494     # $table = $tstr[2]; $alias = $tstr[0];
1495     }
1496     else {
1497     return $self->do_err("Can't find table names in FROM clause!")
1498     }
1499     return undef unless $self->TABLE_NAME($table);
1500     $table = $self->replace_quoted_ids($table);
1501     push @tables, $table;
1502     if ($alias) {
1503     #die $alias, $table;
1504     return undef unless $self->TABLE_NAME($alias);
1505     $alias = $self->replace_quoted_ids($alias);
1506     if ($alias =~ /^"/) {
1507     push @{$aliases{$table}},"$alias";
1508     $is_table_alias{"$alias"}=$table;
1509     }
1510     else {
1511     push @{$aliases{$table}},"\L$alias";
1512     $is_table_alias{"\L$alias"}=$table;
1513     }
1514     # $aliases{$alias} = $table;
1515     }
1516     }
1517    
1518     # my %is_table_name = map { $_ => 1 } @tables,keys %aliases;
1519     my %is_table_name = map { lc $_ => 1 } @tables;
1520     #%is_table_alias = map { lc $_ => 1 } @aliases;
1521     $self->{"tmp"}->{"is_table_alias"} = \%is_table_alias;
1522     $self->{"tmp"}->{"is_table_name"} = \%is_table_name;
1523     $self->{"struct"}->{"table_names"} = \@tables;
1524     $self->{"struct"}->{"table_alias"} = \%aliases;
1525     $self->{"struct"}->{"multiple_tables"} = 1 if @tables > 1;
1526     return 1;
1527     }
1528    
1529     #############################
1530     # TABLE_NAME := <identifier>
1531     #############################
1532     sub TABLE_NAME {
1533     my $self = shift;
1534     my $table_name = shift;
1535     if ($table_name =~ /\s*(\S+)\s+\S+/s) {
1536     return $self->do_err("Junk after table name '$1'!");
1537     }
1538     $table_name =~ s/\s+//s;
1539     if (!$table_name) {
1540     return $self->do_err('No table name specified!');
1541     }
1542 joko 1.3 return $self->IDENTIFIER($table_name);
1543     # return undef if !($self->IDENTIFIER($table_name));
1544     # return 1;
1545 joko 1.1 }
1546    
1547    
1548     ###################################################################
1549     # IDENTIFIER ::= <alphabetic_char> { <alphanumeric_char> | _ }...
1550     #
1551     # and must not be a reserved word or over 128 chars in length
1552     ###################################################################
1553     sub IDENTIFIER {
1554     my $self = shift;
1555     my $id = shift;
1556     if ($id =~ /^\?QI(.+)\?$/ ) {
1557     return 1;
1558     }
1559     return 1 if $id =~ /^".+?"$/s; # QUOTED IDENTIFIER
1560     my $err = "Bad table or column name '$id' "; # BAD CHARS
1561 joko 1.4 if ($id !~ /\w|\./) {
1562 joko 1.1 $err .= "has chars not alphanumeric or underscore!";
1563     return $self->do_err( $err );
1564     }
1565     if ($id =~ /^_/ or $id =~ /^\d/) { # BAD START
1566     $err .= "starts with non-alphabetic character!";
1567     return $self->do_err( $err );
1568     }
1569     if ( length $id > 128 ) { # BAD LENGTH
1570     $err .= "contains more than 128 characters!";
1571     return $self->do_err( $err );
1572     }
1573 joko 1.3 $id = uc $id;
1574     #print "<$id>";
1575     #use mylibs; zwarn $self->{opts}->{reserved_words};
1576     #exit;
1577     if ( $self->{"opts"}->{"reserved_words"}->{$id} ) { # BAD RESERVED WORDS
1578 joko 1.1 $err .= "is a SQL reserved word!";
1579     return $self->do_err( $err );
1580     }
1581     return 1;
1582     }
1583    
1584     ########################################
1585     # PRIVATE METHODS AND UTILITY FUNCTIONS
1586     ########################################
1587     sub order_joins {
1588     my $self = shift;
1589     my $links = shift;
1590     for my $link(@$links) {
1591     if ($link !~ /\./) {
1592     return [];
1593     }
1594     }
1595     @$links = map { s/^(.+)\..*$/$1/; $1; } @$links;
1596     my @all_tables;
1597     my %relations;
1598     my %is_table;
1599     while (@$links) {
1600     my $t1 = shift @$links;
1601     my $t2 = shift @$links;
1602     return undef unless defined $t1 and defined $t2;
1603     push @all_tables, $t1 unless $is_table{$t1}++;
1604     push @all_tables, $t2 unless $is_table{$t2}++;
1605     $relations{$t1}{$t2}++;
1606     $relations{$t2}{$t1}++;
1607     }
1608     my @tables = @all_tables;
1609     my @order = shift @tables;
1610     my %is_ordered = ( $order[0] => 1 );
1611     my %visited;
1612     while(@tables) {
1613     my $t = shift @tables;
1614     my @rels = keys %{$relations{$t}};
1615     for my $t2(@rels) {
1616     next unless $is_ordered{$t2};
1617     push @order, $t;
1618     $is_ordered{$t}++;
1619     last;
1620     }
1621     if (!$is_ordered{$t}) {
1622     push @tables, $t if $visited{$t}++ < @all_tables;
1623     }
1624     }
1625     return $self->do_err(
1626     "Unconnected tables in equijoin statement!"
1627     ) if @order < @all_tables;
1628     return \@order;
1629     }
1630    
1631     sub bless_me {
1632     my $class = shift;
1633     my $self = shift || {};
1634     return bless $self, $class;
1635     }
1636    
1637     # PROVIDE BACKWARD COMPATIBILIT FOR JOCHEN'S FEATURE ATTRIBUTES TO NEW
1638     #
1639     #
1640     sub set_feature_flags {
1641     my($self,$select,$create) = @_;
1642     if (defined $select) {
1643     delete $self->{"select"};
1644     $self->{"opts"}->{"valid_options"}->{"SELECT_MULTIPLE_TABLES"} =
1645     $self->{"opts"}->{"select"}->{"join"} = $select->{"join"};
1646     }
1647     if (defined $create) {
1648     delete $self->{"create"};
1649     for my $key(keys %$create) {
1650     my $type = $key;
1651     $type =~ s/type_(.*)/\U$1/;
1652     $self->{"opts"}->{"valid_data_types"}->{"$type"} =
1653     $self->{"opts"}->{"create"}->{"$key"} = $create->{"$key"};
1654     }
1655     }
1656     }
1657    
1658     sub clean_sql {
1659     my $self = shift;
1660     my $sql = shift;
1661     my $fields;
1662     my $i=-1;
1663     my $e = '\\';
1664     $e = quotemeta($e);
1665     ###new
1666     # CAN'T HANDLE BLOBS!!!
1667     # $sql = quotemeta($sql);
1668     # print "[$sql]\n";
1669     # if ($sql =~ s/^(.*,\s*)''(\s*[,\)].*)$/${1}NULL$2/g ) {
1670     # }
1671     # $sql =~ s/^([^\\']+?)''(.*)$/${1} NULL $2/g;
1672    
1673     # $sql =~ s/([^\\]+?)''/$1 ~E~ /g;
1674    
1675     # print "$sql\n";
1676     ###newend
1677    
1678     # $sql =~ s~'(([^'$e]|$e.)+)'~push(@$fields,$1);$i++;"?$i?"~ge;
1679     $sql =~ s~'(([^'$e]|$e.|'')+)'~push(@$fields,$1);$i++;"?$i?"~ge;
1680    
1681     # $sql =~ s/([^\\]+?)''/$1 ~E~ /g;
1682     #print "<$sql>";
1683     @$fields = map { s/''/\\'/g; $_ } @$fields;
1684    
1685     ###new
1686     # if ( $sql =~ /'/) {
1687     if ( $sql =~ tr/[^\\]'// % 2 == 1 ) {
1688     ###endnew
1689     $sql =~ s/^.*\?(.+)$/$1/;
1690     die "Mismatched single quote before: '$sql\n";
1691     }
1692     if ($sql =~ /\?\?(\d)\?/) {
1693     $sql = $fields->[$1];
1694     die "Mismatched single quote: '$sql\n";
1695     }
1696     @$fields = map { s/$e'/'/g; s/^'(.*)'$/$1/; $_} @$fields;
1697     $self->{"struct"}->{"literals"} = $fields;
1698    
1699     my $qids;
1700     $i=-1;
1701     $e = q/""/;
1702     # $sql =~ s~"(([^"$e]|$e.)+)"~push(@$qids,$1);$i++;"?QI$i?"~ge;
1703     $sql =~ s~"(([^"]|"")+)"~push(@$qids,$1);$i++;"?QI$i?"~ge;
1704     #@$qids = map { s/$e'/'/g; s/^'(.*)'$/$1/; $_} @$qids;
1705     $self->{"struct"}->{"quoted_ids"} = $qids if $qids;
1706    
1707     # $sql =~ s~'(([^'\\]|\\.)+)'~push(@$fields,$1);$i++;"?$i?"~ge;
1708     # @$fields = map { s/\\'/'/g; s/^'(.*)'$/$1/; $_} @$fields;
1709     #print "$sql [@$fields]\n";# if $sql =~ /SELECT/;
1710    
1711     ## before line 1511
1712     my $comment_re = $self->{"comment_re"};
1713     # if ( $sql =~ s/($comment_re)//gs) {
1714     # $self->{"comment"} = $1;
1715     # }
1716     if ( $sql =~ /(.*)$comment_re$/s) {
1717     $sql = $1;
1718     $self->{"comment"} = $2;
1719     }
1720     if ($sql =~ /^(.*)--(.*)(\n|$)/) {
1721     $sql = $1;
1722     $self->{"comment"} = $2;
1723     }
1724    
1725     $sql =~ s/\n/ /g;
1726     $sql =~ s/\s+/ /g;
1727     $sql =~ s/(\S)\(/$1 (/g; # ensure whitespace before (
1728     $sql =~ s/\)(\S)/) $1/g; # ensure whitespace after )
1729     $sql =~ s/\(\s*/(/g; # trim whitespace after (
1730     $sql =~ s/\s*\)/)/g; # trim whitespace before )
1731     #
1732     # $sql =~ s/\s*\(/(/g; # trim whitespace before (
1733     # $sql =~ s/\)\s*/)/g; # trim whitespace after )
1734     for my $op( qw( = <> < > <= >= \|\|) ) {
1735     $sql =~ s/(\S)$op/$1 $op/g;
1736     $sql =~ s/$op(\S)/$op $1/g;
1737     }
1738     $sql =~ s/< >/<>/g;
1739     $sql =~ s/< =/<=/g;
1740     $sql =~ s/> =/>=/g;
1741     $sql =~ s/\s*,/,/g;
1742     $sql =~ s/,\s*/,/g;
1743     $sql =~ s/^\s+//;
1744     $sql =~ s/\s+$//;
1745     return $sql;
1746     }
1747    
1748     sub trim {
1749     my $str = shift or return '';
1750     $str =~ s/^\s+//;
1751     $str =~ s/\s+$//;
1752     return $str;
1753     }
1754    
1755     sub do_err {
1756     my $self = shift;
1757     my $err = shift;
1758     my $errtype = shift;
1759     my @c = caller 4;
1760     $err = "$err\n\n";
1761     # $err = $errtype ? "DIALECT ERROR: $err in $c[3]"
1762     # : "SQL ERROR: $err in $c[3]";
1763     $err = $errtype ? "DIALECT ERROR: $err"
1764     : "SQL ERROR: $err";
1765     $self->{"struct"}->{"errstr"} = $err;
1766     #$self->{"errstr"} = $err;
1767     warn $err if $self->{"PrintError"};
1768 joko 1.3 die $err if $self->{"RaiseError"};
1769 joko 1.1 return undef;
1770     }
1771    
1772     1;
1773    
1774     __END__
1775    
1776 joko 1.3 =pod
1777 joko 1.1
1778     =head1 NAME
1779    
1780     SQL::Parser -- validate, parse, or build SQL strings
1781    
1782     =head1 SYNOPSIS
1783    
1784     use SQL::Parser; # CREATE A PARSER OBJECT
1785     my $parser = SQL::Parser->new( $dialect, \%attrs );
1786    
1787     my $success = $parser->parse( $sql_string ); # PARSE A SQL STRING &
1788     if ($success) { # DISPLAY RESULTING DATA
1789     use Data::Dumper; # STRUCTURE
1790     print Dumper $parser->structure;
1791     }
1792    
1793     $parser->feature( $class, $name, $value ); # SET OR FIND STATUS OF
1794     my $has_feature = $parser->feature( $class, $name ); # A PARSER FEATURE
1795    
1796     $parser->dialect( $dialect_name ); # SET OR FIND STATUS OF
1797     my $current_dialect = $parser->dialect; # A PARSER DIALECT
1798    
1799     print $parser->errstr; # DISPLAY CURRENT ERROR
1800     # STRING
1801    
1802    
1803     =head1 DESCRIPTION
1804    
1805     SQL::Parser is a parser, builder, and sytax validator for a
1806     small but useful subset of SQL (Structured Query Language). It
1807     accepts SQL strings and returns either a detailed error message
1808     if the syntax is invalid or a data structure containing the
1809     results of the parse if the syntax is valid. It will soon also
1810     work in reverse to build a SQL string from a supplied data
1811     structure.
1812    
1813     The module can be used in batch mode to validate a series of
1814     statements, or as middle-ware for DBI drivers or other related
1815     projects. When combined with SQL::Statement version 0.2 or
1816     greater, the module can be used to actually perform the SQL
1817     commands on a variety of file formats using DBD::AnyData, or
1818     DBD::CSV, or DBD::Excel.
1819    
1820     The module makes use of a variety of configuration files
1821     located in the SQL/Dialects directory, each of which is
1822     essentially a simple text file listing things like supported
1823     data types, reserved words, and other features specific to a
1824     given dialect of SQL. These features can also be turned on or
1825     off during program execution.
1826    
1827     =head1 SUPPORTED SQL SYNTAX
1828    
1829     This module is meant primarly as a base class for DBD drivers
1830     and as such concentrates on a small but useful subset of SQL 92.
1831     It does *not* in any way pretend to be a complete SQL 92 parser.
1832     The module will continue to add new supported syntax, currently,
1833     this is what is supported:
1834    
1835     =head2 CREATE TABLE
1836    
1837     CREATE [ {LOCAL|GLOBAL} TEMPORARY ] TABLE $table
1838     (
1839 joko 1.3 $col_1 $col_type1 $col_constraints1,
1840 joko 1.1 ...,
1841 joko 1.3 $col_N $col_typeN $col_constraintsN,
1842 joko 1.1 )
1843     [ ON COMMIT {DELETE|PRESERVE} ROWS ]
1844    
1845     * col_type must be a valid data type as defined in the
1846     "valid_data_types" section of the dialect file for the
1847     current dialect
1848    
1849     * col_constriaints may be "PRIMARY KEY" or one or both of
1850     "UNIQUE" and/or "NOT NULL"
1851    
1852     * IMPORTANT NOTE: temporary tables, data types and column
1853     constraints are checked for syntax violations but are
1854     currently otherwise *IGNORED* -- they are recognized by
1855     the parser, but not by the execution engine
1856    
1857     * The following valid ANSI SQL92 options are not currently
1858     supported: table constraints, named constraints, check
1859     constriants, reference constraints, constraint
1860     attributes, collations, default clauses, domain names as
1861     data types
1862    
1863     =head2 DROP TABLE
1864    
1865     DROP TABLE $table [ RESTRICT | CASCADE ]
1866    
1867     * IMPORTANT NOTE: drop behavior (cascade or restrict) is
1868     checked for valid syntax but is otherwise *IGNORED* -- it
1869     is recognized by the parser, but not by the execution
1870     engine
1871    
1872     =head2 INSERT INTO
1873    
1874     INSERT INTO $table [ ( $col1, ..., $colN ) ] VALUES ( $val1, ... $valN )
1875    
1876     * default values are not currently supported
1877     * inserting from a subquery is not currently supported
1878    
1879     =head2 DELETE FROM
1880    
1881     DELETE FROM $table [ WHERE search_condition ]
1882    
1883     * see "search_condition" below
1884    
1885     =head2 UPDATE
1886    
1887     UPDATE $table SET $col1 = $val1, ... $colN = $valN [ WHERE search_condition ]
1888    
1889     * default values are not currently supported
1890     * see "search_condition" below
1891    
1892     =head2 SELECT
1893    
1894     SELECT select_clause
1895     FROM from_clause
1896 joko 1.3 [ WHERE search_condition ]
1897 joko 1.1 [ ORDER BY $ocol1 [ASC|DESC], ... $ocolN [ASC|DESC] ]
1898     [ LIMIT [start,] length ]
1899    
1900     * select clause ::=
1901     [DISTINCT|ALL] *
1902     | [DISTINCT|ALL] col1 [,col2, ... colN]
1903     | set_function1 [,set_function2, ... set_functionN]
1904    
1905     * set function ::=
1906     COUNT ( [DISTINCT|ALL] * )
1907     | COUNT | MIN | MAX | AVG | SUM ( [DISTINCT|ALL] col_name )
1908    
1909     * from clause ::=
1910     table1 [, table2, ... tableN]
1911     | table1 NATURAL [join_type] JOIN table2
1912     | table1 [join_type] table2 USING (col1,col2, ... colN)
1913 joko 1.3 | table1 [join_type] JOIN table2 ON table1.colA = table2.colB
1914 joko 1.1
1915     * join type ::=
1916     INNER
1917     | [OUTER] LEFT | RIGHT | FULL
1918    
1919     * if join_type is not specified, INNER is the default
1920     * if DISTINCT or ALL is not specified, ALL is the default
1921     * if start position is omitted from LIMIT clause, position 0 is
1922     the default
1923     * ON clauses may only contain equal comparisons and AND combiners
1924     * self-joins are not currently supported
1925     * if implicit joins are used, the WHERE clause must contain
1926     and equijoin condition for each table
1927 joko 1.3
1928 joko 1.1
1929     =head2 SEARCH CONDITION
1930    
1931     [NOT] $val1 $op1 $val1 [ ... AND|OR $valN $opN $valN ]
1932    
1933    
1934     =head2 OPERATORS
1935    
1936 joko 1.3 $op = | <> | < | > | <= | >=
1937 joko 1.1 | IS NULL | IS NOT NULL | LIKE | CLIKE | BETWEEN | IN
1938    
1939 joko 1.3 The "CLIKE" operator works exactly the same as the "LIKE"
1940 joko 1.1 operator, but is case insensitive. For example:
1941    
1942     WHERE foo LIKE 'bar%' # succeeds if foo is "barbaz"
1943     # fails if foo is "BARBAZ" or "Barbaz"
1944    
1945     WHERE foo CLIKE 'bar%' # succeeds for "barbaz", "Barbaz", and "BARBAZ"
1946    
1947    
1948     =head2 STRING FUNCTIONS & MATH EXPRESSIONS
1949    
1950     String functions and math expressions are supported in WHERE
1951     clauses, in the VALUES part of an INSERT and UPDATE
1952     statements. They are not currently supported in the SELECT
1953     statement. For example:
1954    
1955     SELECT * FROM foo WHERE UPPER(bar) = 'baz' # SUPPORTED
1956    
1957     SELECT UPPER(foo) FROM bar # NOT SUPPORTED
1958    
1959     =over
1960    
1961     =item TRIM ( [ [LEADING|TRAILING|BOTH] ['trim_char'] FROM ] string )
1962    
1963     Removes all occurrences of <trim_char> from the front, back, or
1964     both sides of a string.
1965    
1966     BOTH is the default if neither LEADING nor TRAILING is specified.
1967    
1968     Space is the default if no trim_char is specified.
1969    
1970     Examples:
1971    
1972 joko 1.3 TRIM( string )
1973 joko 1.1 trims leading and trailing spaces from string
1974    
1975     TRIM( LEADING FROM str )
1976     trims leading spaces from string
1977    
1978     TRIM( 'x' FROM str )
1979     trims leading and trailing x's from string
1980    
1981     =item SUBSTRING( string FROM start_pos [FOR length] )
1982    
1983     Returns the substring starting at start_pos and extending for
1984     "length" character or until the end of the string, if no
1985     "length" is supplied. Examples:
1986    
1987     SUBSTRING( 'foobar' FROM 4 ) # returns "bar"
1988    
1989     SUBSTRING( 'foobar' FROM 4 FOR 2) # returns "ba"
1990    
1991    
1992     =item UPPER(string) and LOWER(string)
1993    
1994     These return the upper-case and lower-case variants of the string:
1995    
1996     UPPER('foo') # returns "FOO"
1997     LOWER('FOO') # returns "foo"
1998    
1999     =back
2000    
2001     =head2 Identifiers (table & column names)
2002    
2003     Regular identifiers (table and column names *without* quotes around them) are case INSENSITIVE so column foo, fOo, FOO all refer to the same column.
2004    
2005     Delimited identifiers (table and column names *with* quotes around them) are case SENSITIVE so column "foo", "fOo", "FOO" each refer to different columns.
2006    
2007     A delimited identifier is *never* equal to a regular identifer (so "foo" and foo are two different columns). But don't do that :-).
2008    
2009     Remember thought that, in DBD::CSV if table names are used directly as file names, the case sensitivity depends on the OS e.g. on Windows files named foo, FOO, and fOo are the same as each other while on Unix they are different.
2010    
2011    
2012     =head1 METHODS
2013    
2014     =head2 new()
2015    
2016     The new() method creates a SQL::Parser object which can then be
2017     used to parse, validate, or build SQL strings. It takes one
2018     required parameter -- the name of the SQL dialect that will
2019     define the rules for the parser. A second optional parameter is
2020     a reference to a hash which can contain additional attributes of
2021     the parser.
2022    
2023     use SQL::Parser;
2024     my $parser = SQL::Parser->new( $dialect_name, \%attrs );
2025    
2026     The dialect_name parameter is a string containing any valid
2027     dialect such as 'ANSI', 'AnyData', or 'CSV'. See the section on
2028     the dialect() method below for details.
2029    
2030     The attribute parameter is a reference to a hash that can
2031     contain error settings for the PrintError and RaiseError
2032     attributes. See the section below on the parse() method for
2033     details.
2034    
2035     An example:
2036    
2037     use SQL::Parser;
2038     my $parser = SQL::Parser->new('AnyData', {RaiseError=>1} );
2039    
2040     This creates a new parser that uses the grammar rules
2041     contained in the .../SQL/Dialects/AnyData.pm file and which
2042     sets the RaiseError attribute to true.
2043    
2044     For those needing backwards compatibility with SQL::Statement
2045     version 0.1x and lower, the attribute hash may also contain
2046     feature settings. See the section "FURTHER DETAILS - Backwards
2047     Compatibility" below for details.
2048    
2049    
2050     =head2 parse()
2051    
2052     Once a SQL::Parser object has been created with the new()
2053     method, the parse() method can be used to parse any number of
2054     SQL strings. It takes a single required parameter -- a string
2055     containing a SQL command. The SQL string may optionally be
2056     terminated by a semicolon. The parse() method returns a true
2057     value if the parse is successful and a false value if the parse
2058     finds SQL syntax errors.
2059    
2060     Examples:
2061    
2062     1) my $success = $parser->parse('SELECT * FROM foo');
2063    
2064     2) my $sql = 'SELECT * FROM foo';
2065     my $success = $parser->parse( $sql );
2066    
2067     3) my $success = $parser->parse(qq!
2068     SELECT id,phrase
2069     FROM foo
2070     WHERE id < 7
2071     AND phrase <> 'bar'
2072     ORDER BY phrase;
2073     !);
2074    
2075     4) my $success = $parser->parse('SELECT * FRoOM foo ');
2076    
2077     In examples #1,#2, and #3, the value of $success will be true
2078     because the strings passed to the parse() method are valid SQL
2079     strings.
2080    
2081     In example #4, however, the value of $success will be false
2082     because the string contains a SQL syntax error ('FRoOM' instead
2083     of 'FROM').
2084    
2085     In addition to checking the return value of parse() with a
2086     variable like $success, you may use the PrintError and
2087     RaiseError attributes as you would in a DBI script:
2088    
2089     * If PrintError is true, then SQL syntax errors will be sent as
2090     warnings to STDERR (i.e. to the screen or to a file if STDERR
2091     has been redirected). This is set to true by default which
2092     means that unless you specifically turn it off, all errors
2093     will be reported.
2094    
2095     * If RaiseError is true, then SQL syntax errors will cause the
2096     script to die, (i.e. the script will terminate unless wrapped
2097     in an eval). This is set to false by default which means
2098     that unless you specifically turn it on, scripts will
2099     continue to operate even if there are SQL syntax errors.
2100    
2101     Basically, you should leave PrintError on or else you will not
2102     be warned when an error occurs. If you are simply validating a
2103     series of strings, you will want to leave RaiseError off so that
2104     the script can check all strings regardless of whether some of
2105     them contain SQL errors. However, if you are going to try to
2106     execute the SQL or need to depend that it is correct, you should
2107     set RaiseError on so that the program will only continue to
2108     operate if all SQL strings use correct syntax.
2109    
2110     IMPORTANT NOTE #1: The parse() method only checks syntax, it
2111     does NOT verify if the objects listed actually exist. For
2112     example, given the string "SELECT model FROM cars", the parse()
2113     method will report that the string contains valid SQL but that
2114     will not tell you whether there actually is a table called
2115     "cars" or whether that table contains a column called 'model'.
2116     Those kinds of verifications can be performed by the
2117     SQL::Statement module, not by SQL::Parser by itself.
2118    
2119     IMPORTANT NOTE #2: The parse() method uses rules as defined by
2120     the selected dialect configuration file and the feature()
2121     method. This means that a statement that is valid in one
2122     dialect may not be valid in another. For example the 'CSV' and
2123     'AnyData' dialects define 'BLOB' as a valid data type but the
2124     'ANSI' dialect does not. Therefore the statement 'CREATE TABLE
2125     foo (picture BLOB)' would be valid in the first two dialects but
2126     would produce a syntax error in the 'ANSI' dialect.
2127    
2128     =head2 structure()
2129    
2130     After a SQL::Parser object has been created and the parse()
2131     method used to parse a SQL string, the structure() method
2132     returns the data structure of that string. This data structure
2133     may be passed on to other modules (e.g. SQL::Statement) or it
2134     may be printed out using, for example, the Data::Dumper module.
2135    
2136     The data structure contains all of the information in the SQL
2137     string as parsed into its various components. To take a simple
2138     example:
2139    
2140     $parser->parse('SELECT make,model FROM cars');
2141     use Data::Dumper;
2142     print Dumper $parser->structure;
2143    
2144     Would produce:
2145    
2146     $VAR1 = {
2147     'column_names' => [
2148     'make',
2149     'model'
2150     ],
2151     'command' => 'SELECT',
2152     'table_names' => [
2153     'cars'
2154     ]
2155     };
2156    
2157     Please see the section "FURTHER DETAILS -- Parse structures"
2158     below for further examples.
2159    
2160     =head2 build()
2161    
2162     This method is in progress and should be available soon.
2163    
2164     =head2 dialect()
2165    
2166     $parser->dialect( $dialect_name ); # load a dialect configuration file
2167     my $dialect = $parser->dialect; # get the name of the current dialect
2168    
2169     For example:
2170    
2171     $parser->dialect('AnyData'); # loads the AnyData config file
2172     print $parser->dialect; # prints 'AnyData'
2173    
2174     The $dialect_name parameter may be the name of any dialect
2175     configuration file on your system. Use the
2176     $parser->list('dialects') method to see a list of available
2177     dialects. At a minimum it will include "ANSI", "CSV", and
2178     "AnyData". For backwards compatiblity 'Ansi' is accepted as a
2179     synonym for 'ANSI', otherwise the names are case sensitive.
2180    
2181     Loading a new dialect configuration file erases all current
2182     parser features and resets them to those defined in the
2183     configuration file.
2184    
2185     See the section above on "Dialects" for details of these
2186     configuration files.
2187    
2188     =head2 feature()
2189    
2190     Features define the rules to be used by a specific parser
2191     instance. They are divided into the following classes:
2192    
2193     * valid_commands
2194     * valid_options
2195     * valid_comparison_operators
2196     * valid_data_types
2197     * reserved_words
2198    
2199     Within each class a feature name is either enabled or
2200     disabled. For example, under "valid_data_types" the name "BLOB"
2201     may be either disabled or enabled. If it is not eneabled
2202     (either by being specifically disabled, or simply by not being
2203     specified at all) then any SQL string using "BLOB" as a data
2204     type will throw a syntax error "Invalid data type: 'BLOB'".
2205    
2206     The feature() method allows you to enable, disable, or check the
2207     status of any feature.
2208    
2209     $parser->feature( $class, $name, 1 ); # enable a feature
2210    
2211     $parser->feature( $class, $name, 0 ); # disable a feature
2212    
2213     my $feature = $parser->feature( $class, $name ); # show status of a feature
2214    
2215     For example:
2216    
2217     $parser->feature('reserved_words','FOO',1); # make 'FOO' a reserved word
2218    
2219     $parser->feature('valid_data_types','BLOB',0); # disallow 'BLOB' as a
2220     # data type
2221    
2222     # determine if the LIKE
2223     # operator is supported
2224     my $LIKE = $parser->feature('valid_operators','LIKE');
2225    
2226     See the section below on "Backwards Compatibility" for use of
2227     the feature() method with SQL::Statement 0.1x style parameters.
2228    
2229     =head2 list()
2230    
2231     =head2 errstr()
2232    
2233     =head1 FURTHER DETAILS
2234    
2235     =head2 Dialect Configuration Files
2236    
2237     These will change completely when Tim finalizes the DBI get_info method.
2238    
2239     =head2 Parse Structures
2240    
2241     Here are some further examples of the data structures returned
2242     by the structure() method after a call to parse(). Only
2243     specific details are shown for each SQL instance, not the entire
2244     struture.
2245    
2246     'SELECT make,model, FROM cars'
2247    
2248     command => 'SELECT',
2249     table_names => [ 'cars' ],
2250     column_names => [ 'make', 'model' ],
2251    
2252     'CREATE TABLE cars ( id INTEGER, model VARCHAR(40) )'
2253    
2254     column_defs => {
2255     id => { data_type => INTEGER },
2256     model => { data_type => VARCHAR(40) },
2257     },
2258    
2259     'SELECT DISTINCT make FROM cars'
2260    
2261     set_quantifier => 'DISTINCT',
2262    
2263     'SELECT MAX (model) FROM cars'
2264    
2265     set_function => {
2266     name => 'MAX',
2267     arg => 'models',
2268     },
2269    
2270     'SELECT * FROM cars LIMIT 5,10'
2271    
2272     limit_clause => {
2273     offset => 5,
2274     limit => 10,
2275     },
2276    
2277     'SELECT * FROM vars ORDER BY make, model DESC'
2278    
2279     sort_spec_list => [
2280     { make => 'ASC' },
2281     { model => 'DESC' },
2282     ],
2283    
2284     "INSERT INTO cars VALUES ( 7, 'Chevy', 'Impala' )"
2285    
2286     values => [ 7, 'Chevy', 'Impala' ],
2287    
2288    
2289     =head2 Backwards Compatibility
2290    
2291     This module can be used in conjunction with SQL::Statement,
2292     version 0.2 and higher. Earlier versions of SQL::Statement
2293     included a SQL::Parser as a submodule that used slightly
2294     different syntax than the current version. The current version
2295     supports all of this earlier syntax although new users are
2296     encouraged to use the new syntax listed above. If the syntax
2297     listed below is used, the module should be able to be subclassed
2298     exactly as it was with the older SQL::Statement versions and
2299     will therefore not require any modules or scripts that used it
2300     to make changes.
2301    
2302     In the old style, features of the parser were accessed with this
2303     syntax:
2304    
2305     feature('create','type_blob',1); # allow BLOB as a data type
2306     feature('create','type_blob',0); # disallow BLOB as a data type
2307     feature('select','join',1); # allow multi-table statements
2308    
2309     The same settings could be acheieved in calls to new:
2310    
2311     my $parser = SQL::Parser->new(
2312     'Ansi',
2313     {
2314     create => {type_blob=>1},
2315     select => {join=>1},
2316     },
2317     );
2318    
2319     Both of these styles of setting features are supported in the
2320     current SQL::Parser.
2321    
2322     =head1 ACKNOWLEDGEMENTS
2323    
2324     *Many* thanks to Ilya Sterin who wrote most of code for the
2325     build() method and who assisted on the parentheses parsing code
2326     and who proved a great deal of support, advice, and testing
2327     throughout the development of the module.
2328    
2329     =head1 AUTHOR & COPYRIGHT
2330    
2331     This module is copyright (c) 2001 by Jeff Zucker.
2332     All rights reserved.
2333    
2334     The module may be freely distributed under the same terms as
2335     Perl itself using either the "GPL License" or the "Artistic
2336     License" as specified in the Perl README file.
2337    
2338     Jeff can be reached at: jeff@vpservices.com.
2339    
2340     =cut

MailToCvsAdmin">MailToCvsAdmin
ViewVC Help
Powered by ViewVC 1.1.26 RSS 2.0 feed