-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathroachize.pl
executable file
·117 lines (100 loc) · 3.35 KB
/
roachize.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#!/usr/bin/env perl
use strict;
use Getopt::Long 'HelpMessage', qw(:config bundling);
GetOptions ('source=s' => \my $sourceDumpFile,
'dest=s' => \my $destDumpFile,
'omitdata' => \my $omitData,
'help' => sub { HelpMessage(0) },
) or HelpMessage(1);
HelpMessage(1) unless $sourceDumpFile && $destDumpFile;
my $haveDumpedFKIndicies = 0;
my %pk;
my %pkc;
my @fk;
# First find the primary key definitions and primary key column names for serial conversion
# and save them
# Find foreign key constraints and collect them so we can add required indexes
open(F, "<$sourceDumpFile") || die;
while(<F>) {
if (/^ALTER TABLE ONLY (?:public\.)?(\S+)/i .. /^\s*$/) {
my $tablename = $1;
if (/\s+ADD CONSTRAINT (\S+)_pkey (PRIMARY KEY .*);/) {
my $pkdef = $2;
$pk{$tablename} = $pkdef;
if ($pkdef =~ /\(([a-z0-9_]+)\)/) {
$pkc{$tablename} = $1;
}
}
if (/\s+ADD CONSTRAINT (\S+) FOREIGN KEY \((\S+)\)/) {
push @fk, "CREATE INDEX i_${tablename}_$1 ON $tablename ($2);";
}
}
#ALTER TABLE ONLY alert
#ADD CONSTRAINT a_customer_fk FOREIGN KEY (customer_id) REFERENCES customer(customer_id) ;
}
close(F);
open(F, "<$sourceDumpFile") || die;
open(D, ">$destDumpFile") || die;
my $curtable;
while(<F>) {
# remove copy tables if $omitData is true
if (/^COPY / .. /^\\\./) {
$_ = "" if $omitData;
next;
} else {
$_ = cleanUnsupportedKeywords($_);
}
if (/^CREATE SEQUENCE/ .. /^\s*$/) {
$_ = "-- $_";
}
# In a create table we check the table name to see if we have a PK stanza saved to print
# at the end
if (my $num = /^CREATE TABLE (?:public\.)?(\S+)/i .. /^\)/) {
if ($num == 1) {
$curtable = $1;
}
# if we are on the column that will be the pk, change the type to serial
if ($pkc{$curtable} && /^\s+$pkc{$curtable} integer/) {
s/integer/serial/;
}
if ($num =~ /E0/ && $pk{$curtable}) {
print D ", $pk{$curtable}\n";
$curtable = '';
} else {
# quote all column names that are not already quoted
s/^(\s+)([^\" ]\S+)/$1"$2"/;
}
}
# if we hit a create index, dump out the added ones for FKs we collcted in first pass
if (/^CREATE INDEX/ && !$haveDumpedFKIndicies ) {
$haveDumpedFKIndicies = 1;
print D join("\n", @fk), "\n";
}
# If this is a PRIMARY KEY addition, we comment it out otherwise we leave it alone
if (/^ALTER TABLE ONLY (\S+)/) {
my $lookahead = <F>;
$lookahead = cleanUnsupportedKeywords($lookahead);
if ($lookahead =~ /PRIMARY KEY/) {
print D "-- $_";
print D "-- $lookahead";
} else {
print D $_;
print D $lookahead;
}
$_ = "";
}
} continue {
print D;
}
close(F);
close(D);
sub cleanUnsupportedKeywords {
my($f) = @_;
$f =~ s/^(ALTER TABLE \S+ OWNER |CREATE EXTENSION|COMMENT ON|ALTER SEQUENCE|ALTER TABLE.*nextval|SELECT pg_catalog)|GRANT|REVOKE/-- \1/;
$f =~ s/^(SET (?!client_encoding|standard_conforming_strings|client_min_messages|search_path))/-- \1/;
$f =~ s/DEFERRABLE INITIALLY DEFERRED//;
$f =~ s/ON DELETE SET NULL//;
$f =~ s/USING btree //;
return $f;
}
#ERROR: expected 17 values, got 16