File Coverage

blib/lib/DBM/Deep/Sector/File/Scalar.pm
Criterion Covered Total %
statement 71 73 97.2
branch 12 14 85.7
condition n/a
subroutine 11 11 100.0
pod 0 4 0.0
total 94 102 92.1


line stmt bran cond sub pod time code
1             package DBM::Deep::Sector::File::Scalar;
2              
3 50     50   928 use 5.008_004;
  50         176  
4              
5 50     50   294 use strict;
  50         112  
  50         1670  
6 50     50   268 use warnings FATAL => 'all';
  50         104  
  50         1911  
7 50     50   298 no warnings 'recursion';
  50         109  
  50         2363  
8              
9 50     50   345 use base qw( DBM::Deep::Sector::File::Data );
  50         105  
  50         12802  
10              
11             my $STALE_SIZE = 2;
12              
13             # Please refer to the pack() documentation for further information
14             my %StP = (
15             1 => 'C', # Unsigned char value (no order needed as it's just one byte)
16             2 => 'n', # Unsigned short in "network" (big-endian) order
17             4 => 'N', # Unsigned long in "network" (big-endian) order
18             8 => 'Q', # Usigned quad (no order specified, presumably machine-dependent)
19             );
20              
21             sub free {
22 382     382 0 709 my $self = shift;
23              
24 382         856 my $chain_loc = $self->chain_loc;
25              
26 382         1828 $self->SUPER::free();
27              
28 382 50       936 if ( $chain_loc ) {
29 0         0 $self->engine->load_sector( $chain_loc )->free;
30             }
31              
32 382         994 return;
33             }
34              
35             sub _init {
36 6346     6346   10048 my $self = shift;
37              
38 6346         13933 my $engine = $self->engine;
39              
40 6346 100       12981 unless ( $self->offset ) {
41 2179         5503 my $data_section = $self->size - $self->base_size - $engine->byte_size - 1;
42              
43 2179         5520 $self->{offset} = $engine->_request_data_sector( $self->size );
44              
45 2179         5774 my $data = delete $self->{data};
46 50     50   424 my $utf8 = do { no warnings 'utf8'; $data !~ /^[\0-\xff]*\z/ };
  50         130  
  50         28964  
  2179         3508  
  2179         13276  
47 2179 100       5157 if($utf8){
48 7 50       37 if($engine->{v} < 4) {
49 0         0 DBM::Deep->_throw_error(
50             "This database format version is too old for Unicode"
51             );
52             }
53 7         25 utf8::encode $data;
54 7         20 $self->{type} = $engine->SIG_UNIDATA;
55             }
56 2172         6875 else { $self->{type} = $engine->SIG_DATA; }
57              
58 2179         3511 my $dlen = length $data;
59 2179         3029 my $continue = 1;
60 2179         5736 my $curr_offset = $self->offset;
61 2179         5208 while ( $continue ) {
62              
63 4695         6860 my $next_offset = 0;
64              
65 4695         7390 my ($leftover, $this_len, $chunk);
66 4695 100       8796 if ( $dlen > $data_section ) {
67 2516         3258 $leftover = 0;
68 2516         3678 $this_len = $data_section;
69 2516         4945 $chunk = substr( $data, 0, $this_len );
70              
71 2516         3433 $dlen -= $data_section;
72 2516         7184 $next_offset = $engine->_request_data_sector( $self->size );
73 2516         18585 $data = substr( $data, $this_len );
74             }
75             else {
76 2179         3370 $leftover = $data_section - $dlen;
77 2179         3824 $this_len = $dlen;
78 2179         3197 $chunk = $data;
79              
80 2179         3388 $continue = 0;
81             }
82              
83 4695         12964 $engine->storage->print_at( $curr_offset, $self->type ); # Sector type
84             # Skip staleness
85             $engine->storage->print_at( $curr_offset + $self->base_size,
86             pack( $StP{$engine->byte_size}, $next_offset ), # Chain loc
87 4695         14974 pack( $StP{1}, $this_len ), # Data length
88             $chunk, # Data to be stored in this sector
89             chr(0) x $leftover, # Zero-fill the rest
90             );
91              
92 4695         14931 $curr_offset = $next_offset;
93             }
94              
95 2179         5635 return;
96             }
97             }
98              
99             sub data_length {
100 3733     3733 0 6011 my $self = shift;
101              
102 3733         7929 my $buffer = $self->engine->storage->read_at(
103             $self->offset + $self->base_size + $self->engine->byte_size, 1
104             );
105              
106 3733         16532 return unpack( $StP{1}, $buffer );
107             }
108              
109             sub chain_loc {
110 4115     4115 0 6086 my $self = shift;
111             return unpack(
112 4115         8050 $StP{$self->engine->byte_size},
113             $self->engine->storage->read_at(
114             $self->offset + $self->base_size,
115             $self->engine->byte_size,
116             ),
117             );
118             }
119              
120             sub data {
121 1200     1200 0 1910 my $self = shift;
122 1200         2358 my $engine = $self->engine;
123              
124 1200         1871 my $data;
125 1200         2007 while ( 1 ) {
126 3733         7495 my $chain_loc = $self->chain_loc;
127              
128 3733         11600 $data .= $engine->storage->read_at(
129             $self->offset + $self->base_size + $engine->byte_size + 1, $self->data_length,
130             );
131              
132 3733 100       11691 last unless $chain_loc;
133              
134 2533         7208 $self = $engine->load_sector( $chain_loc );
135             }
136              
137 1200 100       3436 utf8::decode $data if $self->type eq $engine->SIG_UNIDATA;
138              
139 1200         5671 return $data;
140             }
141              
142             1;
143             __END__