source: web/trunk/lib/Sophie/Base/RpmsPath.pm @ 5

Last change on this file since 5 was 5, checked in by nanardon, 14 years ago
  • reverse sort of rpms to add new rpm before removing old
  • trap error
File size: 4.2 KB
Line 
1package Sophie::Base::RpmsPath;
2
3use strict;
4use warnings;
5use base qw(Sophie::Base);
6use Sophie::Base::Header;
7use RPM4;
8use File::Temp;
9use File::Copy;
10use Archive::Cpio;
11use Encode::Guess;
12use Encode;
13
14sub new {
15    my ($class, $pathkey) = @_;
16
17    bless(\$pathkey, $class);
18}
19
20sub path {
21    my ($self) = @_;
22   
23    my $sth = $self->db->prepare_cached(
24        q{select path from d_path where d_path_key = ?}
25    );
26    $sth->execute($$self);
27    my $res = $sth->fetchrow_hashref;
28    $sth->finish;
29    return $res->{path}
30}
31
32sub ls_rpms {
33    my ($self) = @_;
34
35    my $sth = $self->db->prepare_cached(
36        q{select * from rpmfiles where d_path = ?}
37    );
38    $sth->execute($$self);
39    $sth->fetchall_hashref([ 'filename' ]);
40}
41
42sub local_ls_rpms {
43    my ($self) = @_;
44
45    if (opendir(my $dh, $self->path)) {
46        my %list;
47        while (my $entry = readdir($dh)) {
48            $entry eq '.' and next;
49            $entry eq '..' and next;
50            $list{$entry} = 1;
51        }
52        closedir($dh);
53        return \%list;
54    } else {
55        return;
56    }
57}
58
59sub update_content {
60    my ($self) = @_;
61
62    warn $self->path;
63
64    my $localrpms = $self->local_ls_rpms || {};
65    my $baserpms  = $self->ls_rpms;
66
67    my %list;
68    foreach (keys %{ $localrpms }, keys %{ $baserpms }) {
69        $list{$_} = 1;
70    }
71
72    foreach my $rpm (sort { $b cmp $a } keys %list) {
73        if ($localrpms->{$rpm} && $baserpms->{$rpm}) {
74            # nothing to do
75        } elsif ($localrpms->{$rpm}) {
76            warn "adding $rpm";
77            $self->add_rpm($rpm);
78        } elsif ($baserpms->{$rpm}) {
79            my $remove = $self->db->prepare_cached(
80                q{
81                DELETE FROM rpmfiles where d_path = ? and filename = ?
82                }
83            );
84            $remove->execute($$self, $rpm);
85            warn "deleting $rpm";
86        }
87    }
88
89}
90
91sub add_rpm {
92    my ($self, $rpm) = @_;
93
94    if (my $pkgid = $self->_add_header($rpm)) {
95        my $register = $self->db->prepare_cached(
96            q{
97            INSERT INTO rpmfiles (d_path, filename, pkgid)
98            values (?,?,?)
99            }
100        );
101        $register->execute($$self, $rpm, $pkgid);
102
103    } else {
104    }
105    $self->db->commit;
106}
107
108sub _add_header {
109    my ($self, $rpm) = @_;
110
111    my $header;
112    eval {
113        $header = RPM4::Header->new($self->path . '/' . $rpm) 
114    };
115    $header or do {
116        warn "Cannot read " . $self->path . '/' . $rpm;
117        return;
118    };
119
120    {
121        my $find = $self->db->prepare_cached(q{
122            select pkgid from rpms where pkgid = ?
123        });
124        $find->execute($header->queryformat('%{PKGID}'));
125        my $rows = $find->rows;
126        $find->finish;
127        if ($rows) {
128            warn "Find";
129            return $header->queryformat('%{PKGID}');
130        }
131    }
132    my $tmp = File::Temp->new( UNLINK => 1, SUFFIX => '.hdr' );
133    unlink($tmp->filename);
134    $header->write($tmp, 0);
135    seek($tmp, 0, 0);
136    my $string = '';
137    while (read($tmp, my $str, 1024)) { $string .= $str }
138    $tmp = undef;
139    my $add_header = $self->db->prepare_cached(
140        q{
141        INSERT into rpms (pkgid, header, evr, issrc, description, summary)
142        values (?,rpmheader_in(decode(?, 'hex')::bytea),?,?,?,?)
143        }
144    );
145    my $description = $header->queryformat('%{DESCRIPTION}');
146    {
147        my $enc = guess_encoding($description, qw/latin1/);
148        $description = $enc->decode($description) if ($enc && ref $enc);
149    }
150    my $summary = $header->queryformat('%{SUMMARY}');
151    {
152        my $enc = guess_encoding($summary, qw/latin1/);
153        $summary = $enc->decode($summary) if ($enc && ref $enc);
154    }
155
156    $add_header->execute(
157        $header->queryformat('%{PKGID}'),
158        unpack('H*', $string),
159        $header->queryformat('%|EPOCH?{%{EPOCH}:}:{}|%{VERSION}-%{RELEASE}'),
160        $header->hastag('SOURCERPM') ? 'f' : 't',
161        $description,
162        $summary,
163    );
164    my $index_tag = $self->db->prepare_cached(
165        q{
166        select index_rpms(?);
167        }
168    );
169    $index_tag->execute($header->queryformat('%{PKGID}'));
170    $index_tag->finish;
171    Sophie::Base::Header->new($header->queryformat('%{PKGID}'))
172        ->addfiles_content({ path => $self->path, filename => $rpm});
173
174    $header->queryformat('%{PKGID}');
175}
176
1771;
Note: See TracBrowser for help on using the repository browser.