File Coverage

blib/lib/Biblio/Zotero/DB.pm
Criterion Covered Total %
statement 25 27 92.5
branch n/a
condition n/a
subroutine 9 9 100.0
pod n/a
total 34 36 94.4


line stmt bran cond sub pod time code
1             package Biblio::Zotero::DB;
2             # ABSTRACT: helper module to access the Zotero SQLite database
3             $Biblio::Zotero::DB::VERSION = '0.004';
4 11     11   96957 use strict;
  11         18  
  11         446  
5 11     11   51 use warnings;
  11         15  
  11         391  
6              
7 11     11   136 use v5.14;
  11         33  
  11         474  
8 11     11   6211 use Moo;
  11         171907  
  11         79  
9 11     11   797291 use File::HomeDir;
  11         55922  
  11         783  
10 11     11   1469 use Path::Class;
  11         143594  
  11         680  
11 11     11   17835 use Path::Iterator::Rule;
  11         137551  
  11         490  
12 11     11   6685 use List::AllUtils qw(first);
  11         23731  
  11         890  
13              
14 11     11   4802 use Biblio::Zotero::DB::Schema;
  0            
  0            
15             use Biblio::Zotero::DB::Library;
16              
17              
18             # used for L and L attr
19             my $make_directory_absolute = sub {
20             my $orig = shift;
21             my $self = $_[0];
22             my $dir = $_[1];
23              
24             return $orig->($self, dir($dir)->absolute) if($dir);
25              
26             return $orig->(@_);
27             };
28              
29             has schema => ( is => 'rw', builder => 1, lazy => 1, clearer => 1 );
30              
31             sub _build_schema {
32             my ($self) = @_;
33             Biblio::Zotero::DB::Schema->connect('dbi:SQLite:dbname='.$self->db_file,
34             '', '',
35             {
36             (zotero_storage_directory => $self->storage_directory)x!! $self->storage_directory
37             },
38             );
39             }
40              
41             has db_file => ( is => 'rw', builder => 1, lazy => 1 );
42              
43             sub _build_db_file {
44             my ($self) = @_;
45             dir($self->profile_directory)->file('zotero.sqlite');
46             }
47              
48             has storage_directory => ( is => 'rw', builder => 1, lazy => 1 );
49              
50             sub _build_storage_directory {
51             my ($self) = @_;
52             dir($self->profile_directory)->subdir('storage')->absolute if $self->profile_directory;
53             }
54              
55             around storage_directory => $make_directory_absolute;
56              
57             has profile_directory => ( is => 'rw' );
58              
59             around profile_directory => $make_directory_absolute;
60              
61             has profile_name => ( is => 'rw', trigger => 1, builder => 1, lazy => 1 );
62              
63             sub _trigger_profile_name {
64             my ($self) = @_;
65             $self->profile_directory(
66             first { dir($_)->components(-2) eq $self->profile_name }
67             @{$self->find_profile_directories});
68             }
69              
70             sub _build_profile_name {
71             my ($self) = @_;
72             dir($self->profile_directory)->components(-2) if $self->profile_directory;
73             }
74              
75              
76             # From
77             # Zotero for Firefox
78             #
79             # OS X /Users//Library/Application Support/Firefox/Profiles//zotero
80             # Windows 7/Vista C:\Users\\AppData\Roaming\Mozilla\Firefox\Profiles\\zotero
81             # Windows XP/2000 C:\Documents and Settings\\Application Data\Mozilla\Firefox\Profiles\\zotero
82             # Linux (most distributions) ~/.mozilla/firefox/Profiles//zotero
83             #
84             ####
85             #
86             # Zotero Standalone
87             #
88             # OS X /Users//Library/Application Support/Zotero/Profiles//zotero
89             # Windows 7/Vista C:\Users\\AppData\Roaming\Zotero\Profiles\\zotero
90             # Windows XP/2000 C:\Documents and Settings\\Application Data\Zotero\Profiles\\zotero
91             # Linux (most distributions) ~/.zotero/Profiles//zotero
92              
93              
94             sub find_profile_directories {
95             my ($self) = @_;
96             for($^O) {
97             return $self->_find_profile_directories_linux if($_ eq 'linux');
98             return $self->_find_profile_directories_osx if($_ eq 'darwin');
99             return $self->_find_profile_directories_win if($_ eq 'MSWin32');
100             return [];
101             }
102             }
103              
104             sub _find_profile_directories_osx {
105             my ($self) = @_;
106             my $my_data = dir(File::HomeDir->my_data)->absolute;
107             # gives /Users//Library/Application Support
108             my $find = [
109             dir($my_data, 'Firefox'),
110             dir($my_data, 'Zotero') ];
111             return $self->_find_profile_directories_under($find);
112             }
113              
114             sub _find_profile_directories_win {
115             my ($self) = @_;
116             my $my_data = dir(File::HomeDir->my_data)->absolute;
117             # gives C:\Users\\AppData\Local
118             # or C:\Documents and Settings\\Local Settings\Application Data
119             # depending on OS version
120             if( $my_data->components(-1) eq "Local" ) {
121             # for Windows 7 / Vista
122             # this returns the \Local directory
123             # we want the \Roaming directory that is on the same level
124             $my_data = $my_data->parent->subdir('Roaming');
125             }
126             my $find = [
127             dir( $my_data, 'Mozilla','Firefox'),
128             dir( $my_data, 'Zotero') ];
129             return $self->_find_profile_directories_under($find);
130             }
131              
132             sub _find_profile_directories_linux {
133             my ($self) = @_;
134             my $find = [
135             dir(File::HomeDir->my_home , '.mozilla/firefox'),
136             dir(File::HomeDir->my_home , '.zotero/zotero') ];
137             return $self->_find_profile_directories_under($find);
138             }
139              
140             sub _find_profile_directories_under {
141             my ($self, $dirs) = @_;
142             # finds either
143             # dir('Profiles',,'zotero')
144             # (actually, currently it doesn't check that it is indeed in the Profiles
145             # directory)
146             # or
147             # dir(, 'zotero')
148             return [ Path::Iterator::Rule->new
149             ->min_depth(1)->max_depth(3)
150             ->dir->name('zotero')->all( @$dirs ) ];
151             }
152              
153             sub library {
154             my $self = shift;
155             return Biblio::Zotero::DB::Library->new( _db => $self );
156             }
157              
158             1;
159              
160             __END__