openbsd-ports/databases/sqlports/files/mksqlitedb

924 lines
18 KiB
Perl

#! /usr/bin/perl
# $OpenBSD: mksqlitedb,v 1.9 2008/11/25 18:01:15 espie Exp $
#
# Copyright (c) 2006 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.
# example script that shows how to store all variable values into a
# database, using SQLite for that purpose.
#
# usage: cd /usr/ports && make dump-vars |mksqlitedb
use strict;
use warnings;
use Getopt::Std;
our ($opt_v);
sub words($)
{
my $v = shift;
$v =~ s/^\s+//;
$v =~ s/\s+$//;
return split(/\s+/, $v);
}
package AbstractInserter;
# this is the object to use to put stuff into the db...
sub new
{
my ($class, $db, $i) = @_;
bless {
db => $db,
transaction => 0,
threshold => $i,
vars => {},
tables_created => {}
}, $class;
}
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;
if ($self->{transaction}++ % $self->{threshold} == 0) {
$self->db->commit;
}
}
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 $main::opt_v;
$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} = {};
}
sub add_to_port
{
my ($self, $var, $value) = @_;
$self->{vars}->{$var} = $value;
}
sub create_tables
{
my ($self, $vars) = @_;
$self->db->commit;
$self->new_table("Ports", $self->pathref." PRIMARY KEY",
map {$_." ".$vars->{$_}->sqltype } sort @{$self->{varlist}});
$self->prepare_normal_inserter('Ports', @{$self->{varlist}});
$self->prepare_normal_inserter('Paths', 'PKGPATH');
}
sub ref
{
return shift->{ref};
}
sub insert
{
my $self = shift;
my $table = shift;
$self->{insert}->{$table}->execute(@_);
$self->insert_done;
}
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
{
return "FULLPKGPATH INTEGER NOT NULL";
}
sub create_tables
{
my ($self, $vars) = @_;
# create the various tables, dropping old versions
my @keys;
while (my ($name, $class) = each %$vars) {
if (!defined( $class->table() )) {
push(@keys, $name." ".$class->sqltype);
push(@{$self->{varlist}}, $name);
}
$class->create_table($self);
}
$self->new_table("Paths", "ID INTEGER PRIMARY KEY",
"FULLPKGPATH TEXT NOT NULL UNIQUE", "PKGPATH INTEGER");
$self->SUPER::create_tables($vars);
$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_tables
{
my ($self, $vars) = @_;
# create the various tables, dropping old versions
my @keys;
while (my ($name, $class) = each %$vars) {
push(@keys, $name." ".$class->sqltype);
push(@{$self->{varlist}}, $name);
$class->create_table($self);
}
$self->new_table("Paths", "FULLPKGPATH TEXT NOT NULL PRIMARY KEY",
"PKGPATH TEXT NOT NULL");
$self->SUPER::create_tables($vars);
}
sub pathref
{
return "FULLPKGPATH TEXT NOT NULL";
}
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
{
}
# use a Template Method approach to store the variable values.
# rule: we store each value in the main table, after converting YesNo
# variables to undef/1. Then, in addition, we process specific variables
# to store them in secondary tables (because of one/many associations).
package AnyVar;
sub new
{
my ($class, $var, $value) = @_;
bless [$var, $value], $class;
}
sub var
{
return shift->[0];
}
sub value
{
return shift->[1];
}
sub add
{
my ($self, $ins) = @_;
$ins->add_to_port($self->var, $self->value);
}
sub add_value
{
my ($self, $ins, $value) = @_;
$ins->add_to_port($self->var, $value);
}
sub sqltype()
{
return "TEXT";
}
sub table()
{
return undef;
}
sub keyword
{
my ($self, $ins, $value) = @_;
return $ins->find_keyword_id($value, $self->keyword_table);
}
# by default, there's no separate table
sub create_table
{
}
sub keyword_table()
{
return "Keywords";
}
sub insert
{
my $self = shift;
my $ins = shift;
$ins->insert($self->table, @_);
}
sub normal_insert
{
my $self = shift;
my $ins = shift;
$ins->insert($self->table, $ins->ref, @_);
}
package KeyVar;
our @ISA=(qw(AnyVar));
sub add
{
my ($self, $ins) = @_;
$self->add_value($ins, $self->keyword($ins, $self->value));
}
sub sqltype()
{
return "INTEGER NOT NULL";
}
sub create_table
{
my ($self, $inserter) = @_;
$inserter->create_keyword_table($self->keyword_table);
}
package ArchKeyVar;
our @ISA=(qw(KeyVar));
sub keyword_table()
{ 'Arch' }
package OptKeyVar;
our @ISA=(qw(KeyVar));
sub add
{
my ($self, $ins) = @_;
if ($self->value ne '') {
$self->SUPER::add($ins);
}
}
sub sqltype()
{
return "INTEGER";
}
package YesNoVar;
our @ISA=(qw(AnyVar));
sub add
{
my ($self, $ins) = @_;
$self->add_value($ins, $self->value =~ m/^Yes/i ? 1 : undef);
}
sub sqltype()
{
return "INTEGER";
}
# variable is always defined, but we don't need to store empty values.
package DefinedVar;
our @ISA=(qw(AnyVar));
sub add
{
my ($self, $ins) = @_;
return if $self->value eq '';
$self->SUPER::add($ins);
}
# all the dependencies are converted into list. Stuff like LIB_DEPENDS will
# end up being treated as WANTLIB as well.
package DependsVar;
our @ISA=(qw(AnyVar));
sub add
{
my ($self, $ins) = @_;
$self->SUPER::add($ins);
for my $depends (main::words $self->value) {
my ($libs, $pkgspec, $pkgpath2, $rest) = split(/\:/, $depends);
if (!defined $pkgpath2) {
print STDERR "Wrong depends $depends\n";
return;
}
$self->normal_insert($ins, $depends,
$ins->find_pathkey($pkgpath2),
$ins->convert_depends($self->depends_type),
$pkgspec, $rest);
if ($libs ne '') {
for my $lib (split(/\,/, $libs)) {
$self->add_lib($ins, $lib);
}
}
}
}
sub create_table
{
my ($self, $inserter) = @_;
$inserter->new_table($self->table, $inserter->pathref,
"FULLDEPENDS TEXT NOT NULL", "PKGSPEC TEXT" , "REST TEXT",
"DEPENDSPATH TEXT NOT NULL", "TYPE TEXT NOT NULL");
$inserter->prepare_normal_inserter($self->table,
"FULLDEPENDS", "DEPENDSPATH", "TYPE", "PKGSPEC", "REST");
}
sub add_lib
{
}
sub table()
{
return "Depends";
}
sub sqltype()
{
return "TEXT";
}
package LibDependsVar;
our @ISA=(qw(DependsVar));
sub depends_type() { 'Library' }
sub add_lib
{
my ($self, $ins, $lib) = @_;
WantlibVar->add_value($ins, $lib);
}
package RunDependsVar;
our @ISA=(qw(DependsVar));
sub depends_type() { 'Run' }
package BuildDependsVar;
our @ISA=(qw(DependsVar));
sub depends_type() { 'Build' }
package RegressDependsVar;
our @ISA=(qw(DependsVar));
sub depends_type() { 'Regress' }
# Stuff that gets stored in another table
package SecondaryVar;
our @ISA=(qw(KeyVar));
sub add_value
{
my ($self, $ins, $value) = @_;
$self->normal_insert($ins, $value);
}
sub add_keyword
{
my ($self, $ins, $value) = @_;
$self->add_value($ins, $self->keyword($ins, $value));
}
sub create_table
{
my ($self, $inserter) = @_;
$inserter->new_table($self->table, $inserter->pathref,
"VALUE TEXT NOT NULL", "UNIQUE(FULLPKGPATH, VALUE)");
$self->SUPER::create_table($inserter);
$inserter->prepare_normal_inserter($self->table, "VALUE");
}
sub keyword_table()
{
return "Keywords";
}
sub sqltype() { "TEXT" }
package MasterSitesVar;
our @ISA=(qw(OptKeyVar));
sub add
{
my ($self, $ins) = @_;
my $n;
if ($self->var =~ m/^MASTER_SITES(\d)$/) {
$n = $1;
}
$self->normal_insert($ins, $n, $self->value);
}
sub create_table
{
my ($self, $inserter) = @_;
$inserter->new_table($self->table, $inserter->pathref,
"N INTEGER", "VALUE TEXT NOT NULL", "UNIQUE(FULLPKGPATH, N, VALUE)");
$inserter->prepare_normal_inserter($self->table, "N", "VALUE");
$self->SUPER::create_table($inserter);
}
sub table()
{
return "MasterSites";
}
# Generic handling for any blank-separated list
package ListVar;
our @ISA=(qw(SecondaryVar));
sub add
{
my ($self, $ins) = @_;
$self->AnyVar::add($ins);
for my $d (main::words $self->value) {
$self->add_value($ins, $d) if $d ne '';
}
}
package ListKeyVar;
our @ISA=(qw(SecondaryVar));
sub add
{
my ($self, $ins) = @_;
$self->AnyVar::add($ins);
for my $d (main::words $self->value) {
$self->add_keyword($ins, $d) if $d ne '';
}
}
package QuotedListVar;
our @ISA=(qw(SecondaryVar));
sub add
{
my ($self, $ins) = @_;
$self->AnyVar::add($ins);
my @l = (main::words $self->value);
while (my $v = shift @l) {
while ($v =~ m/^[^']*\'[^']*$/ || $v =~m/^[^"]*\"[^"]*$/) {
$v.=' '.shift @l;
}
if ($v =~ m/^\"(.*)\"$/) {
$v = $1;
}
if ($v =~ m/^\'(.*)\'$/) {
$v = $1;
}
$self->add_value($ins, $v) if $v ne '';
}
}
package DefinedListKeyVar;
our @ISA=(qw(ListKeyVar));
sub add
{
my ($self, $ins) = @_;
return if $self->value eq '';
$self->SUPER::add($ins);
}
package FlavorsVar;
our @ISA=(qw(DefinedListKeyVar));
sub table() { 'Flavors' }
package PseudoFlavorsVar;
our @ISA=(qw(DefinedListKeyVar));
sub table() { 'PseudoFlavors' }
package ArchListVar;
our @ISA=(qw(DefinedListKeyVar));
sub keyword_table() { 'Arch' }
package OnlyForArchListVar;
our @ISA=(qw(ArchListVar));
sub table() { 'OnlyForArch' }
package NotForArchListVar;
our @ISA=(qw(ArchListVar));
sub table() { 'NotForArch' }
package CategoriesVar;
our @ISA=(qw(ListKeyVar));
sub table() { 'Categories' }
sub keyword_table() { 'CategoryKeys' }
package MultiVar;
our @ISA=(qw(ListVar));
sub table() { 'Multi' }
sub add
{
my ($self, $ins) = @_;
return if $self->value eq '-';
$self->SUPER::add($ins);
}
package ModulesVar;
our @ISA=(qw(ListKeyVar));
sub table() { 'Modules' }
sub keyword_table() { 'ModuleKeys' }
package ConfigureVar;
our @ISA=(qw(DefinedListKeyVar));
sub table() { 'Configure' }
sub keyword_table() { 'ConfigureKeys' }
package ConfigureArgsVar;
our @ISA=(qw(QuotedListVar));
sub table() { 'ConfigureArgs' }
package WantlibVar;
our @ISA=(qw(ListVar));
sub table() { 'Wantlib' }
sub _add
{
my ($self, $ins, $value, $extra) = @_;
$self->normal_insert($ins, $self->keyword($ins, $value), $extra);
}
sub add_value
{
my ($self, $ins, $value) = @_;
if ($value =~ m/^(.*)(\.\>?\=\d+\.\d+)$/) {
$self->_add($ins, $1, $2);
} elsif ($value =~ m/^(.*)(\.\>?\=\d+)$/) {
$self->_add($ins, $1, $2);
} else {
$self->_add($ins, $value, undef);
}
}
sub create_table
{
my ($self, $inserter) = @_;
$inserter->new_table($self->table, $inserter->pathref,
"VALUE TEXT NOT NULL", "EXTRA TEXT", "UNIQUE(FULLPKGPATH, VALUE)");
KeyVar::create_table($self, $inserter);
$inserter->prepare_normal_inserter($self->table, "VALUE", "EXTRA");
}
sub keyword_table() { 'Library' }
package OnlyForArchVar;
our @ISA=(qw(DefinedListKeyVar));
sub table() { 'OnlyForArch' }
sub keyword_table() { 'Arches' }
package FileVar;
our @ISA=(qw(SecondaryVar));
sub add
{
my ($self, $ins) = @_;
$self->AnyVar::add($ins);
open my $file, '<', $self->value or return;
local $/ = undef;
$self->add_value($ins, <$file>);
}
sub table() { 'Descr' }
package SharedLibsVar;
our @ISA=(qw(KeyVar));
sub add
{
my ($self, $ins) = @_;
$self->AnyVar::add($ins);
my %t = main::words($self->value);
while (my ($k, $v) = each %t) {
$self->normal_insert($ins, $self->keyword($ins, $k), $v);
}
}
sub create_table
{
my ($self, $inserter) = @_;
$inserter->new_table("Shared_Libs", $inserter->pathref,
"LIBNAME TEXT NOT NULL", "VERSION TEXT NOT NULL",
"UNIQUE (FULLPKGPATH, LIBNAME)");
$inserter->prepare_normal_inserter($self->table, "LIBNAME", "VERSION");
$self->SUPER::create_table($inserter);
}
sub table()
{
"Shared_Libs"
}
sub keyword_table()
{
return "Library";
}
package EmailVar;
our @ISA=(qw(KeyVar));
sub keyword_table()
{
return "Email";
}
package YesKeyVar;
our @ISA=(qw(KeyVar));
sub keyword_table()
{
return "Keywords2";
}
package AutoVersionVar;
our @ISA=(qw(OptKeyVar));
sub keyword_table()
{
return "AutoVersion";
}
package main;
use DBI;
getopts('v');
my $dbname;
if (@ARGV > 0) {
$dbname = shift;
} else {
$dbname = 'sqlports';
}
my @inserters;
my $db =DBI->connect("dbi:SQLite:dbname=$dbname", '', '', {AutoCommit => 0});
my $db2 =DBI->connect("dbi:SQLite:dbname=$dbname-compact", '', '',
{AutoCommit => 0});
push(@inserters, NormalInserter->new($db, 1000));
push(@inserters, CompactInserter->new($db2, 1000));
my $vars = {
AUTOCONF_VERSION => 'AutoVersionVar',
AUTOMAKE_VERSION => 'AutoVersionVar',
BROKEN => 'AnyVar',
BUILD_DEPENDS => 'BuildDependsVar',
CATEGORIES=> 'CategoriesVar',
COMMENT => 'AnyVar',
CONFIGURE_ARGS => 'ConfigureArgsVar',
CONFIGURE_STYLE => 'ConfigureVar',
DESCR => 'FileVar',
DISTFILES => 'AnyVar',
DISTNAME => 'AnyVar',
DIST_SUBDIR => 'DefinedVar',
FLAVORS => 'FlavorsVar',
FULLPKGNAME => 'AnyVar',
HOMEPAGE => 'AnyVar',
IS_INTERACTIVE => 'AnyVar',
LIB_DEPENDS => 'LibDependsVar',
MAINTAINER=> 'EmailVar',
MASTER_SITES => 'MasterSitesVar',
MASTER_SITES0 => 'MasterSitesVar',
MASTER_SITES1 => 'MasterSitesVar',
MASTER_SITES2 => 'MasterSitesVar',
MASTER_SITES3 => 'MasterSitesVar',
MASTER_SITES4=> 'MasterSitesVar',
MASTER_SITES5 => 'MasterSitesVar',
MASTER_SITES6 => 'MasterSitesVar',
MASTER_SITES7 => 'MasterSitesVar',
MASTER_SITES8 => 'MasterSitesVar',
MASTER_SITES9=> 'MasterSitesVar',
MODULES => 'ModulesVar',
MULTI_PACKAGES => 'MultiVar',
NO_BUILD => 'YesNoVar',
NO_REGRESS => 'YesNoVar',
NOT_FOR_ARCHS => 'NotForArchListVar',
ONLY_FOR_ARCHS => 'OnlyForArchListVar',
PERMIT_DISTFILES_CDROM => 'YesKeyVar',
PERMIT_DISTFILES_FTP=> 'YesKeyVar',
PERMIT_PACKAGE_CDROM => 'YesKeyVar',
PERMIT_PACKAGE_FTP=> 'YesKeyVar',
PKGNAME => 'AnyVar',
PKG_ARCH => 'ArchKeyVar',
PSEUDO_FLAVORS => 'PseudoFlavorsVar',
REGRESS_DEPENDS => 'RegressDependsVar',
REGRESS_IS_INTERACTIVE => 'AnyVar',
RUN_DEPENDS => 'RunDependsVar',
SEPARATE_BUILD => 'YesKeyVar',
SHARED_LIBS => 'SharedLibsVar',
SHARED_ONLY => 'YesNoVar',
SUBPACKAGE => 'DefinedVar',
SUPDISTFILES => 'AnyVar',
USE_GMAKE => 'YesNoVar',
USE_LIBTOOL => 'YesNoVar',
USE_MOTIF => 'YesNoVar',
USE_X11 => 'YesNoVar',
WANTLIB => 'WantlibVar',
};
for my $inserter (@inserters) {
$inserter->create_tables($vars);
}
my $lastkey;
while (<STDIN>) {
chomp;
# kill noise
if (m/^\=\=\=/) {
print "---", $_, "\n";
next;
}
next unless m/^(.*?)\.([A-Z][A-Z_0-9]*)\=(.*)$/;
my ($key, $var, $value) = ($1, $2, $3);
# strip extra quotes
if ($value =~ m/^\"(.*)\"$/) {
$value = $1;
}
if (!(defined $lastkey) || $key ne $lastkey) {
if (defined $lastkey) {
for my $inserter (@inserters) {
$inserter->finish_port;
}
}
for my $inserter (@inserters) {
$inserter->set_newkey($key);
}
$lastkey = $key;
}
my $v = $vars->{$var}->new($var, $value);
for my $inserter (@inserters) {
$v->add($inserter);
}
}
for my $inserter (@inserters) {
$inserter->finish_port;
$inserter->db->commit;
}