our
$VERSION
=
'1.1.0'
;
use
5.14.2;
sub
from_config {
my
(
$class
,
$config
) =
@_
;
my
$file
=
$config
->SQLITE_database_file;
my
$data_source_name
=
"DBI:SQLite:dbname=$file"
;
return
$class
->new(
{
data_source_name
=>
$data_source_name
,
user
=>
''
,
password
=>
''
,
dbhandle
=>
undef
,
}
);
}
sub
DEMOLISH {
my
(
$self
) =
@_
;
$self
->dbh->disconnect()
if
defined
$self
->dbhandle &&
$self
->dbhandle->ping;
}
sub
get_dbh_specific_attributes {
return
{
sqlite_extended_result_codes
=> 1 };
}
sub
create_schema {
my
(
$self
) =
@_
;
my
$dbh
=
$self
->dbh;
$dbh
->
do
(
'PRAGMA foreign_keys = ON;'
);
$dbh
->
do
(
'CREATE TABLE IF NOT EXISTS test_results (
id integer PRIMARY KEY AUTOINCREMENT,
hash_id VARCHAR(16) NOT NULL,
domain VARCHAR(255) NOT NULL,
batch_id integer NULL,
created_at DATETIME NOT NULL,
started_at DATETIME DEFAULT NULL,
ended_at DATETIME DEFAULT NULL,
priority integer DEFAULT 10,
queue integer DEFAULT 0,
progress integer DEFAULT 0,
fingerprint character varying(32),
params text NOT NULL,
results text DEFAULT NULL,
undelegated boolean NOT NULL DEFAULT false,
UNIQUE (hash_id)
)
'
) or
die
Zonemaster::Backend::Error::Internal->new(
reason
=>
"SQLite error, could not create 'test_results' table"
,
data
=>
$dbh
->errstr() );
$dbh
->
do
(
'CREATE INDEX IF NOT EXISTS test_results__hash_id ON test_results (hash_id)'
);
$self
->dbh->
do
(
'CREATE INDEX IF NOT EXISTS test_results__fingerprint ON test_results (fingerprint)'
);
$dbh
->
do
(
'CREATE INDEX IF NOT EXISTS test_results__batch_id_progress ON test_results (batch_id, progress)'
);
$dbh
->
do
(
'CREATE INDEX IF NOT EXISTS test_results__progress ON test_results (progress)'
);
$dbh
->
do
(
'CREATE INDEX IF NOT EXISTS test_results__domain_undelegated ON test_results (domain, undelegated)'
);
$dbh
->
do
(
"CREATE TABLE IF NOT EXISTS log_level (
value INTEGER,
level VARCHAR(15),
UNIQUE (value)
)
"
) or
die
Zonemaster::Backend::Error::Internal->new(
reason
=>
"SQLite error, could not create 'log_level' table"
,
data
=>
$dbh
->errstr() );
my
(
$c
) =
$dbh
->selectrow_array(
"SELECT count(*) FROM log_level"
);
if
(
$c
== 0 ) {
$dbh
->
do
(
"INSERT INTO log_level (value, level)
VALUES
(-2,
'DEBUG3'
),
(-1,
'DEBUG2'
),
( 0,
'DEBUG'
),
( 1,
'INFO'
),
( 2,
'NOTICE'
),
( 3,
'WARNING'
),
( 4,
'ERROR'
),
( 5,
'CRITICAL'
)
"
);
}
$dbh
->
do
(
'CREATE TABLE IF NOT EXISTS result_entries (
hash_id VARCHAR(16) NOT NULL,
level INT NOT NULL,
module VARCHAR(255) NOT NULL,
testcase VARCHAR(255) NOT NULL,
tag VARCHAR(255) NOT NULL,
timestamp REAL NOT NULL,
args BLOB NOT NULL,
FOREIGN KEY(hash_id) REFERENCES test_results(hash_id),
FOREIGN KEY(level) REFERENCES log_level(value)
)
'
) or
die
Zonemaster::Backend::Error::Internal->new(
reason
=>
"SQLite error, could not create 'result_entries' table"
,
data
=>
$dbh
->errstr() );
$dbh
->
do
(
'CREATE INDEX IF NOT EXISTS result_entries__hash_id ON result_entries (hash_id)'
);
$dbh
->
do
(
'CREATE INDEX IF NOT EXISTS result_entries__level ON result_entries (level)'
);
$dbh
->
do
(
'CREATE TABLE IF NOT EXISTS batch_jobs (
id integer PRIMARY KEY,
username character varying(50) NOT NULL,
created_at DATETIME NOT NULL
)
'
) or
die
Zonemaster::Backend::Error::Internal->new(
reason
=>
"SQLite error, could not create 'batch_jobs' table"
,
data
=>
$dbh
->errstr() );
$dbh
->
do
(
'CREATE TABLE IF NOT EXISTS users (
id INTEGER PRIMARY KEY AUTOINCREMENT,
username varchar(128),
api_key varchar(512),
UNIQUE (username)
)
'
) or
die
Zonemaster::Backend::Error::Internal->new(
reason
=>
"SQLite error, could not create 'users' table"
,
data
=>
$dbh
->errstr() );
return
;
}
sub
drop_tables {
my
(
$self
) =
@_
;
$self
->dbh->
do
(
"DROP TABLE IF EXISTS test_results"
);
$self
->dbh->
do
(
"DROP TABLE IF EXISTS result_entries"
);
$self
->dbh->
do
(
"DROP TABLE IF EXISTS log_level"
);
$self
->dbh->
do
(
"DROP TABLE IF EXISTS users"
);
$self
->dbh->
do
(
"DROP TABLE IF EXISTS batch_jobs"
);
return
;
}
sub
add_batch_job {
my
(
$self
,
$params
) =
@_
;
my
$batch_id
;
my
$dbh
=
$self
->dbh;
if
(
$self
->user_authorized(
$params
->{username},
$params
->{api_key} ) ) {
$batch_id
=
$self
->create_new_batch_job(
$params
->{username} );
my
$test_params
=
$params
->{test_params};
my
$priority
=
$test_params
->{priority};
my
$queue_label
=
$test_params
->{queue};
$dbh
->{AutoCommit} = 0;
eval
{
$dbh
->
do
(
"DROP INDEX IF EXISTS test_results__hash_id "
);};
eval
{
$dbh
->
do
(
"DROP INDEX IF EXISTS test_results__fingerprint "
);};
eval
{
$dbh
->
do
(
"DROP INDEX IF EXISTS test_results__batch_id_progress "
);};
eval
{
$dbh
->
do
(
"DROP INDEX IF EXISTS test_results__progress "
);};
eval
{
$dbh
->
do
(
"DROP INDEX IF EXISTS test_results__domain_undelegated "
);};
my
$sth
=
$dbh
->prepare( '
INSERT INTO test_results (
hash_id,
domain,
batch_id,
created_at,
priority,
queue,
fingerprint,
params,
undelegated
) VALUES (?,?,?,?,?,?,?,?,?)'
);
foreach
my
$domain
( @{
$params
->{domains}} ) {
$test_params
->{domain} = _normalize_domain(
$domain
);
my
$fingerprint
=
$self
->generate_fingerprint(
$test_params
);
my
$encoded_params
=
$self
->encode_params(
$test_params
);
my
$undelegated
=
$self
->undelegated (
$test_params
);
my
$hash_id
=
substr
(md5_hex(
time
().
rand
()), 0, 16);
$sth
->execute(
$hash_id
,
$test_params
->{domain},
$batch_id
,
$self
->format_time(
time
() ),
$priority
,
$queue_label
,
$fingerprint
,
$encoded_params
,
$undelegated
,
);
}
$dbh
->
do
(
"CREATE INDEX test_results__hash_id ON test_results (hash_id, created_at)"
);
$dbh
->
do
(
"CREATE INDEX test_results__fingerprint ON test_results (fingerprint)"
);
$dbh
->
do
(
"CREATE INDEX test_results__batch_id_progress ON test_results (batch_id, progress)"
);
$dbh
->
do
(
"CREATE INDEX test_results__progress ON test_results (progress)"
);
$dbh
->
do
(
"CREATE INDEX test_results__domain_undelegated ON test_results (domain, undelegated)"
);
$dbh
->commit();
$dbh
->{AutoCommit} = 1;
}
else
{
die
Zonemaster::Backend::Error::PermissionDenied->new(
message
=>
'User not authorized to use batch mode'
,
data
=> {
username
=>
$params
->{username}} );
}
return
$batch_id
;
}
sub
get_relative_start_time {
my
(
$self
,
$hash_id
) =
@_
;
return
$self
->dbh->selectrow_array(
q[
SELECT (julianday(?) - julianday(started_at)) * 3600 * 24
FROM test_results
WHERE hash_id = ?
]
,
undef
,
$self
->format_time(
time
() ),
$hash_id
,
);
}
sub
is_duplicate {
my
(
$self
) =
@_
;
return
(
$self
->dbh->err == 2067 );
}
no
Moose;
__PACKAGE__->meta()->make_immutable();
1;