Thread Textdatenbank - Erweiterung des Codes .. (16 answers)
Opened by Yagyu at 2011-02-10 04:56

topeg
 2011-02-10 11:13
#145552 #145552
User since
2006-07-10
2611 Artikel
BenutzerIn

user image
Eine Klasse die deinen Anforderungen für die DB entspricht:

more (55.1kb):
Code (perl): (dl )
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
use strict;
use warnings;
use utf8;

# Namen und Position der Werte in der DB-Datei
my @options=qw(name pass rights mail active);

# Datenbank zur Verwaltung von Nutzern
package mydb;

our %update;

# Objekt erzeugen
sub new
{
  my $class=shift;
  my $file=shift;
  return undef unless($file);
  my $self={};
  $self->{file}  =$file;
  $self->{table} ={};
  $self->{last}  =0;
  bless($self,ref($class)||$class);
  $self->load;
  return $self;
}

# wenn Objekt zerstört, dann speichern
sub DESTROY
{ shift->save(); }

# db laden
sub load
{
  my $self=shift;
  $self->{update}=1;
  return 0 unless(-f $self->{file});
  if(open(my $fh, '<', $self->{file}))
  {
    # zeilenweise lesen.
    while(<$fh>)
    {
      chomp;
      next unless($_);
      my @row=split(/#/,$_);
      my %data;
      for my $key (@options)
      {
        last unless(@row);
        my $val=shift(@row);
        $val=__unquote($val) if(defined($val));
        $data{$key}=$val;
      }

      # user existiert schon
      if($self->{table}->{$data{name}})
      {
        # Benutzerdaten aktualisieren
        $self->{table}->{$data{name}}->update(\%data);
      }
      # neuen User erzeugen
      else
      {
        # neues user-objekt erzeugen wenn es noch nicht existiert
        my $user=mydb::user->new(\%data,"$self");
        $self->{table}->{$user->name()}=$user if($user);
      }
    }
    close($fh);
    $self->{update}=0;
    $self->{last}=-M $self->{file};
  }
}

sub save
{
  my $self=shift;
  if($update{"$self"} && %{$self->{table}})
  {
    my $str='';
    while(my ($n,$o)=each(%{$self->{table}}))
    { $str.= $o->to_string() if($o); }

    $self->_update();
    if(open(my $fh, '>', $self->{file}))
    {
      print $fh $str;
      close($fh);
      delete($update{"$self"});
      return 1;
    }
    return 0
  }
  return 1;
}

sub del
{
  my $self=shift;
  my $name=shift;
  $self->_update();
  return 0 unless($name && exists($self->{table}->{$name}));
  $self->{table}->{$name}->DESTROY();
  delete($self->{table}->{$name});
  $update{"$self"}=1;
  return 1;
}

sub has{ return exists($_[0]->{table}->{$_[1]}); }

# user Objekt setzen/holen
sub user
{
  my $self=shift;
  return undef unless($_[0]);

  my $user;

  # es ist ein User-Objekt
  if(ref($_[0]) eq 'mydb::user')
  {
    $user=shift();
    if(!$self->{table}->{$user->name()})
    {
      $user->{parent}="$self";
      $self->{table}->{$user->name()}=$user;
      $update{"$self"}=1;
    }
  }
  # es ist ein Hash mit den Werten des neuen Nutzers
  elsif(ref($_[0]) eq 'HASH')
  {
    if(!$self->{table}->{$_[0]->{name}})
    {
      $user=mydb::user->new(shift(),"$self");
      if($user)
      {
        $self->{table}->{$user->name()}=$user;
        $update{"$self"}=1;
      }
    }
  }
  # es ist ein String
  # also wird ein Nutzer verlangt
  else
  {
    my $name=shift();
    $user=$self->{table}->{$name} if($name)
  }

  return $user;
}

sub user_name_list { return keys(%{shift()->{table}}); }
sub user_list      { return values(%{shift()->{table}}); }

########################################################################
sub _update
{
  my $self=shift;
  return $self->load() if(-f $self->{file} && (!$self->{last} || -M $self->{file} != $self->{last}));
  return 1;
}

########################################################################
sub __unquote
{
  my $str=shift;
  $str=~s/\Q&raute;/#/g;
  return $str;
}

########################################################################
########################################################################
########################################################################

# das Paket mydb::user ist die Klasse,
# mit der die Daten zu einem Nutzer verwaltet werden
package mydb::user;
use Digest::MD5 qw(md5_hex);

# neues Objekt erzeugen.
# es muss eine Hashrefenzen mit den Nutzerdaten übergeben werden
sub new
{
  my $class=shift;
  my $data=shift;
  my $parent=shift;

  my $self={};
  $self->{parent}=$parent;

  bless($self, $class);

  return undef unless($self->update($data));

  return $self;
}

# werte aktualisieren
sub update
{
  my $self=shift;
  my $data=shift;

  # richtiges Refenzformat?
  return 0 unless(ref($data) eq 'HASH');
  # alle Werte vorhanden?
  return 0 unless(grep{defined($data->{$_})}@options);

  # werte neu setzten
  $self->{$_}=$data->{$_} for(@options);

  return 1;
}

# allgemeiner getter/setter
sub _value
{
  my $self=shift;

  #$self->{parent}->_update() if($self->{parent});

  my $name=shift;
  return undef unless(grep{ $name eq $_ }@options);

  my $val=shift;
  if(defined($val))
  {
    $self->{$name}=$val;
    $update{$self->{parent}}=1 if($self->{parent});
  }

  return $self->{$name};
}

# namen lesen
sub name  { shift()->{name}; }
# rechte setzen/lesen
sub rights{ shift->_value('rights',@_); }
# mail setzen/lesen
sub mail  { shift->_value('mail',@_); }
# status setzen/lesen
sub active{ shift->_value('active',@_); }
# passwort setzen/lesen
sub pass
{
  my $self=shift;
  my $pass=shift;
  # passwort verschlüsseln wenn es gesetzt wird.
  $pass=md5_hex($pass) if($pass);
  $self->_value('pass', $pass);
}

# passwort prüfen
sub chk
{
  my $self=shift;
  my $pass=shift;
  return 0 unless($pass && $self->pass() eq md5_hex($pass));
  return 1;
}

# Werte serialisieren.
sub to_string
{
  my $self=shift;
  return join('#',map{ __quote($self->{$_}) }@options)."\n";
}

# string quoten
# wird zum serialisieren gebraucht.
sub __quote
{
  my $str=shift;
  $str=~s/#/&raute;/g;
  return $str;
}

1;


Beispiel er Nutzung:
Code (perl): (dl )
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
#!/usr/bin/perl;
use strict;
use warnings;
use mydb;

my $db=mydb->new('test.db');

$db->user({name=>'test1',pass=>'12345',mail=>'test1@example.org', active=>1, rights=>'rwx'});
$db->user({name=>'test2',pass=>'12345',mail=>'test2@example.org', active=>1, rights=>'rwx'});
$db->user({name=>'test3',pass=>'12345',mail=>'test3@example.org', active=>1, rights=>'rwx'});
$db->user({name=>'test4',pass=>'12345',mail=>'test4@example.org', active=>1, rights=>'rwx'});

$db->user('test1')->pass('test1');
$db->user('test2')->pass('test2');
$db->user('test3')->pass('test3');
$db->user('test4')->pass('test4');

for(qw(test1 test2 test3 test4))
{
  my $user=$db->user($_);
  print "$_ PASS OK\n" if($user->chk($_));
}

for my $user ($db->user_list)
{
  print $user->name."\n";
  print "  RIGHTS: ".$user->rights."\n";
  print "  MAIL  : ".$user->mail."\n";
  print "  ACTIVE: ".$user->active."\n";
  print "\n";
}

$db->del('test1') if($db->has('test1'));

View full thread Textdatenbank - Erweiterung des Codes ..