| line | stmt | bran | cond | sub | pod | time | code | 
| 1 |  |  |  |  |  |  | ##---------------------------------------------------------------------------- | 
| 2 |  |  |  |  |  |  | ## Apache2 Server Side Include Parser's Notes - ~/lib/Apache2/SSI/Notes.pm | 
| 3 |  |  |  |  |  |  | ## Version v0.1.0 | 
| 4 |  |  |  |  |  |  | ## Copyright(c) 2021 DEGUEST Pte. Ltd. | 
| 5 |  |  |  |  |  |  | ## Author: Jacques Deguest <jack@deguest.jp> | 
| 6 |  |  |  |  |  |  | ## Created 2021/01/18 | 
| 7 |  |  |  |  |  |  | ## Modified 2021/01/19 | 
| 8 |  |  |  |  |  |  | ## All rights reserved | 
| 9 |  |  |  |  |  |  | ## | 
| 10 |  |  |  |  |  |  | ## This program is free software; you can redistribute  it  and/or  modify  it | 
| 11 |  |  |  |  |  |  | ## under the same terms as Perl itself. | 
| 12 |  |  |  |  |  |  | ##---------------------------------------------------------------------------- | 
| 13 |  |  |  |  |  |  | package Apache2::SSI::Notes; | 
| 14 |  |  |  |  |  |  | BEGIN | 
| 15 |  |  |  |  |  |  | { | 
| 16 | 15 |  |  | 15 |  | 90829 | use strict; | 
|  | 15 |  |  |  |  | 37 |  | 
|  | 15 |  |  |  |  | 462 |  | 
| 17 | 15 |  |  | 15 |  | 69 | use warnings; | 
|  | 15 |  |  |  |  | 24 |  | 
|  | 15 |  |  |  |  | 397 |  | 
| 18 | 15 |  |  | 15 |  | 73 | use warnings::register; | 
|  | 15 |  |  |  |  | 26 |  | 
|  | 15 |  |  |  |  | 2154 |  | 
| 19 | 15 |  |  | 15 |  | 534 | use parent qw( Module::Generic ); | 
|  | 15 |  |  |  |  | 331 |  | 
|  | 15 |  |  |  |  | 124 |  | 
| 20 |  |  |  |  |  |  | ## 512Kb | 
| 21 | 15 |  |  | 15 |  | 9468243 | use constant MAX_SIZE => 524288; | 
|  | 15 |  |  |  |  | 27 |  | 
|  | 15 |  |  |  |  | 820 |  | 
| 22 | 15 |  |  | 15 |  | 6216 | use Apache2::SSI::SharedMem ':all'; | 
|  | 15 |  |  |  |  | 39 |  | 
|  | 15 |  |  |  |  | 284 |  | 
| 23 | 15 |  |  | 15 |  | 7985 | use Nice::Try; | 
|  | 15 |  |  |  |  | 31 |  | 
|  | 15 |  |  |  |  | 118 |  | 
| 24 | 15 |  |  | 15 |  | 4236381 | our $VERSION = 'v0.1.0'; | 
| 25 |  |  |  |  |  |  | }; | 
| 26 |  |  |  |  |  |  |  | 
| 27 |  |  |  |  |  |  | sub init | 
| 28 |  |  |  |  |  |  | { | 
| 29 | 63 |  |  | 63 | 1 | 1560 | my $self = shift( @_ ); | 
| 30 | 63 |  |  |  |  | 893 | $self->{key}  = 'ap2_ssi_notes'; | 
| 31 | 63 |  |  |  |  | 190 | $self->{size} = MAX_SIZE; | 
| 32 | 63 |  |  |  |  | 169 | $self->{_init_strict_use_sub} = 1; | 
| 33 | 63 |  |  |  |  | 359 | $self->SUPER::init( @_ ); | 
| 34 | 63 | 50 |  |  |  | 5597 | return( $self->error( "Notes under this system $^O are unsupported." ) ) if( !Apache2::SSI::SharedMem->supported ); | 
| 35 |  |  |  |  |  |  | my $mem = Apache2::SSI::SharedMem->new( | 
| 36 |  |  |  |  |  |  | key => ( length( $self->{key} ) ? $self->{key} : 'ap2_ssi_notes' ), | 
| 37 |  |  |  |  |  |  | ## 512 Kb max | 
| 38 |  |  |  |  |  |  | size => $self->{size}, | 
| 39 |  |  |  |  |  |  | ## Create if necessary | 
| 40 | 63 |  | 50 |  |  | 438 | create => 1, | 
| 41 |  |  |  |  |  |  | debug => $self->debug, | 
| 42 |  |  |  |  |  |  | ) || return( $self->pass_error( Apache2::SSI::SharedMem->error ) ); | 
| 43 | 63 |  | 50 |  |  | 310 | my $shem = $mem->open || return( $self->pass_error( $mem->error ) ); | 
| 44 | 63 |  |  |  |  | 342 | $self->shem( $shem ); | 
| 45 | 63 |  |  |  |  | 2501 | return( $self ); | 
| 46 |  |  |  |  |  |  | }; | 
| 47 |  |  |  |  |  |  |  | 
| 48 | 0 |  |  | 0 | 1 | 0 | sub add { return( shift->set( @_ ) ); } | 
| 49 |  |  |  |  |  |  |  | 
| 50 |  |  |  |  |  |  | sub clear | 
| 51 |  |  |  |  |  |  | { | 
| 52 | 1 |  |  | 1 | 1 | 749 | my $self = shift( @_ ); | 
| 53 | 1 |  |  |  |  | 3 | my $data = {}; | 
| 54 | 1 | 50 |  |  |  | 5 | $self->write_mem( $data ) || return; | 
| 55 | 1 |  |  |  |  | 7 | return( $self ); | 
| 56 |  |  |  |  |  |  | } | 
| 57 |  |  |  |  |  |  |  | 
| 58 |  |  |  |  |  |  | sub do | 
| 59 |  |  |  |  |  |  | { | 
| 60 | 1 |  |  | 1 | 1 | 4 | my $self = shift( @_ ); | 
| 61 | 1 |  |  |  |  | 9 | my $code = shift( @_ ); | 
| 62 | 1 |  |  |  |  | 3 | my @keys = @_; | 
| 63 | 1 | 50 |  |  |  | 7 | return( $self->error( "Code provided ($code) is not actually a code reference." ) ) if( ref( $code ) ne 'CODE' ); | 
| 64 | 1 |  | 50 |  |  | 10 | my $data = $self->read_mem || return; | 
| 65 | 1 | 50 |  |  |  | 8 | @keys = sort( keys( %$data ) ) unless( scalar( @keys ) ); | 
| 66 | 1 |  |  |  |  | 10 | foreach my $k ( @keys ) | 
| 67 |  |  |  |  |  |  | { | 
| 68 | 1 |  |  |  |  | 3 | my $k_orig = $k; | 
| 69 | 1 |  |  |  |  | 3 | my $v = $data->{ $k }; | 
| 70 | 1 |  |  |  |  | 2 | try | 
| 71 | 1 |  |  | 1 |  | 2 | { | 
| 72 |  |  |  |  |  |  | ## Code can modify values in-place like: | 
| 73 |  |  |  |  |  |  | ## sub | 
| 74 |  |  |  |  |  |  | ## { | 
| 75 |  |  |  |  |  |  | ##     $_[1] = 'new value' if( $_[0] eq 'some_key_name' ); | 
| 76 |  |  |  |  |  |  | ## } | 
| 77 | 1 |  |  |  |  | 5 | $code->( $k, $v ); | 
| 78 |  |  |  |  |  |  | ## Store possibly updated value | 
| 79 | 1 |  |  |  |  | 9 | $data->{ $k_orig } = $v; | 
| 80 |  |  |  |  |  |  | } | 
| 81 | 1 | 50 |  |  |  | 17 | catch( $e ) | 
|  | 1 | 50 |  |  |  | 3 |  | 
|  | 1 | 50 |  |  |  | 4 |  | 
|  | 1 | 0 |  |  |  | 3 |  | 
|  | 1 | 50 |  |  |  | 3 |  | 
|  | 1 |  |  |  |  | 3 |  | 
|  | 1 |  |  |  |  | 2 |  | 
|  | 1 |  |  |  |  | 2 |  | 
|  | 1 |  |  |  |  | 19 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 1 |  |  |  |  | 4 |  | 
|  | 1 |  |  |  |  | 14 |  | 
|  | 1 |  |  |  |  | 4 |  | 
|  | 1 |  |  |  |  | 4 |  | 
|  | 1 |  |  |  |  | 5 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 0 |  |  |  |  | 0 |  | 
| 82 | 0 |  |  | 0 |  | 0 | { | 
| 83 | 0 |  |  |  |  | 0 | return( $self->error( "Callback died with error: $e" ) ); | 
| 84 | 0 | 0 | 0 |  |  | 0 | } | 
|  | 0 | 0 | 33 |  |  | 0 |  | 
|  | 0 | 0 |  |  |  | 0 |  | 
|  | 0 | 0 |  |  |  | 0 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 0 |  |  |  |  | 0 |  | 
|  | 1 |  |  |  |  | 46 |  | 
|  | 0 |  |  |  |  | 0 |  | 
| 85 |  |  |  |  |  |  | } | 
| 86 |  |  |  |  |  |  | ## No need to bother if there was no keys in the first place | 
| 87 | 1 | 50 |  |  |  | 6 | if( scalar( @keys ) ) | 
| 88 |  |  |  |  |  |  | { | 
| 89 | 1 | 50 |  |  |  | 6 | $self->write_mem( $data ) || return; | 
| 90 |  |  |  |  |  |  | } | 
| 91 | 1 |  |  |  |  | 6 | return( $self ); | 
| 92 |  |  |  |  |  |  | } | 
| 93 |  |  |  |  |  |  |  | 
| 94 |  |  |  |  |  |  | sub get | 
| 95 |  |  |  |  |  |  | { | 
| 96 | 6 |  |  | 6 | 1 | 784 | my $self = shift( @_ ); | 
| 97 | 6 |  |  |  |  | 12 | my $key; | 
| 98 | 6 | 100 |  |  |  | 23 | if( @_ ) | 
| 99 |  |  |  |  |  |  | { | 
| 100 | 3 |  |  |  |  | 7 | $key = shift( @_ ); | 
| 101 | 3 | 50 |  |  |  | 18 | return( $self->error( "Key provided to retrieve is empty." ) ) if( !length( $key ) ); | 
| 102 |  |  |  |  |  |  | } | 
| 103 | 6 |  | 50 |  |  | 120 | my $data = $self->read_mem || return; | 
| 104 |  |  |  |  |  |  | ## As it is the case for the first time, before any write | 
| 105 | 6 | 50 |  |  |  | 19 | $data = {} if( !ref( $data ) ); | 
| 106 | 6 | 100 |  |  |  | 34 | return( $data ) if( !defined( $key ) ); | 
| 107 | 3 |  |  |  |  | 24 | return( $data->{ $key } ); | 
| 108 |  |  |  |  |  |  | } | 
| 109 |  |  |  |  |  |  |  | 
| 110 | 1 |  |  | 1 | 0 | 515 | sub key { return( shift->_set_get_scalar( 'key', @_ ) ); } | 
| 111 |  |  |  |  |  |  |  | 
| 112 |  |  |  |  |  |  | sub read_mem | 
| 113 |  |  |  |  |  |  | { | 
| 114 | 10 |  |  | 10 | 1 | 24 | my $self = shift( @_ ); | 
| 115 | 10 |  | 50 |  |  | 32 | my $shem = $self->shem || | 
| 116 |  |  |  |  |  |  | return( $self->error( "Oh no, the shared memory object is gone! That should not happen." ) ); | 
| 117 | 10 |  |  |  |  | 239 | my $data; | 
| 118 | 10 |  |  |  |  | 44 | my $len = $shem->read( $data ); | 
| 119 | 10 | 50 |  |  |  | 45 | return( $self->pass_error( $shem->error ) ) if( !defined( $len ) ); | 
| 120 |  |  |  |  |  |  | ## $self->message( 3, "Data read is: ", sub{ $self->dump( $data ) } ); | 
| 121 | 10 | 100 |  |  |  | 46 | $data = {} unless( ref( $data ) eq 'HASH' ); | 
| 122 | 10 |  |  |  |  | 62 | return( $data ); | 
| 123 |  |  |  |  |  |  | } | 
| 124 |  |  |  |  |  |  |  | 
| 125 |  |  |  |  |  |  | sub remove | 
| 126 |  |  |  |  |  |  | { | 
| 127 | 1 |  |  | 1 | 0 | 449 | my $self = shift( @_ ); | 
| 128 | 1 |  | 50 |  |  | 11 | my $shem = $self->shem || | 
| 129 |  |  |  |  |  |  | return( $self->error( "Oh no, the shared memory object is gone! That should not happen." ) ); | 
| 130 | 1 |  |  |  |  | 25 | my $rv; | 
| 131 | 1 | 50 |  |  |  | 13 | if( !defined( $rv = $shem->remove ) ) | 
| 132 |  |  |  |  |  |  | { | 
| 133 | 0 |  |  |  |  | 0 | return( $self->pass_error( $shem->error ) ); | 
| 134 |  |  |  |  |  |  | } | 
| 135 | 1 |  |  |  |  | 86 | return( $rv ); | 
| 136 |  |  |  |  |  |  | } | 
| 137 |  |  |  |  |  |  |  | 
| 138 |  |  |  |  |  |  | sub set | 
| 139 |  |  |  |  |  |  | { | 
| 140 | 2 |  |  | 2 | 1 | 903 | my $self = shift( @_ ); | 
| 141 | 2 |  | 50 |  |  | 14 | my $data = $self->read_mem || return; | 
| 142 | 2 |  |  |  |  | 10 | my @callinfo = caller; | 
| 143 |  |  |  |  |  |  | ## $self->message( 3, "Called from file $callinfo[1] at line $callinfo[2]" ); | 
| 144 | 2 |  |  |  |  | 24 | my( $key, $value ) = @_; | 
| 145 |  |  |  |  |  |  | ## $self->message( 3, "Set key '$key' with value '$value'" ); | 
| 146 | 2 | 50 |  |  |  | 10 | return( $self->error( "Key provided to set value is empty." ) ) if( !length( $key ) ); | 
| 147 | 2 |  |  |  |  | 13 | $data->{ $key } = $value; | 
| 148 | 2 | 50 |  |  |  | 25 | $self->write_mem( $data ) || return; | 
| 149 | 2 |  |  |  |  | 15 | return( $self ); | 
| 150 |  |  |  |  |  |  | } | 
| 151 |  |  |  |  |  |  |  | 
| 152 | 79 |  |  | 79 | 1 | 454 | sub shem { return( shift->_set_get_object_without_init( 'shem', 'Apache2::SSI::SharedMem', @_ ) ); } | 
| 153 |  |  |  |  |  |  |  | 
| 154 | 0 |  |  | 0 | 1 | 0 | sub size { return( shift->_set_get_scalar( 'size', @_ ) ); } | 
| 155 |  |  |  |  |  |  |  | 
| 156 | 63 |  |  | 63 | 0 | 916 | sub supported { return( Apache2::SSI::SharedMem->supported ); } | 
| 157 |  |  |  |  |  |  |  | 
| 158 |  |  |  |  |  |  | sub unset | 
| 159 |  |  |  |  |  |  | { | 
| 160 | 1 |  |  | 1 | 1 | 3 | my $self = shift( @_ ); | 
| 161 | 1 |  |  |  |  | 9 | my $key  = shift( @_ ); | 
| 162 | 1 | 50 |  |  |  | 6 | return( $self->error( "Key provided to unset value is empty." ) ) if( !length( $key ) ); | 
| 163 | 1 |  | 50 |  |  | 12 | my $data = $self->read_mem || return; | 
| 164 | 1 |  |  |  |  | 4 | delete( $data->{ $key } ); | 
| 165 | 1 | 50 |  |  |  | 14 | $self->write_mem( $data ) || return; | 
| 166 | 1 |  |  |  |  | 15 | return( $self ); | 
| 167 |  |  |  |  |  |  | } | 
| 168 |  |  |  |  |  |  |  | 
| 169 |  |  |  |  |  |  | sub write_mem | 
| 170 |  |  |  |  |  |  | { | 
| 171 | 5 |  |  | 5 | 1 | 14 | my $self = shift( @_ ); | 
| 172 | 5 |  | 50 |  |  | 21 | my $shem = $self->shem || | 
| 173 |  |  |  |  |  |  | return( $self->error( "Oh no, the shared memory object is gone! That should not happen." ) ); | 
| 174 | 5 |  |  |  |  | 117 | my $data = shift( @_ ); | 
| 175 | 5 | 50 |  |  |  | 20 | return( $self->error( "I was expecting an hash reference and got instead '$data'" ) ) if( ref( $data ) ne 'HASH' ); | 
| 176 | 5 | 50 |  |  |  | 31 | if( !defined( $shem->lock( ( LOCK_EX | LOCK_NB ) ) ) ) | 
| 177 |  |  |  |  |  |  | { | 
| 178 |  |  |  |  |  |  | ## $self->message( 3, "Error setting a non-blocking lock on the semaphore" ); | 
| 179 | 0 |  |  |  |  | 0 | return( $self->pass_error( $shem->error ) ); | 
| 180 |  |  |  |  |  |  | } | 
| 181 | 5 |  |  |  |  | 32 | my $rc = $shem->write( $data ); | 
| 182 | 5 |  |  |  |  | 41 | $shem->unlock; | 
| 183 | 5 | 50 |  |  |  | 17 | return( $self->pass_error( $shem->error ) ) if( !defined( $rc ) ); | 
| 184 | 5 |  |  |  |  | 22 | return( $self ); | 
| 185 |  |  |  |  |  |  | } | 
| 186 |  |  |  |  |  |  |  | 
| 187 |  |  |  |  |  |  | 1; | 
| 188 |  |  |  |  |  |  |  | 
| 189 |  |  |  |  |  |  | __END__ | 
| 190 |  |  |  |  |  |  |  | 
| 191 |  |  |  |  |  |  | =encoding utf-8 | 
| 192 |  |  |  |  |  |  |  | 
| 193 |  |  |  |  |  |  | =head1 NAME | 
| 194 |  |  |  |  |  |  |  | 
| 195 |  |  |  |  |  |  | Apache2::SSI::Notes - Apache2 Server Side Include Notes | 
| 196 |  |  |  |  |  |  |  | 
| 197 |  |  |  |  |  |  | =head1 SYNOPSIS | 
| 198 |  |  |  |  |  |  |  | 
| 199 |  |  |  |  |  |  | my $notes = Apache2::SSI::Notes->new( | 
| 200 |  |  |  |  |  |  | # 100K | 
| 201 |  |  |  |  |  |  | size => 102400, | 
| 202 |  |  |  |  |  |  | debug => 3, | 
| 203 |  |  |  |  |  |  | ) || die( Apache2::SSI::Notes->error ); | 
| 204 |  |  |  |  |  |  |  | 
| 205 |  |  |  |  |  |  | $notes->add( key => $val ); | 
| 206 |  |  |  |  |  |  |  | 
| 207 |  |  |  |  |  |  | $notes->clear; | 
| 208 |  |  |  |  |  |  |  | 
| 209 |  |  |  |  |  |  | $notes->do(sub | 
| 210 |  |  |  |  |  |  | { | 
| 211 |  |  |  |  |  |  | # $_[0] = key | 
| 212 |  |  |  |  |  |  | # $_[1] = value | 
| 213 |  |  |  |  |  |  | $_[1] = Encode::decode( 'utf8', $_[1] ); | 
| 214 |  |  |  |  |  |  | }); | 
| 215 |  |  |  |  |  |  |  | 
| 216 |  |  |  |  |  |  | # Or specify the keys to check | 
| 217 |  |  |  |  |  |  | $notes->do(sub | 
| 218 |  |  |  |  |  |  | { | 
| 219 |  |  |  |  |  |  | # $_[0] = key | 
| 220 |  |  |  |  |  |  | # $_[1] = value | 
| 221 |  |  |  |  |  |  | $_[1] = Encode::decode( 'utf8', $_[1] ); | 
| 222 |  |  |  |  |  |  | }, qw( first_name last_name location ) ); | 
| 223 |  |  |  |  |  |  |  | 
| 224 |  |  |  |  |  |  | my $val = $notes-get( 'name' ); | 
| 225 |  |  |  |  |  |  |  | 
| 226 |  |  |  |  |  |  | # Get all as an hash reference | 
| 227 |  |  |  |  |  |  | my $hashref = $notes->get; | 
| 228 |  |  |  |  |  |  |  | 
| 229 |  |  |  |  |  |  | $notes->set( name => 'John Doe' ); | 
| 230 |  |  |  |  |  |  |  | 
| 231 |  |  |  |  |  |  | # remove entry. This is different from $notes->set( name => undef() ); | 
| 232 |  |  |  |  |  |  | # equivalent to delete( $hash->{name} ); | 
| 233 |  |  |  |  |  |  | $notes->unset( 'name' ); | 
| 234 |  |  |  |  |  |  |  | 
| 235 |  |  |  |  |  |  | =head1 VERSION | 
| 236 |  |  |  |  |  |  |  | 
| 237 |  |  |  |  |  |  | v0.1.0 | 
| 238 |  |  |  |  |  |  |  | 
| 239 |  |  |  |  |  |  | =head1 DESCRIPTION | 
| 240 |  |  |  |  |  |  |  | 
| 241 |  |  |  |  |  |  | L<Apache2::SSI::Notes> provides a mean to share notes in and out of Apache/mod_perl2 environment. | 
| 242 |  |  |  |  |  |  |  | 
| 243 |  |  |  |  |  |  | The interface is loosely mimicking L<APR::Table> on some, but not all, methods. | 
| 244 |  |  |  |  |  |  |  | 
| 245 |  |  |  |  |  |  | So you could have in your script, outside of Apache: | 
| 246 |  |  |  |  |  |  |  | 
| 247 |  |  |  |  |  |  | $notes->set( API_ID => 1234567 ); | 
| 248 |  |  |  |  |  |  |  | 
| 249 |  |  |  |  |  |  | And then, under mod_perl, in your file: | 
| 250 |  |  |  |  |  |  |  | 
| 251 |  |  |  |  |  |  | <!--#if expr="note('API_ID')" --> | 
| 252 |  |  |  |  |  |  |  | 
| 253 |  |  |  |  |  |  | Normally, the C<note> function would work only for values set and retrieved inside the Apache/mod_perl2 framework, but with L<Apache2::SSI::Notes>, you can set a note, say, in a command line script and share it with your Server-Side Includes files. | 
| 254 |  |  |  |  |  |  |  | 
| 255 |  |  |  |  |  |  | To achieve this sharing of notes, L<Apache2::SSI::Notes> uses shared memory (see L<perlipc>) with L<Apache2::SSI::SharedMem> that does the heavy work. | 
| 256 |  |  |  |  |  |  |  | 
| 257 |  |  |  |  |  |  | However, this only works when L<Apache2::SSI> is in charge of parsing SSI files. Apache mod_includes module will not recognise notes stored outside of Apache/mod_perl framework. | 
| 258 |  |  |  |  |  |  |  | 
| 259 |  |  |  |  |  |  | =head1 METHODS | 
| 260 |  |  |  |  |  |  |  | 
| 261 |  |  |  |  |  |  | =head2 new | 
| 262 |  |  |  |  |  |  |  | 
| 263 |  |  |  |  |  |  | This instantiates a notes object. It takes the following parameters: | 
| 264 |  |  |  |  |  |  |  | 
| 265 |  |  |  |  |  |  | =over 4 | 
| 266 |  |  |  |  |  |  |  | 
| 267 |  |  |  |  |  |  | =item I<debug> | 
| 268 |  |  |  |  |  |  |  | 
| 269 |  |  |  |  |  |  | A debug value will enable debugging output (equal or above 3 actually) | 
| 270 |  |  |  |  |  |  |  | 
| 271 |  |  |  |  |  |  | =item I<size> | 
| 272 |  |  |  |  |  |  |  | 
| 273 |  |  |  |  |  |  | The fixed size of the memory allocation. It defaults to 524,288 bytes which is 512 Kb, which should be ample enough. | 
| 274 |  |  |  |  |  |  |  | 
| 275 |  |  |  |  |  |  | =back | 
| 276 |  |  |  |  |  |  |  | 
| 277 |  |  |  |  |  |  | An object will be returned if it successfully initiated, or undef() upon error, which can then be retrieved with C<Apache2::SSI::Notes->error>. You should always check the return value of the methods used here for their definedness. | 
| 278 |  |  |  |  |  |  |  | 
| 279 |  |  |  |  |  |  | my $notes = Apache2::SSI::Notes->new || | 
| 280 |  |  |  |  |  |  | die( Apache2::SSI::Notes->error ); | 
| 281 |  |  |  |  |  |  |  | 
| 282 |  |  |  |  |  |  | =head2 add | 
| 283 |  |  |  |  |  |  |  | 
| 284 |  |  |  |  |  |  | This is an alias for set. | 
| 285 |  |  |  |  |  |  |  | 
| 286 |  |  |  |  |  |  | =head2 clear | 
| 287 |  |  |  |  |  |  |  | 
| 288 |  |  |  |  |  |  | Empty all the notes. Beware that this will empty the notes for all the processes, since the notes are stored in a shared memory. | 
| 289 |  |  |  |  |  |  |  | 
| 290 |  |  |  |  |  |  | =head2 do | 
| 291 |  |  |  |  |  |  |  | 
| 292 |  |  |  |  |  |  | Provided with a callback as a code reference, and optionally an array of keys, and this will loop through all keys or the given keys if any, and call the callback passing it the key and its value. | 
| 293 |  |  |  |  |  |  |  | 
| 294 |  |  |  |  |  |  | For example: | 
| 295 |  |  |  |  |  |  |  | 
| 296 |  |  |  |  |  |  | $notes->do(sub | 
| 297 |  |  |  |  |  |  | { | 
| 298 |  |  |  |  |  |  | my( $n, $v ) = @_; | 
| 299 |  |  |  |  |  |  | if( $n =~ /name/ ) | 
| 300 |  |  |  |  |  |  | { | 
| 301 |  |  |  |  |  |  | $_[1] = Encode::decode( 'utf8', $_[1] ); | 
| 302 |  |  |  |  |  |  | } | 
| 303 |  |  |  |  |  |  | }); | 
| 304 |  |  |  |  |  |  |  | 
| 305 |  |  |  |  |  |  | =head2 get | 
| 306 |  |  |  |  |  |  |  | 
| 307 |  |  |  |  |  |  | Provided with a key and this retrieve its corresponding value, whatever that may be. | 
| 308 |  |  |  |  |  |  |  | 
| 309 |  |  |  |  |  |  | my $val = $notes->get( 'name' ); | 
| 310 |  |  |  |  |  |  |  | 
| 311 |  |  |  |  |  |  | If no key is provided, it returns all the notes as an hash reference. | 
| 312 |  |  |  |  |  |  |  | 
| 313 |  |  |  |  |  |  | my $all = $notes->get; | 
| 314 |  |  |  |  |  |  | print( "API id is $all->{api}\n" ); | 
| 315 |  |  |  |  |  |  |  | 
| 316 |  |  |  |  |  |  | Or maybe | 
| 317 |  |  |  |  |  |  |  | 
| 318 |  |  |  |  |  |  | print( "API id is ", $notes->get->{api}, "\n" ); | 
| 319 |  |  |  |  |  |  |  | 
| 320 |  |  |  |  |  |  | =head2 read_mem | 
| 321 |  |  |  |  |  |  |  | 
| 322 |  |  |  |  |  |  | Access the shared memory and return the hash reference stored. | 
| 323 |  |  |  |  |  |  |  | 
| 324 |  |  |  |  |  |  | If an error occurred, C<undef()> is returned and an L<Module::Generic/error> is set, which can be retrieved like: | 
| 325 |  |  |  |  |  |  |  | 
| 326 |  |  |  |  |  |  | die( $notes->error ); | 
| 327 |  |  |  |  |  |  |  | 
| 328 |  |  |  |  |  |  | Be careful however, that L</get> may return C<undef()> not because an error would have occurred, but because this is the value you would have previously set. | 
| 329 |  |  |  |  |  |  |  | 
| 330 |  |  |  |  |  |  | =head2 set | 
| 331 |  |  |  |  |  |  |  | 
| 332 |  |  |  |  |  |  | Provided with a key and value pair, and this will set its entry into the notes hash accordingly. | 
| 333 |  |  |  |  |  |  |  | 
| 334 |  |  |  |  |  |  | $notes->set( name => 'John Doe' ); | 
| 335 |  |  |  |  |  |  |  | 
| 336 |  |  |  |  |  |  | It returns the notes object to enable chaining. | 
| 337 |  |  |  |  |  |  |  | 
| 338 |  |  |  |  |  |  | =head2 shem | 
| 339 |  |  |  |  |  |  |  | 
| 340 |  |  |  |  |  |  | Returns the current value of the L<Apache2::SSI::SharedMem> object. | 
| 341 |  |  |  |  |  |  |  | 
| 342 |  |  |  |  |  |  | You can also set an alternative value, but this is not advised unless you know what you are doing. | 
| 343 |  |  |  |  |  |  |  | 
| 344 |  |  |  |  |  |  | =head2 size | 
| 345 |  |  |  |  |  |  |  | 
| 346 |  |  |  |  |  |  | Sets or gets the shared memory block size. | 
| 347 |  |  |  |  |  |  |  | 
| 348 |  |  |  |  |  |  | This should really not be changed. If you do want to change it, you first need to remove the shared memory. | 
| 349 |  |  |  |  |  |  |  | 
| 350 |  |  |  |  |  |  | $notes->shem->remove; | 
| 351 |  |  |  |  |  |  |  | 
| 352 |  |  |  |  |  |  | And then create a new L<Apache2::SSI::Notes> object with a different size parameter value. | 
| 353 |  |  |  |  |  |  |  | 
| 354 |  |  |  |  |  |  | =head2 unset | 
| 355 |  |  |  |  |  |  |  | 
| 356 |  |  |  |  |  |  | Remove the notes entry for the given key. | 
| 357 |  |  |  |  |  |  |  | 
| 358 |  |  |  |  |  |  | # No more name key: | 
| 359 |  |  |  |  |  |  | $notes->unset( 'name' ); | 
| 360 |  |  |  |  |  |  |  | 
| 361 |  |  |  |  |  |  | It returns the notes object to enable chaining. | 
| 362 |  |  |  |  |  |  |  | 
| 363 |  |  |  |  |  |  | =head2 write_mem | 
| 364 |  |  |  |  |  |  |  | 
| 365 |  |  |  |  |  |  | Provided with data, and this will write the data to the shared memory. | 
| 366 |  |  |  |  |  |  |  | 
| 367 |  |  |  |  |  |  | =head1 CAVEAT | 
| 368 |  |  |  |  |  |  |  | 
| 369 |  |  |  |  |  |  | L<Apache2::SSI::Notes> do not work under threaded perl | 
| 370 |  |  |  |  |  |  |  | 
| 371 |  |  |  |  |  |  | =head1 AUTHOR | 
| 372 |  |  |  |  |  |  |  | 
| 373 |  |  |  |  |  |  | Jacques Deguest E<lt>F<jack@deguest.jp>E<gt> | 
| 374 |  |  |  |  |  |  |  | 
| 375 |  |  |  |  |  |  | CPAN ID: jdeguest | 
| 376 |  |  |  |  |  |  |  | 
| 377 |  |  |  |  |  |  | L<https://git.deguest.jp/jack/Apache2-SSI> | 
| 378 |  |  |  |  |  |  |  | 
| 379 |  |  |  |  |  |  | =head1 SEE ALSO | 
| 380 |  |  |  |  |  |  |  | 
| 381 |  |  |  |  |  |  | mod_include, mod_perl(3), L<APR::Finfo>, L<perlfunc/stat> | 
| 382 |  |  |  |  |  |  | L<https://httpd.apache.org/docs/current/en/mod/mod_include.html>, | 
| 383 |  |  |  |  |  |  | L<https://httpd.apache.org/docs/current/en/howto/ssi.html>, | 
| 384 |  |  |  |  |  |  | L<https://httpd.apache.org/docs/current/en/expr.html> | 
| 385 |  |  |  |  |  |  | L<https://perl.apache.org/docs/2.0/user/handlers/filters.html#C_PerlOutputFilterHandler_> | 
| 386 |  |  |  |  |  |  |  | 
| 387 |  |  |  |  |  |  | =head1 COPYRIGHT & LICENSE | 
| 388 |  |  |  |  |  |  |  | 
| 389 |  |  |  |  |  |  | Copyright (c) 2020-2021 DEGUEST Pte. Ltd. | 
| 390 |  |  |  |  |  |  |  | 
| 391 |  |  |  |  |  |  | You can use, copy, modify and redistribute this package and associated | 
| 392 |  |  |  |  |  |  | files under the same terms as Perl itself. | 
| 393 |  |  |  |  |  |  |  | 
| 394 |  |  |  |  |  |  | =cut | 
| 395 |  |  |  |  |  |  |  |