486 lines
9.0 KiB
Perl
486 lines
9.0 KiB
Perl
#! /usr/bin/perl
|
|
# $OpenBSD: Inserter.pm,v 1.7 2010/04/26 08:52:09 espie Exp $
|
|
#
|
|
# Copyright (c) 2006-2010 Marc Espie <espie@openbsd.org>
|
|
#
|
|
# Permission to use, copy, modify, and distribute this software for any
|
|
# purpose with or without fee is hereby granted, provided that the above
|
|
# copyright notice and this permission notice appear in all copies.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
# WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
# MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
# ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
# WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
# ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
# OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
use strict;
|
|
use warnings;
|
|
|
|
package Composite;
|
|
sub AUTOLOAD
|
|
{
|
|
our $AUTOLOAD;
|
|
my $fullsub = $AUTOLOAD;
|
|
(my $sub = $fullsub) =~ s/.*:://o;
|
|
return if $sub eq 'DESTROY'; # special case
|
|
my $self = $_[0];
|
|
# verify it makes sense
|
|
if ($self->element_class->can($sub)) {
|
|
no strict "refs";
|
|
# create the sub to avoid regenerating further calls
|
|
*$fullsub = sub {
|
|
my $self = shift;
|
|
$self->visit($sub, @_);
|
|
};
|
|
# and jump to it
|
|
goto &$fullsub;
|
|
} else {
|
|
die "Can't call $sub on ".ref($self);
|
|
}
|
|
}
|
|
|
|
|
|
package InserterList;
|
|
our @ISA = qw(Composite);
|
|
sub element_class() { 'NormalInserter' }
|
|
|
|
sub new
|
|
{
|
|
my $class = shift;
|
|
bless [], $class;
|
|
}
|
|
|
|
sub add
|
|
{
|
|
my $self = shift;
|
|
push(@$self, @_);
|
|
}
|
|
|
|
sub visit
|
|
{
|
|
my ($self, $method, @r) = @_;
|
|
for my $i (@$self) {
|
|
$i->$method(@r);
|
|
}
|
|
}
|
|
|
|
package AbstractInserter;
|
|
# this is the object to use to put stuff into the db...
|
|
sub new
|
|
{
|
|
my ($class, $db, $i, $verbose) = @_;
|
|
$db->do("PRAGMA foreign_keys=ON");
|
|
bless {
|
|
db => $db,
|
|
transaction => 0,
|
|
threshold => $i,
|
|
vars => {},
|
|
tables_created => {},
|
|
verbose => $verbose,
|
|
}, $class;
|
|
}
|
|
|
|
sub create_tables
|
|
{
|
|
my ($self, $vars) = @_;
|
|
|
|
$self->create_path_table;
|
|
while (my ($name, $varclass) = each %$vars) {
|
|
$self->handle_column($varclass->column($name));
|
|
$varclass->create_table($self);
|
|
}
|
|
|
|
$self->create_ports_table;
|
|
$self->prepare_normal_inserter('Ports', @{$self->{varlist}});
|
|
$self->prepare_normal_inserter('Paths', 'PKGPATH');
|
|
$self->create_view_info;
|
|
$self->commit_to_db;
|
|
print '-'x50, "\n" if $self->{verbose};
|
|
}
|
|
|
|
sub handle_column
|
|
{
|
|
my ($self, $column) = @_;
|
|
push(@{$self->{varlist}}, $column->{name});
|
|
push(@{$self->{columnlist}}, $column);
|
|
}
|
|
|
|
sub create_view_info
|
|
{
|
|
}
|
|
|
|
sub map_columns
|
|
{
|
|
my ($self, $mapper, $o, @columns) = @_;
|
|
$mapper .= '_schema';
|
|
return grep {defined $_} (map {$_->$mapper($o)} @columns);
|
|
}
|
|
|
|
sub make_table
|
|
{
|
|
my ($self, $class, $constraint, @columns) = @_;
|
|
|
|
return if defined $self->{tables_created}->{$class->table};
|
|
|
|
unshift(@columns, PathColumn->new);
|
|
for my $c (@columns) {
|
|
$c->set_vartype($class) unless defined $c->{vartype};
|
|
}
|
|
my @l = $self->map_columns('normal', $self, @columns);
|
|
push(@l, $constraint) if defined $constraint;
|
|
$self->new_table($class->table, @l);
|
|
}
|
|
|
|
sub set
|
|
{
|
|
my ($self, $ref) = @_;
|
|
$self->{ref} = $ref;
|
|
}
|
|
|
|
sub db
|
|
{
|
|
return shift->{db};
|
|
}
|
|
|
|
sub last_id
|
|
{
|
|
return shift->db->func('last_insert_rowid');
|
|
}
|
|
|
|
sub insert_done
|
|
{
|
|
my $self = shift;
|
|
$self->{transaction}++;
|
|
}
|
|
|
|
sub new_table
|
|
{
|
|
my ($self, $name, @cols) = @_;
|
|
|
|
return if defined $self->{tables_created}->{$name};
|
|
|
|
$self->db->do("DROP TABLE IF EXISTS $name");
|
|
print "CREATE TABLE $name (".join(', ', @cols).")\n"
|
|
if $self->{verbose};
|
|
$self->db->do("CREATE TABLE $name (".join(', ', @cols).")");
|
|
$self->{tables_created}->{$name} = 1;
|
|
}
|
|
|
|
sub prepare
|
|
{
|
|
my ($self, $s) = @_;
|
|
return $self->db->prepare($s);
|
|
}
|
|
|
|
sub prepare_inserter
|
|
{
|
|
my ($ins, $table, @cols) = @_;
|
|
$ins->{insert}->{$table} = $ins->prepare(
|
|
"INSERT OR REPLACE INTO $table (".
|
|
join(', ', @cols).
|
|
") VALUES (".
|
|
join(', ', map {'?'} @cols).")");
|
|
}
|
|
|
|
sub prepare_normal_inserter
|
|
{
|
|
my ($ins, $table, @cols) = @_;
|
|
$ins->prepare_inserter($table, "FULLPKGPATH", @cols);
|
|
}
|
|
|
|
sub finish_port
|
|
{
|
|
my $self = shift;
|
|
my @values = ($self->ref);
|
|
for my $i (@{$self->{varlist}}) {
|
|
push(@values, $self->{vars}->{$i});
|
|
}
|
|
$self->insert('Ports', @values);
|
|
$self->{vars} = {};
|
|
if ($self->{transaction} >= $self->{threshold}) {
|
|
$self->commit_to_db;
|
|
$self->{transaction} = 0;
|
|
}
|
|
}
|
|
|
|
sub add_to_port
|
|
{
|
|
my ($self, $var, $value) = @_;
|
|
$self->{vars}->{$var} = $value;
|
|
}
|
|
|
|
sub create_ports_table
|
|
{
|
|
my $self = shift;
|
|
|
|
my @columns = sort {$a->name cmp $b->name} @{$self->{columnlist}};
|
|
unshift(@columns, PathColumn->new);
|
|
my @l = $self->map_columns('normal', $self, @columns);
|
|
$self->new_table("Ports", @l, "UNIQUE(FULLPKGPATH)");
|
|
}
|
|
|
|
sub ref
|
|
{
|
|
return shift->{ref};
|
|
}
|
|
|
|
sub insert
|
|
{
|
|
my $self = shift;
|
|
my $table = shift;
|
|
$self->{insert}->{$table}->execute(@_);
|
|
$self->insert_done;
|
|
}
|
|
|
|
sub add_var
|
|
{
|
|
my ($self, $v) = @_;
|
|
$v->add($self);
|
|
}
|
|
|
|
sub commit_to_db
|
|
{
|
|
my $self = shift;
|
|
$self->db->commit;
|
|
}
|
|
|
|
package CompactInserter;
|
|
our @ISA = qw(AbstractInserter);
|
|
|
|
our $c = {
|
|
Library => 0,
|
|
Run => 1,
|
|
Build => 2,
|
|
Regress => 3
|
|
};
|
|
|
|
sub convert_depends
|
|
{
|
|
my ($self, $value) = @_;
|
|
return $c->{$value};
|
|
}
|
|
|
|
|
|
sub pathref
|
|
{
|
|
my ($self, $name) = @_;
|
|
$name = "FULLPKGPATH" if !defined $name;
|
|
return "$name INTEGER NOT NULL REFERENCES Paths(ID)";
|
|
}
|
|
|
|
sub value
|
|
{
|
|
my ($self, $k, $name) = @_;
|
|
$name = "VALUE" if !defined $name;
|
|
if (defined $k) {
|
|
return "$name INTEGER NOT NULL REFERENCES $k(KEYREF)";
|
|
} else {
|
|
return "$name TEXT NOT NULL";
|
|
}
|
|
}
|
|
|
|
sub optvalue
|
|
{
|
|
my ($self, $k, $name) = @_;
|
|
$name = "VALUE" if !defined $name;
|
|
if (defined $k) {
|
|
return "$name INTEGER REFERENCES $k(KEYREF)";
|
|
} else {
|
|
return "$name TEXT";
|
|
}
|
|
}
|
|
|
|
sub create_view
|
|
{
|
|
my ($self, $table, @columns) = @_;
|
|
|
|
unshift(@columns, PathColumn->new);
|
|
my $name = "_$table";
|
|
my @l = $self->map_columns('view', $table, @columns);
|
|
my @j = $self->map_columns('join', $table, @columns);
|
|
my $v = "CREATE VIEW $name AS SELECT ".join(", ", @l). " FROM ".$table.' '.join(' ', @j);
|
|
$self->db->do("DROP VIEW IF EXISTS $name");
|
|
print "$v\n" if $self->{verbose};
|
|
$self->db->do($v);
|
|
}
|
|
|
|
sub make_table
|
|
{
|
|
my ($self, $class, $constraint, @columns) = @_;
|
|
|
|
return if defined $self->{tables_created}->{$class->table};
|
|
|
|
$self->SUPER::make_table($class, $constraint, @columns);
|
|
$self->create_view($class->table, @columns);
|
|
}
|
|
|
|
sub create_path_table
|
|
{
|
|
my $self = shift;
|
|
$self->new_table("Paths", "ID INTEGER PRIMARY KEY",
|
|
"FULLPKGPATH TEXT NOT NULL UNIQUE",
|
|
"PKGPATH INTEGER REFERENCES Paths(ID)");
|
|
}
|
|
|
|
sub handle_column
|
|
{
|
|
my ($self, $column) = @_;
|
|
if (!defined($column->{vartype}->table)) {
|
|
$self->SUPER::handle_column($column);
|
|
}
|
|
}
|
|
|
|
sub create_view_info
|
|
{
|
|
my $self = shift;
|
|
my @columns = sort {$a->name cmp $b->name} @{$self->{columnlist}};
|
|
$self->create_view("Ports", @columns);
|
|
$self->{find_pathkey} =
|
|
$self->prepare("SELECT ID From Paths WHERE FULLPKGPATH=?");
|
|
}
|
|
|
|
sub find_pathkey
|
|
{
|
|
my ($self, $key) = @_;
|
|
|
|
if (!defined $key or $key eq '') {
|
|
print STDERR "Empty pathkey\n";
|
|
return 0;
|
|
}
|
|
|
|
# get pathkey for existing value
|
|
$self->{find_pathkey}->execute($key);
|
|
my $z = $self->{find_pathkey}->fetchrow_arrayref;
|
|
if (!defined $z) {
|
|
# if none, we create one
|
|
my $path = $key;
|
|
$path =~ s/\,.*//;
|
|
if ($path ne $key) {
|
|
$path = $self->find_pathkey($path);
|
|
} else {
|
|
$path = undef;
|
|
}
|
|
$self->insert('Paths', $key, $path);
|
|
return $self->last_id;
|
|
} else {
|
|
return $z->[0];
|
|
}
|
|
}
|
|
|
|
sub set_newkey
|
|
{
|
|
my ($self, $key) = @_;
|
|
|
|
$self->set($self->find_pathkey($key));
|
|
}
|
|
|
|
sub find_keyword_id
|
|
{
|
|
my ($self, $key, $t) = @_;
|
|
$self->{$t}->{find_key1}->execute($key);
|
|
my $a = $self->{$t}->{find_key1}->fetchrow_arrayref;
|
|
if (!defined $a) {
|
|
$self->{$t}->{find_key2}->execute($key);
|
|
$self->insert_done;
|
|
return $self->last_id;
|
|
} else {
|
|
return $a->[0];
|
|
}
|
|
}
|
|
|
|
sub create_keyword_table
|
|
{
|
|
my ($self, $t) = @_;
|
|
$self->new_table($t,
|
|
"KEYREF INTEGER PRIMARY KEY AUTOINCREMENT",
|
|
"VALUE TEXT NOT NULL UNIQUE");
|
|
$self->{$t}->{find_key1} = $self->prepare("SELECT KEYREF FROM $t WHERE VALUE=?");
|
|
$self->{$t}->{find_key2} = $self->prepare("INSERT INTO $t (VALUE) VALUES (?)");
|
|
}
|
|
|
|
package NormalInserter;
|
|
our @ISA = qw(AbstractInserter);
|
|
|
|
our $c = {
|
|
Library => 'L',
|
|
Run => 'R',
|
|
Build => 'B',
|
|
Regress => 'Regress'
|
|
};
|
|
|
|
sub convert_depends
|
|
{
|
|
my ($self, $value) = @_;
|
|
return $c->{$value};
|
|
}
|
|
|
|
sub create_path_table
|
|
{
|
|
my $self = shift;
|
|
$self->new_table("Paths", "FULLPKGPATH TEXT NOT NULL PRIMARY KEY",
|
|
"PKGPATH TEXT NOT NULL");
|
|
}
|
|
|
|
sub pathref
|
|
{
|
|
my ($self, $name) = @_;
|
|
$name = "FULLPKGPATH" if !defined $name;
|
|
return "$name TEXT NOT NULL";
|
|
}
|
|
|
|
sub value
|
|
{
|
|
my ($self, $k, $name) = @_;
|
|
$name = "VALUE" if !defined $name;
|
|
return "$name TEXT NOT NULL";
|
|
}
|
|
|
|
sub optvalue
|
|
{
|
|
my ($self, $k, $name) = @_;
|
|
$name = "VALUE" if !defined $name;
|
|
return "$name TEXT";
|
|
}
|
|
|
|
sub key
|
|
{
|
|
return "TEXT NOT NULL";
|
|
}
|
|
|
|
sub optkey
|
|
{
|
|
return "TEXT";
|
|
}
|
|
|
|
sub set_newkey
|
|
{
|
|
my ($self, $key) = @_;
|
|
|
|
my $path = $key;
|
|
$path =~ s/\,.*//;
|
|
$self->insert('Paths', $key, $path);
|
|
$self->set($key);
|
|
}
|
|
|
|
sub find_pathkey
|
|
{
|
|
my ($self, $key) = @_;
|
|
|
|
return $key;
|
|
}
|
|
|
|
# no keyword for this dude
|
|
sub find_keyword_id
|
|
{
|
|
my ($self, $key, $t) = @_;
|
|
return $key;
|
|
}
|
|
|
|
sub create_keyword_table
|
|
{
|
|
}
|
|
|
|
1;
|