diff --git a/docs/pyShelf/html/HTML/D/index.html b/docs/pyShelf/html/HTML/D/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/D/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/FILEMAP b/docs/pyShelf/html/HTML/FILEMAP
new file mode 100644
index 0000000..d2c3b38
--- /dev/null
+++ b/docs/pyShelf/html/HTML/FILEMAP
@@ -0,0 +1 @@
+frontend/node_modules/flatted/php/flatted.php S/2760.html
diff --git a/docs/pyShelf/html/HTML/GTAGSROOT b/docs/pyShelf/html/HTML/GTAGSROOT
new file mode 100644
index 0000000..998e0fa
--- /dev/null
+++ b/docs/pyShelf/html/HTML/GTAGSROOT
@@ -0,0 +1 @@
+/home/th3r00t/.local/builds/pyShelf/src
diff --git a/docs/pyShelf/html/HTML/I/index.html b/docs/pyShelf/html/HTML/I/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/I/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/J/index.html b/docs/pyShelf/html/HTML/J/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/J/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/R/1.html b/docs/pyShelf/html/HTML/R/1.html
new file mode 100644
index 0000000..39423de
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/1.html
@@ -0,0 +1,26 @@
+
+
+
+Flatted
+
+
+
+
+
+
+
+Flatted 42 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(false, array_keys($value), $input, $set, $value);
+Flatted 44 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+Flatted 54 frontend/node_modules/flatted/php/flatted.php $i = intval(Flatted::index($known, $input, $value));
+Flatted 56 frontend/node_modules/flatted/php/flatted.php $output[$i] = Flatted::transform($known, $input, $input[$i]);
+Flatted 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+Flatted 98 frontend/node_modules/flatted/php/flatted.php return Flatted::index($known, $input, $value);
+Flatted 106 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(false, array_keys($value), $input, $set, $value);
+Flatted 110 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+Flatted 124 frontend/node_modules/flatted/php/flatted.php return Flatted::relate($known, $input, $value);
+Flatted 131 frontend/node_modules/flatted/php/flatted.php $keys = Flatted::keys($value);
+Flatted 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+Flatted 147 frontend/node_modules/flatted/php/flatted.php $keys = Flatted::keys($value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/R/2.html b/docs/pyShelf/html/HTML/R/2.html
new file mode 100644
index 0000000..e17eeaf
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/2.html
@@ -0,0 +1,17 @@
+
+
+
+FlattedString
+
+
+
+
+
+
+
+FlattedString 64 frontend/node_modules/flatted/php/flatted.php return $value instanceof FlattedString ? $value->value : $value;
+FlattedString 87 frontend/node_modules/flatted/php/flatted.php if ($value instanceof FlattedString)
+FlattedString 141 frontend/node_modules/flatted/php/flatted.php return new FlattedString($value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/R/3.html b/docs/pyShelf/html/HTML/R/3.html
new file mode 100644
index 0000000..2800772
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/3.html
@@ -0,0 +1,19 @@
+
+
+
+index
+
+
+
+
+
+
+
+index 54 frontend/node_modules/flatted/php/flatted.php $i = intval(Flatted::index($known, $input, $value));
+index 69 frontend/node_modules/flatted/php/flatted.php $index = strval(count($input) - 1);
+index 71 frontend/node_modules/flatted/php/flatted.php $known->value[] = &$index;
+index 72 frontend/node_modules/flatted/php/flatted.php return $index;
+index 98 frontend/node_modules/flatted/php/flatted.php return Flatted::index($known, $input, $value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/R/4.html b/docs/pyShelf/html/HTML/R/4.html
new file mode 100644
index 0000000..5c1d9d7
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/4.html
@@ -0,0 +1,25 @@
+
+
+
+keys
+
+
+
+
+
+
+
+keys 44 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+keys 78 frontend/node_modules/flatted/php/flatted.php $keys = array();
+keys 80 frontend/node_modules/flatted/php/flatted.php $keys[] = $prop->getName();
+keys 81 frontend/node_modules/flatted/php/flatted.php return $keys;
+keys 84 frontend/node_modules/flatted/php/flatted.php private static function loop($obj, $keys, &$input, &$set, &$output) {
+keys 85 frontend/node_modules/flatted/php/flatted.php foreach ($keys as $key) {
+keys 110 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+keys 131 frontend/node_modules/flatted/php/flatted.php $keys = Flatted::keys($value);
+keys 132 frontend/node_modules/flatted/php/flatted.php foreach ($keys as $key)
+keys 147 frontend/node_modules/flatted/php/flatted.php $keys = Flatted::keys($value);
+keys 148 frontend/node_modules/flatted/php/flatted.php foreach ($keys as $key) {
+
+
+
diff --git a/docs/pyShelf/html/HTML/R/5.html b/docs/pyShelf/html/HTML/R/5.html
new file mode 100644
index 0000000..36a457a
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/5.html
@@ -0,0 +1,18 @@
+
+
+
+loop
+
+
+
+
+
+
+
+loop 42 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(false, array_keys($value), $input, $set, $value);
+loop 44 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+loop 106 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(false, array_keys($value), $input, $set, $value);
+loop 110 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/R/7.html b/docs/pyShelf/html/HTML/R/7.html
new file mode 100644
index 0000000..c565773
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/7.html
@@ -0,0 +1,16 @@
+
+
+
+relate
+
+
+
+
+
+
+
+relate 124 frontend/node_modules/flatted/php/flatted.php return Flatted::relate($known, $input, $value);
+relate 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+
+
+
diff --git a/docs/pyShelf/html/HTML/R/index.html b/docs/pyShelf/html/HTML/R/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/R/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/S/2760.html b/docs/pyShelf/html/HTML/S/2760.html
new file mode 100644
index 0000000..ad1de9c
--- /dev/null
+++ b/docs/pyShelf/html/HTML/S/2760.html
@@ -0,0 +1,191 @@
+
+
+
+frontend/node_modules/flatted/php/flatted.php
+
+
+
+
+
+
+
+
+
+
+This source file includes following definitions.
+
+__construct
+parse
+stringify
+asString
+index
+keys
+loop
+relate
+ref
+transform
+wrap
+
+
+
+ 1 <?php
+ 2
+ 3
+ 4
+ 5
+ 6
+ 7
+ 8
+ 9
+ 10
+ 11
+ 12
+ 13
+ 14
+ 15
+ 16
+ 17
+ 18
+ 19
+ 20
+ 21 class FlattedString {
+ 22 public function __construct($value ) {
+ 23 $this->value = $value ;
+ 24 }
+ 25 }
+ 26
+ 27 class Flatted {
+ 28
+ 29
+ 30 public static function parse($json , $assoc = false , $depth = 512, $options = 0) {
+ 31 $input = array_map (
+ 32 'Flatted::asString',
+ 33 array_map (
+ 34 'Flatted::wrap',
+ 35 json_decode($json , $assoc , $depth , $options )
+ 36 )
+ 37 );
+ 38 $value = &$input [0];
+ 39 $set = array ();
+ 40 $set [] = &$value ;
+ 41 if (is_array ($value ))
+ 42 return Flatted ::loop (false , array_keys ($value ), $input , $set , $value );
+ 43 if (is_object ($value ))
+ 44 return Flatted ::loop (true , Flatted ::keys ($value ), $input , $set , $value );
+ 45 return $value ;
+ 46 }
+ 47
+ 48 public static function stringify($value , $options = 0, $depth = 512) {
+ 49 $known = new stdClass ;
+ 50 $known ->key = array ();
+ 51 $known ->value = array ();
+ 52 $input = array ();
+ 53 $output = array ();
+ 54 $i = intval(Flatted ::index ($known , $input , $value ));
+ 55 while ($i < count ($input )) {
+ 56 $output [$i ] = Flatted ::transform ($known , $input , $input [$i ]);
+ 57 $i ++;
+ 58 }
+ 59 return json_encode($output , $options , $depth );
+ 60 }
+ 61
+ 62
+ 63 private static function asString($value ) {
+ 64 return $value instanceof FlattedString ? $value ->value : $value ;
+ 65 }
+ 66
+ 67 private static function index (&$known , &$input , &$value ) {
+ 68 $input [] = &$value ;
+ 69 $index = strval(count ($input ) - 1);
+ 70 $known ->key [] = &$value ;
+ 71 $known ->value [] = &$index;
+ 72 return $index;
+ 73 }
+ 74
+ 75 private static function keys (&$value ) {
+ 76 $obj = new ReflectionObject($value );
+ 77 $props = $obj ->getProperties();
+ 78 $keys = array ();
+ 79 foreach ($props as $prop )
+ 80 $keys[] = $prop ->getName();
+ 81 return $keys;
+ 82 }
+ 83
+ 84 private static function loop ($obj , $keys, &$input , &$set , &$output ) {
+ 85 foreach ($keys as $key ) {
+ 86 $value = $obj ? $output ->$key : $output [$key ];
+ 87 if ($value instanceof FlattedString )
+ 88 Flatted ::ref ($obj , $key , $input [$value ->value ], $input , $set , $output );
+ 89 }
+ 90 return $output ;
+ 91 }
+ 92
+ 93 private static function relate (&$known , &$input , &$value ) {
+ 94 if (is_string ($value ) || is_array ($value ) || is_object ($value )) {
+ 95 $key = array_search($value , $known ->key , true );
+ 96 if ($key !== false )
+ 97 return $known ->value [$key ];
+ 98 return Flatted ::index ($known , $input , $value );
+ 99 }
+ 100 return $value ;
+ 101 }
+ 102
+ 103 private static function ref ($obj , &$key , &$value , &$input , &$set , &$output ) {
+ 104 if (is_array ($value ) && !in_array ($value , $set , true )) {
+ 105 $set [] = $value ;
+ 106 $value = Flatted ::loop (false , array_keys ($value ), $input , $set , $value );
+ 107 }
+ 108 elseif (is_object ($value ) && !in_array ($value , $set , true )) {
+ 109 $set [] = $value ;
+ 110 $value = Flatted ::loop (true , Flatted ::keys ($value ), $input , $set , $value );
+ 111 }
+ 112 if ($obj ) {
+ 113 $output ->$key = &$value ;
+ 114 }
+ 115 else {
+ 116 $output [$key ] = &$value ;
+ 117 }
+ 118 }
+ 119
+ 120 private static function transform (&$known , &$input , &$value ) {
+ 121 if (is_array ($value )) {
+ 122 return array_map (
+ 123 function ($value ) use (&$known , &$input ) {
+ 124 return Flatted ::relate ($known , $input , $value );
+ 125 } ,
+ 126 $value
+ 127 );
+ 128 }
+ 129 if (is_object ($value )) {
+ 130 $object = new stdClass ;
+ 131 $keys = Flatted ::keys ($value );
+ 132 foreach ($keys as $key )
+ 133 $object ->$key = Flatted ::relate ($known , $input , $value ->$key );
+ 134 return $object ;
+ 135 }
+ 136 return $value ;
+ 137 }
+ 138
+ 139 private static function wrap ($value ) {
+ 140 if (is_string ($value )) {
+ 141 return new FlattedString ($value );
+ 142 }
+ 143 if (is_array ($value )) {
+ 144 return array_map ('Flatted::wrap', $value );
+ 145 }
+ 146 if (is_object ($value )) {
+ 147 $keys = Flatted ::keys ($value );
+ 148 foreach ($keys as $key ) {
+ 149 $value ->$key = self::wrap ($value ->$key );
+ 150 }
+ 151 return $value ;
+ 152 }
+ 153 return $value ;
+ 154 }
+ 155 }
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/HTML/S/index.html b/docs/pyShelf/html/HTML/S/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/S/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/Y/10.html b/docs/pyShelf/html/HTML/Y/10.html
new file mode 100644
index 0000000..c2640e8
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/10.html
@@ -0,0 +1,18 @@
+
+
+
+i
+
+
+
+
+
+
+
+i 54 frontend/node_modules/flatted/php/flatted.php $i = intval(Flatted::index($known, $input, $value));
+i 55 frontend/node_modules/flatted/php/flatted.php while ($i < count($input)) {
+i 56 frontend/node_modules/flatted/php/flatted.php $output[$i] = Flatted::transform($known, $input, $input[$i]);
+i 57 frontend/node_modules/flatted/php/flatted.php $i++;
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/11.html b/docs/pyShelf/html/HTML/Y/11.html
new file mode 100644
index 0000000..600fcb1
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/11.html
@@ -0,0 +1,16 @@
+
+
+
+in_array
+
+
+
+
+
+
+
+in_array 104 frontend/node_modules/flatted/php/flatted.php if (is_array($value) && !in_array($value, $set, true)) {
+in_array 108 frontend/node_modules/flatted/php/flatted.php elseif (is_object($value) && !in_array($value, $set, true)) {
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/12.html b/docs/pyShelf/html/HTML/Y/12.html
new file mode 100644
index 0000000..28807a7
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/12.html
@@ -0,0 +1,36 @@
+
+
+
+input
+
+
+
+
+
+
+
+input 31 frontend/node_modules/flatted/php/flatted.php $input = array_map(
+input 38 frontend/node_modules/flatted/php/flatted.php $value = &$input[0];
+input 42 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(false, array_keys($value), $input, $set, $value);
+input 44 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+input 52 frontend/node_modules/flatted/php/flatted.php $input = array();
+input 54 frontend/node_modules/flatted/php/flatted.php $i = intval(Flatted::index($known, $input, $value));
+input 55 frontend/node_modules/flatted/php/flatted.php while ($i < count($input)) {
+input 56 frontend/node_modules/flatted/php/flatted.php $output[$i] = Flatted::transform($known, $input, $input[$i]);
+input 67 frontend/node_modules/flatted/php/flatted.php private static function index(&$known, &$input, &$value) {
+input 68 frontend/node_modules/flatted/php/flatted.php $input[] = &$value;
+input 69 frontend/node_modules/flatted/php/flatted.php $index = strval(count($input) - 1);
+input 84 frontend/node_modules/flatted/php/flatted.php private static function loop($obj, $keys, &$input, &$set, &$output) {
+input 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+input 93 frontend/node_modules/flatted/php/flatted.php private static function relate(&$known, &$input, &$value) {
+input 98 frontend/node_modules/flatted/php/flatted.php return Flatted::index($known, $input, $value);
+input 103 frontend/node_modules/flatted/php/flatted.php private static function ref($obj, &$key, &$value, &$input, &$set, &$output) {
+input 106 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(false, array_keys($value), $input, $set, $value);
+input 110 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+input 120 frontend/node_modules/flatted/php/flatted.php private static function transform(&$known, &$input, &$value) {
+input 123 frontend/node_modules/flatted/php/flatted.php function ($value) use(&$known, &$input) {
+input 124 frontend/node_modules/flatted/php/flatted.php return Flatted::relate($known, $input, $value);
+input 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/14.html b/docs/pyShelf/html/HTML/Y/14.html
new file mode 100644
index 0000000..2b8a384
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/14.html
@@ -0,0 +1,19 @@
+
+
+
+is_array
+
+
+
+
+
+
+
+is_array 41 frontend/node_modules/flatted/php/flatted.php if (is_array($value))
+is_array 94 frontend/node_modules/flatted/php/flatted.php if (is_string($value) || is_array($value) || is_object($value)) {
+is_array 104 frontend/node_modules/flatted/php/flatted.php if (is_array($value) && !in_array($value, $set, true)) {
+is_array 121 frontend/node_modules/flatted/php/flatted.php if (is_array($value)) {
+is_array 143 frontend/node_modules/flatted/php/flatted.php if (is_array($value)) {
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/15.html b/docs/pyShelf/html/HTML/Y/15.html
new file mode 100644
index 0000000..c6dddf9
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/15.html
@@ -0,0 +1,19 @@
+
+
+
+is_object
+
+
+
+
+
+
+
+is_object 43 frontend/node_modules/flatted/php/flatted.php if (is_object($value))
+is_object 94 frontend/node_modules/flatted/php/flatted.php if (is_string($value) || is_array($value) || is_object($value)) {
+is_object 108 frontend/node_modules/flatted/php/flatted.php elseif (is_object($value) && !in_array($value, $set, true)) {
+is_object 129 frontend/node_modules/flatted/php/flatted.php if (is_object($value)) {
+is_object 146 frontend/node_modules/flatted/php/flatted.php if (is_object($value)) {
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/16.html b/docs/pyShelf/html/HTML/Y/16.html
new file mode 100644
index 0000000..1d96fcc
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/16.html
@@ -0,0 +1,16 @@
+
+
+
+is_string
+
+
+
+
+
+
+
+is_string 94 frontend/node_modules/flatted/php/flatted.php if (is_string($value) || is_array($value) || is_object($value)) {
+is_string 140 frontend/node_modules/flatted/php/flatted.php if (is_string($value)) {
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/17.html b/docs/pyShelf/html/HTML/Y/17.html
new file mode 100644
index 0000000..3805311
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/17.html
@@ -0,0 +1,16 @@
+
+
+
+json
+
+
+
+
+
+
+
+json 30 frontend/node_modules/flatted/php/flatted.php public static function parse($json, $assoc = false, $depth = 512, $options = 0) {
+json 35 frontend/node_modules/flatted/php/flatted.php json_decode($json, $assoc, $depth, $options)
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/2.html b/docs/pyShelf/html/HTML/Y/2.html
new file mode 100644
index 0000000..1d42fe2
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/2.html
@@ -0,0 +1,16 @@
+
+
+
+array_keys
+
+
+
+
+
+
+
+array_keys 42 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(false, array_keys($value), $input, $set, $value);
+array_keys 106 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(false, array_keys($value), $input, $set, $value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/20.html b/docs/pyShelf/html/HTML/Y/20.html
new file mode 100644
index 0000000..41b681c
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/20.html
@@ -0,0 +1,29 @@
+
+
+
+key
+
+
+
+
+
+
+
+key 50 frontend/node_modules/flatted/php/flatted.php $known->key = array();
+key 70 frontend/node_modules/flatted/php/flatted.php $known->key[] = &$value;
+key 85 frontend/node_modules/flatted/php/flatted.php foreach ($keys as $key) {
+key 86 frontend/node_modules/flatted/php/flatted.php $value = $obj ? $output->$key : $output[$key];
+key 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+key 95 frontend/node_modules/flatted/php/flatted.php $key = array_search($value, $known->key, true);
+key 96 frontend/node_modules/flatted/php/flatted.php if ($key !== false)
+key 97 frontend/node_modules/flatted/php/flatted.php return $known->value[$key];
+key 103 frontend/node_modules/flatted/php/flatted.php private static function ref($obj, &$key, &$value, &$input, &$set, &$output) {
+key 113 frontend/node_modules/flatted/php/flatted.php $output->$key = &$value;
+key 116 frontend/node_modules/flatted/php/flatted.php $output[$key] = &$value;
+key 132 frontend/node_modules/flatted/php/flatted.php foreach ($keys as $key)
+key 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+key 148 frontend/node_modules/flatted/php/flatted.php foreach ($keys as $key) {
+key 149 frontend/node_modules/flatted/php/flatted.php $value->$key = self::wrap($value->$key);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/21.html b/docs/pyShelf/html/HTML/Y/21.html
new file mode 100644
index 0000000..f347913
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/21.html
@@ -0,0 +1,30 @@
+
+
+
+known
+
+
+
+
+
+
+
+known 49 frontend/node_modules/flatted/php/flatted.php $known = new stdClass;
+known 50 frontend/node_modules/flatted/php/flatted.php $known->key = array();
+known 51 frontend/node_modules/flatted/php/flatted.php $known->value = array();
+known 54 frontend/node_modules/flatted/php/flatted.php $i = intval(Flatted::index($known, $input, $value));
+known 56 frontend/node_modules/flatted/php/flatted.php $output[$i] = Flatted::transform($known, $input, $input[$i]);
+known 67 frontend/node_modules/flatted/php/flatted.php private static function index(&$known, &$input, &$value) {
+known 70 frontend/node_modules/flatted/php/flatted.php $known->key[] = &$value;
+known 71 frontend/node_modules/flatted/php/flatted.php $known->value[] = &$index;
+known 93 frontend/node_modules/flatted/php/flatted.php private static function relate(&$known, &$input, &$value) {
+known 95 frontend/node_modules/flatted/php/flatted.php $key = array_search($value, $known->key, true);
+known 97 frontend/node_modules/flatted/php/flatted.php return $known->value[$key];
+known 98 frontend/node_modules/flatted/php/flatted.php return Flatted::index($known, $input, $value);
+known 120 frontend/node_modules/flatted/php/flatted.php private static function transform(&$known, &$input, &$value) {
+known 123 frontend/node_modules/flatted/php/flatted.php function ($value) use(&$known, &$input) {
+known 124 frontend/node_modules/flatted/php/flatted.php return Flatted::relate($known, $input, $value);
+known 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/22.html b/docs/pyShelf/html/HTML/Y/22.html
new file mode 100644
index 0000000..38cf29f
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/22.html
@@ -0,0 +1,21 @@
+
+
+
+obj
+
+
+
+
+
+
+
+obj 76 frontend/node_modules/flatted/php/flatted.php $obj = new ReflectionObject($value);
+obj 77 frontend/node_modules/flatted/php/flatted.php $props = $obj->getProperties();
+obj 84 frontend/node_modules/flatted/php/flatted.php private static function loop($obj, $keys, &$input, &$set, &$output) {
+obj 86 frontend/node_modules/flatted/php/flatted.php $value = $obj ? $output->$key : $output[$key];
+obj 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+obj 103 frontend/node_modules/flatted/php/flatted.php private static function ref($obj, &$key, &$value, &$input, &$set, &$output) {
+obj 112 frontend/node_modules/flatted/php/flatted.php if ($obj) {
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/23.html b/docs/pyShelf/html/HTML/Y/23.html
new file mode 100644
index 0000000..1cb2161
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/23.html
@@ -0,0 +1,17 @@
+
+
+
+object
+
+
+
+
+
+
+
+object 130 frontend/node_modules/flatted/php/flatted.php $object = new stdClass;
+object 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+object 134 frontend/node_modules/flatted/php/flatted.php return $object;
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/24.html b/docs/pyShelf/html/HTML/Y/24.html
new file mode 100644
index 0000000..9825d1c
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/24.html
@@ -0,0 +1,18 @@
+
+
+
+options
+
+
+
+
+
+
+
+options 30 frontend/node_modules/flatted/php/flatted.php public static function parse($json, $assoc = false, $depth = 512, $options = 0) {
+options 35 frontend/node_modules/flatted/php/flatted.php json_decode($json, $assoc, $depth, $options)
+options 48 frontend/node_modules/flatted/php/flatted.php public static function stringify($value, $options = 0, $depth = 512) {
+options 59 frontend/node_modules/flatted/php/flatted.php return json_encode($output, $options, $depth);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/25.html b/docs/pyShelf/html/HTML/Y/25.html
new file mode 100644
index 0000000..dcfdcd0
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/25.html
@@ -0,0 +1,24 @@
+
+
+
+output
+
+
+
+
+
+
+
+output 53 frontend/node_modules/flatted/php/flatted.php $output = array();
+output 56 frontend/node_modules/flatted/php/flatted.php $output[$i] = Flatted::transform($known, $input, $input[$i]);
+output 59 frontend/node_modules/flatted/php/flatted.php return json_encode($output, $options, $depth);
+output 84 frontend/node_modules/flatted/php/flatted.php private static function loop($obj, $keys, &$input, &$set, &$output) {
+output 86 frontend/node_modules/flatted/php/flatted.php $value = $obj ? $output->$key : $output[$key];
+output 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+output 90 frontend/node_modules/flatted/php/flatted.php return $output;
+output 103 frontend/node_modules/flatted/php/flatted.php private static function ref($obj, &$key, &$value, &$input, &$set, &$output) {
+output 113 frontend/node_modules/flatted/php/flatted.php $output->$key = &$value;
+output 116 frontend/node_modules/flatted/php/flatted.php $output[$key] = &$value;
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/26.html b/docs/pyShelf/html/HTML/Y/26.html
new file mode 100644
index 0000000..0f85d8e
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/26.html
@@ -0,0 +1,16 @@
+
+
+
+prop
+
+
+
+
+
+
+
+prop 79 frontend/node_modules/flatted/php/flatted.php foreach ($props as $prop)
+prop 80 frontend/node_modules/flatted/php/flatted.php $keys[] = $prop->getName();
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/27.html b/docs/pyShelf/html/HTML/Y/27.html
new file mode 100644
index 0000000..bdcc710
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/27.html
@@ -0,0 +1,16 @@
+
+
+
+props
+
+
+
+
+
+
+
+props 77 frontend/node_modules/flatted/php/flatted.php $props = $obj->getProperties();
+props 79 frontend/node_modules/flatted/php/flatted.php foreach ($props as $prop)
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/29.html b/docs/pyShelf/html/HTML/Y/29.html
new file mode 100644
index 0000000..ba947df
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/29.html
@@ -0,0 +1,27 @@
+
+
+
+set
+
+
+
+
+
+
+
+set 39 frontend/node_modules/flatted/php/flatted.php $set = array();
+set 40 frontend/node_modules/flatted/php/flatted.php $set[] = &$value;
+set 42 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(false, array_keys($value), $input, $set, $value);
+set 44 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+set 84 frontend/node_modules/flatted/php/flatted.php private static function loop($obj, $keys, &$input, &$set, &$output) {
+set 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+set 103 frontend/node_modules/flatted/php/flatted.php private static function ref($obj, &$key, &$value, &$input, &$set, &$output) {
+set 104 frontend/node_modules/flatted/php/flatted.php if (is_array($value) && !in_array($value, $set, true)) {
+set 105 frontend/node_modules/flatted/php/flatted.php $set[] = $value;
+set 106 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(false, array_keys($value), $input, $set, $value);
+set 108 frontend/node_modules/flatted/php/flatted.php elseif (is_object($value) && !in_array($value, $set, true)) {
+set 109 frontend/node_modules/flatted/php/flatted.php $set[] = $value;
+set 110 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/3.html b/docs/pyShelf/html/HTML/Y/3.html
new file mode 100644
index 0000000..6eb3c08
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/3.html
@@ -0,0 +1,18 @@
+
+
+
+array_map
+
+
+
+
+
+
+
+array_map 31 frontend/node_modules/flatted/php/flatted.php $input = array_map(
+array_map 33 frontend/node_modules/flatted/php/flatted.php array_map(
+array_map 122 frontend/node_modules/flatted/php/flatted.php return array_map(
+array_map 144 frontend/node_modules/flatted/php/flatted.php return array_map('Flatted::wrap', $value);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/30.html b/docs/pyShelf/html/HTML/Y/30.html
new file mode 100644
index 0000000..88a97d3
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/30.html
@@ -0,0 +1,16 @@
+
+
+
+stdClass
+
+
+
+
+
+
+
+stdClass 49 frontend/node_modules/flatted/php/flatted.php $known = new stdClass;
+stdClass 130 frontend/node_modules/flatted/php/flatted.php $object = new stdClass;
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/32.html b/docs/pyShelf/html/HTML/Y/32.html
new file mode 100644
index 0000000..4c478ed
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/32.html
@@ -0,0 +1,71 @@
+
+
+
+value
+
+
+
+
+
+
+
+value 22 frontend/node_modules/flatted/php/flatted.php public function __construct($value) {
+value 23 frontend/node_modules/flatted/php/flatted.php $this->value = $value;
+value 38 frontend/node_modules/flatted/php/flatted.php $value = &$input[0];
+value 40 frontend/node_modules/flatted/php/flatted.php $set[] = &$value;
+value 41 frontend/node_modules/flatted/php/flatted.php if (is_array($value))
+value 42 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(false, array_keys($value), $input, $set, $value);
+value 43 frontend/node_modules/flatted/php/flatted.php if (is_object($value))
+value 44 frontend/node_modules/flatted/php/flatted.php return Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+value 45 frontend/node_modules/flatted/php/flatted.php return $value;
+value 48 frontend/node_modules/flatted/php/flatted.php public static function stringify($value, $options = 0, $depth = 512) {
+value 51 frontend/node_modules/flatted/php/flatted.php $known->value = array();
+value 54 frontend/node_modules/flatted/php/flatted.php $i = intval(Flatted::index($known, $input, $value));
+value 63 frontend/node_modules/flatted/php/flatted.php private static function asString($value) {
+value 64 frontend/node_modules/flatted/php/flatted.php return $value instanceof FlattedString ? $value->value : $value;
+value 67 frontend/node_modules/flatted/php/flatted.php private static function index(&$known, &$input, &$value) {
+value 68 frontend/node_modules/flatted/php/flatted.php $input[] = &$value;
+value 70 frontend/node_modules/flatted/php/flatted.php $known->key[] = &$value;
+value 71 frontend/node_modules/flatted/php/flatted.php $known->value[] = &$index;
+value 75 frontend/node_modules/flatted/php/flatted.php private static function keys(&$value) {
+value 76 frontend/node_modules/flatted/php/flatted.php $obj = new ReflectionObject($value);
+value 86 frontend/node_modules/flatted/php/flatted.php $value = $obj ? $output->$key : $output[$key];
+value 87 frontend/node_modules/flatted/php/flatted.php if ($value instanceof FlattedString)
+value 88 frontend/node_modules/flatted/php/flatted.php Flatted::ref($obj, $key, $input[$value->value], $input, $set, $output);
+value 93 frontend/node_modules/flatted/php/flatted.php private static function relate(&$known, &$input, &$value) {
+value 94 frontend/node_modules/flatted/php/flatted.php if (is_string($value) || is_array($value) || is_object($value)) {
+value 95 frontend/node_modules/flatted/php/flatted.php $key = array_search($value, $known->key, true);
+value 97 frontend/node_modules/flatted/php/flatted.php return $known->value[$key];
+value 98 frontend/node_modules/flatted/php/flatted.php return Flatted::index($known, $input, $value);
+value 100 frontend/node_modules/flatted/php/flatted.php return $value;
+value 103 frontend/node_modules/flatted/php/flatted.php private static function ref($obj, &$key, &$value, &$input, &$set, &$output) {
+value 104 frontend/node_modules/flatted/php/flatted.php if (is_array($value) && !in_array($value, $set, true)) {
+value 105 frontend/node_modules/flatted/php/flatted.php $set[] = $value;
+value 106 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(false, array_keys($value), $input, $set, $value);
+value 108 frontend/node_modules/flatted/php/flatted.php elseif (is_object($value) && !in_array($value, $set, true)) {
+value 109 frontend/node_modules/flatted/php/flatted.php $set[] = $value;
+value 110 frontend/node_modules/flatted/php/flatted.php $value = Flatted::loop(true, Flatted::keys($value), $input, $set, $value);
+value 113 frontend/node_modules/flatted/php/flatted.php $output->$key = &$value;
+value 116 frontend/node_modules/flatted/php/flatted.php $output[$key] = &$value;
+value 120 frontend/node_modules/flatted/php/flatted.php private static function transform(&$known, &$input, &$value) {
+value 121 frontend/node_modules/flatted/php/flatted.php if (is_array($value)) {
+value 123 frontend/node_modules/flatted/php/flatted.php function ($value) use(&$known, &$input) {
+value 124 frontend/node_modules/flatted/php/flatted.php return Flatted::relate($known, $input, $value);
+value 126 frontend/node_modules/flatted/php/flatted.php $value
+value 129 frontend/node_modules/flatted/php/flatted.php if (is_object($value)) {
+value 131 frontend/node_modules/flatted/php/flatted.php $keys = Flatted::keys($value);
+value 133 frontend/node_modules/flatted/php/flatted.php $object->$key = Flatted::relate($known, $input, $value->$key);
+value 136 frontend/node_modules/flatted/php/flatted.php return $value;
+value 139 frontend/node_modules/flatted/php/flatted.php private static function wrap($value) {
+value 140 frontend/node_modules/flatted/php/flatted.php if (is_string($value)) {
+value 141 frontend/node_modules/flatted/php/flatted.php return new FlattedString($value);
+value 143 frontend/node_modules/flatted/php/flatted.php if (is_array($value)) {
+value 144 frontend/node_modules/flatted/php/flatted.php return array_map('Flatted::wrap', $value);
+value 146 frontend/node_modules/flatted/php/flatted.php if (is_object($value)) {
+value 147 frontend/node_modules/flatted/php/flatted.php $keys = Flatted::keys($value);
+value 149 frontend/node_modules/flatted/php/flatted.php $value->$key = self::wrap($value->$key);
+value 151 frontend/node_modules/flatted/php/flatted.php return $value;
+value 153 frontend/node_modules/flatted/php/flatted.php return $value;
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/5.html b/docs/pyShelf/html/HTML/Y/5.html
new file mode 100644
index 0000000..1665b5a
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/5.html
@@ -0,0 +1,16 @@
+
+
+
+assoc
+
+
+
+
+
+
+
+assoc 30 frontend/node_modules/flatted/php/flatted.php public static function parse($json, $assoc = false, $depth = 512, $options = 0) {
+assoc 35 frontend/node_modules/flatted/php/flatted.php json_decode($json, $assoc, $depth, $options)
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/6.html b/docs/pyShelf/html/HTML/Y/6.html
new file mode 100644
index 0000000..8730e19
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/6.html
@@ -0,0 +1,16 @@
+
+
+
+count
+
+
+
+
+
+
+
+count 55 frontend/node_modules/flatted/php/flatted.php while ($i < count($input)) {
+count 69 frontend/node_modules/flatted/php/flatted.php $index = strval(count($input) - 1);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/7.html b/docs/pyShelf/html/HTML/Y/7.html
new file mode 100644
index 0000000..0402737
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/7.html
@@ -0,0 +1,18 @@
+
+
+
+depth
+
+
+
+
+
+
+
+depth 30 frontend/node_modules/flatted/php/flatted.php public static function parse($json, $assoc = false, $depth = 512, $options = 0) {
+depth 35 frontend/node_modules/flatted/php/flatted.php json_decode($json, $assoc, $depth, $options)
+depth 48 frontend/node_modules/flatted/php/flatted.php public static function stringify($value, $options = 0, $depth = 512) {
+depth 59 frontend/node_modules/flatted/php/flatted.php return json_encode($output, $options, $depth);
+
+
+
diff --git a/docs/pyShelf/html/HTML/Y/index.html b/docs/pyShelf/html/HTML/Y/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/Y/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/defines.html b/docs/pyShelf/html/HTML/defines.html
new file mode 100644
index 0000000..2c6549d
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines.html
@@ -0,0 +1,24 @@
+
+
+
+DEFINITIONS
+
+
+
+
+
+
+
+[F]
+[_]
+[a]
+[i]
+[k]
+[l]
+[p]
+[r]
+[s]
+[t]
+[w]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/F.html b/docs/pyShelf/html/HTML/defines/F.html
new file mode 100644
index 0000000..7af9b04
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/F.html
@@ -0,0 +1,19 @@
+
+
+
+[F]
+
+
+
+
+
+
+
+[..]
+
+Flatted
+FlattedString
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/_.html b/docs/pyShelf/html/HTML/defines/_.html
new file mode 100644
index 0000000..66df305
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/_.html
@@ -0,0 +1,18 @@
+
+
+
+[_]
+
+
+
+
+
+
+
+[..]
+
+__construct
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/index.html b/docs/pyShelf/html/HTML/defines/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/defines/la.html b/docs/pyShelf/html/HTML/defines/la.html
new file mode 100644
index 0000000..e154d93
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/la.html
@@ -0,0 +1,18 @@
+
+
+
+[a]
+
+
+
+
+
+
+
+[..]
+
+asString
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/li.html b/docs/pyShelf/html/HTML/defines/li.html
new file mode 100644
index 0000000..716bf7e
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/li.html
@@ -0,0 +1,18 @@
+
+
+
+[i]
+
+
+
+
+
+
+
+[..]
+
+index
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/lk.html b/docs/pyShelf/html/HTML/defines/lk.html
new file mode 100644
index 0000000..ac476aa
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/lk.html
@@ -0,0 +1,18 @@
+
+
+
+[k]
+
+
+
+
+
+
+
+[..]
+
+keys
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/ll.html b/docs/pyShelf/html/HTML/defines/ll.html
new file mode 100644
index 0000000..1e66d15
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/ll.html
@@ -0,0 +1,18 @@
+
+
+
+[l]
+
+
+
+
+
+
+
+[..]
+
+loop
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/lp.html b/docs/pyShelf/html/HTML/defines/lp.html
new file mode 100644
index 0000000..caaf28a
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/lp.html
@@ -0,0 +1,18 @@
+
+
+
+[p]
+
+
+
+
+
+
+
+[..]
+
+parse
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/lr.html b/docs/pyShelf/html/HTML/defines/lr.html
new file mode 100644
index 0000000..9987e2c
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/lr.html
@@ -0,0 +1,19 @@
+
+
+
+[r]
+
+
+
+
+
+
+
+[..]
+
+ref
+relate
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/ls.html b/docs/pyShelf/html/HTML/defines/ls.html
new file mode 100644
index 0000000..a238eed
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/ls.html
@@ -0,0 +1,18 @@
+
+
+
+[s]
+
+
+
+
+
+
+
+[..]
+
+stringify
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/lt.html b/docs/pyShelf/html/HTML/defines/lt.html
new file mode 100644
index 0000000..2c137e9
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/lt.html
@@ -0,0 +1,18 @@
+
+
+
+[t]
+
+
+
+
+
+
+
+[..]
+
+transform
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/defines/lw.html b/docs/pyShelf/html/HTML/defines/lw.html
new file mode 100644
index 0000000..2bfcac8
--- /dev/null
+++ b/docs/pyShelf/html/HTML/defines/lw.html
@@ -0,0 +1,18 @@
+
+
+
+[w]
+
+
+
+
+
+
+
+[..]
+
+wrap
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/files.html b/docs/pyShelf/html/HTML/files.html
new file mode 100644
index 0000000..fc76e70
--- /dev/null
+++ b/docs/pyShelf/html/HTML/files.html
@@ -0,0 +1,16 @@
+
+
+
+FILES
+
+
+
+
+
+
+
+
+frontend/
+
+
+
diff --git a/docs/pyShelf/html/HTML/files/3572.html b/docs/pyShelf/html/HTML/files/3572.html
new file mode 100644
index 0000000..04c981a
--- /dev/null
+++ b/docs/pyShelf/html/HTML/files/3572.html
@@ -0,0 +1,18 @@
+
+
+
+frontend/
+
+
+
+
+
+
+
+[..]
+
+node_modules/
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/files/3573.html b/docs/pyShelf/html/HTML/files/3573.html
new file mode 100644
index 0000000..431c95c
--- /dev/null
+++ b/docs/pyShelf/html/HTML/files/3573.html
@@ -0,0 +1,18 @@
+
+
+
+frontend/node_modules/
+
+
+
+
+
+
+
+[..]
+
+flatted/
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/files/3574.html b/docs/pyShelf/html/HTML/files/3574.html
new file mode 100644
index 0000000..347e082
--- /dev/null
+++ b/docs/pyShelf/html/HTML/files/3574.html
@@ -0,0 +1,18 @@
+
+
+
+frontend/node_modules/flatted/
+
+
+
+
+
+
+
+[..]
+
+php/
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/files/3575.html b/docs/pyShelf/html/HTML/files/3575.html
new file mode 100644
index 0000000..fb9faad
--- /dev/null
+++ b/docs/pyShelf/html/HTML/files/3575.html
@@ -0,0 +1,18 @@
+
+
+
+frontend/node_modules/flatted/php/
+
+
+
+
+
+
+
+[..]
+
+flatted.php
+
+[..]
+
+
diff --git a/docs/pyShelf/html/HTML/files/index.html b/docs/pyShelf/html/HTML/files/index.html
new file mode 100644
index 0000000..f931715
--- /dev/null
+++ b/docs/pyShelf/html/HTML/files/index.html
@@ -0,0 +1 @@
+
diff --git a/docs/pyShelf/html/HTML/help.html b/docs/pyShelf/html/HTML/help.html
new file mode 100644
index 0000000..e3184e8
--- /dev/null
+++ b/docs/pyShelf/html/HTML/help.html
@@ -0,0 +1,24 @@
+
+
+
+HELP
+
+
+
+
+
+
+
+/* [<][>][^][v][top][bottom][index][help] */
+
+[<] Previous definition.
+[>] Next definition.
+[^] First definition in this file.
+[v] Last definition in this file.
+[top] Top of this file.
+[bottom] Bottom of this file.
+[index] Return to index page.
+[help] You are seeing now.
+
+
+
diff --git a/docs/pyShelf/html/HTML/index.html b/docs/pyShelf/html/HTML/index.html
new file mode 100644
index 0000000..50e8990
--- /dev/null
+++ b/docs/pyShelf/html/HTML/index.html
@@ -0,0 +1,40 @@
+
+
+
+pyShelf Open Source Ebook Server-0.8.0
+
+
+
+
+
+
+pyShelf Open Source Ebook Server-0.8.0
+
+Last updated Sat Mar 18 00:12:14 EDT 2023
+Powered by
GLOBAL-6.6.9 .
+
+
+
+
+
+
+
+[F]
+[_]
+[a]
+[i]
+[k]
+[l]
+[p]
+[r]
+[s]
+[t]
+[w]
+
+
+
+frontend/
+
+
+
+
diff --git a/docs/pyShelf/html/HTML/mains.html b/docs/pyShelf/html/HTML/mains.html
new file mode 100644
index 0000000..50e8990
--- /dev/null
+++ b/docs/pyShelf/html/HTML/mains.html
@@ -0,0 +1,40 @@
+
+
+
+pyShelf Open Source Ebook Server-0.8.0
+
+
+
+
+
+
+pyShelf Open Source Ebook Server-0.8.0
+
+Last updated Sat Mar 18 00:12:14 EDT 2023
+Powered by
GLOBAL-6.6.9 .
+
+
+
+
+
+
+
+[F]
+[_]
+[a]
+[i]
+[k]
+[l]
+[p]
+[r]
+[s]
+[t]
+[w]
+
+
+
+frontend/
+
+
+
+
diff --git a/docs/pyShelf/html/HTML/rebuild.sh b/docs/pyShelf/html/HTML/rebuild.sh
new file mode 100644
index 0000000..df2f0dc
--- /dev/null
+++ b/docs/pyShelf/html/HTML/rebuild.sh
@@ -0,0 +1,8 @@
+#!/bin/sh
+#
+# rebuild.sh: rebuild hypertext with the previous context.
+#
+# Usage:
+# % sh rebuild.sh
+#
+cd /home/th3r00t/.local/builds/pyShelf/src && GTAGSCONF=':langmap=c\:.c.h,yacc\:.y,asm\:.s.S,java\:.java,cpp\:.c++.cc.hh.cpp.cxx.hxx.hpp.C.H,php\:.php.php3.phtml:skip=HTML/,HTML.pub/,tags,TAGS,ID,y.tab.c,y.tab.h,gtags.files,cscope.files,cscope.out,cscope.po.out,cscope.in.out,SCCS/,RCS/,CVS/,CVSROOT/,{arch}/,autom4te.cache/,*.orig,*.rej,*.bak,*~,#*#,*.swp,*.tmp,*_flymake.*,*_flymake,*.o,*.a,*.so,*.lo,*.zip,*.gz,*.bz2,*.xz,*.lzh,*.Z,*.tgz,*.min.js,*min.css:' htags -g -s -a -n -v -w -t 'pyShelf Open Source Ebook Server-0.8.0' /home/th3r00t/.local/builds/pyShelf/docs/pyShelf/html
diff --git a/docs/pyShelf/html/annotated.html b/docs/pyShelf/html/annotated.html
new file mode 100644
index 0000000..57b4676
--- /dev/null
+++ b/docs/pyShelf/html/annotated.html
@@ -0,0 +1,96 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Class List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
Here are the classes, structs, unions and interfaces with brief descriptions:
+
+
+
+
+
diff --git a/docs/pyShelf/html/bc_s.png b/docs/pyShelf/html/bc_s.png
new file mode 100644
index 0000000..224b29a
Binary files /dev/null and b/docs/pyShelf/html/bc_s.png differ
diff --git a/docs/pyShelf/html/bc_sd.png b/docs/pyShelf/html/bc_sd.png
new file mode 100644
index 0000000..31ca888
Binary files /dev/null and b/docs/pyShelf/html/bc_sd.png differ
diff --git a/docs/pyShelf/html/bdwn.png b/docs/pyShelf/html/bdwn.png
new file mode 100644
index 0000000..940a0b9
Binary files /dev/null and b/docs/pyShelf/html/bdwn.png differ
diff --git a/docs/pyShelf/html/classes.html b/docs/pyShelf/html/classes.html
new file mode 100644
index 0000000..f0c7739
--- /dev/null
+++ b/docs/pyShelf/html/classes.html
@@ -0,0 +1,88 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Class Index
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/closed.png b/docs/pyShelf/html/closed.png
new file mode 100644
index 0000000..98cc2c9
Binary files /dev/null and b/docs/pyShelf/html/closed.png differ
diff --git a/docs/pyShelf/html/d0/d49/classsrc_1_1backend_1_1lib_1_1models_1_1Collection-members.html b/docs/pyShelf/html/d0/d49/classsrc_1_1backend_1_1lib_1_1models_1_1Collection-members.html
new file mode 100644
index 0000000..0a1f181
--- /dev/null
+++ b/docs/pyShelf/html/d0/d49/classsrc_1_1backend_1_1lib_1_1models_1_1Collection-members.html
@@ -0,0 +1,68 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.models.Collection , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d0/d93/namespacesrc_1_1backend_1_1lib_1_1config.html b/docs/pyShelf/html/d0/d93/namespacesrc_1_1backend_1_1lib_1_1config.html
new file mode 100644
index 0000000..3c3a464
--- /dev/null
+++ b/docs/pyShelf/html/d0/d93/namespacesrc_1_1backend_1_1lib_1_1config.html
@@ -0,0 +1,69 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.config Namespace Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Pyshelf's Configuration Object.
+
+
+
+
diff --git a/docs/pyShelf/html/d1/dca/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles-members.html b/docs/pyShelf/html/d1/dca/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles-members.html
new file mode 100644
index 0000000..47d8994
--- /dev/null
+++ b/docs/pyShelf/html/d1/dca/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles-members.html
@@ -0,0 +1,67 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.pyShelf.InitFiles , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d2/dcb/classobjects_1_1JSInterface-members.html b/docs/pyShelf/html/d2/dcb/classobjects_1_1JSInterface-members.html
new file mode 100644
index 0000000..2a12693
--- /dev/null
+++ b/docs/pyShelf/html/d2/dcb/classobjects_1_1JSInterface-members.html
@@ -0,0 +1,67 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for objects.JSInterface , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.html b/docs/pyShelf/html/d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.html
new file mode 100644
index 0000000..25c3688
--- /dev/null
+++ b/docs/pyShelf/html/d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.html
@@ -0,0 +1,138 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.tests.library_test.Test_Config Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.tests.library_test.Test_Config.__init__
+ (
+
+ self )
+
+
+
+
+
+
+
+
◆ open_file()
+
+
+
+
+
+ def src.backend.tests.library_test.Test_Config.open_file
+ (
+
+ self ,
+
+
+
+
+
+ root = "config.json"
+
+
+
+ )
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.png b/docs/pyShelf/html/d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.png
new file mode 100644
index 0000000..fbb3133
Binary files /dev/null and b/docs/pyShelf/html/d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.png differ
diff --git a/docs/pyShelf/html/d3/d4a/classsrc_1_1backend_1_1lib_1_1config_1_1Config-members.html b/docs/pyShelf/html/d3/d4a/classsrc_1_1backend_1_1lib_1_1config_1_1Config-members.html
new file mode 100644
index 0000000..8ff982a
--- /dev/null
+++ b/docs/pyShelf/html/d3/d4a/classsrc_1_1backend_1_1lib_1_1config_1_1Config-members.html
@@ -0,0 +1,91 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.config.Config , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d3/d95/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue-members.html b/docs/pyShelf/html/d3/d95/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue-members.html
new file mode 100644
index 0000000..e3e9ebd
--- /dev/null
+++ b/docs/pyShelf/html/d3/d95/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue-members.html
@@ -0,0 +1,68 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.tests.library_test.TestCatalogue , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d4/d6b/classFastAPIServer_1_1FastAPIServer-members.html b/docs/pyShelf/html/d4/d6b/classFastAPIServer_1_1FastAPIServer-members.html
new file mode 100644
index 0000000..98aee01
--- /dev/null
+++ b/docs/pyShelf/html/d4/d6b/classFastAPIServer_1_1FastAPIServer-members.html
@@ -0,0 +1,76 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for FastAPIServer.FastAPIServer , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d4/d71/classDataHooks_1_1BookInterface-members.html b/docs/pyShelf/html/d4/d71/classDataHooks_1_1BookInterface-members.html
new file mode 100644
index 0000000..6f9c510
--- /dev/null
+++ b/docs/pyShelf/html/d4/d71/classDataHooks_1_1BookInterface-members.html
@@ -0,0 +1,67 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for DataHooks.BookInterface , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d5/d80/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin.html b/docs/pyShelf/html/d5/d80/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin.html
new file mode 100644
index 0000000..f0ecc5c
--- /dev/null
+++ b/docs/pyShelf/html/d5/d80/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin.html
@@ -0,0 +1,162 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.pyShelf.Admin Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+src backend lib pyShelf Admin
+
+
+
+
+
+
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Admin.__init__
+ (
+
+ self ,
+
+
+
+
+
+ root
+
+
+
+ )
+
+
+
+
+
+
+
+
+
+
◆ createsuperuser()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Admin.createsuperuser
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
+
◆ config
+
+
+
+
+
+ src.backend.lib.pyShelf.Admin.config
+
+
+
+
+
+
+
+
◆ db
+
+
+
+
+
+ src.backend.lib.pyShelf.Admin.db
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.html b/docs/pyShelf/html/d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.html
new file mode 100644
index 0000000..167164d
--- /dev/null
+++ b/docs/pyShelf/html/d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.html
@@ -0,0 +1,732 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.library.Catalogue Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+def __init__ (self, config)
+
+def scan_folder (self, _path=None)
+
+def filter_books (self)
+
+def process_by_filetype (self, book)
+
+def extract_metadata_epub (self, book)
+
+def extract_metadata_pdf (self, book)
+
+def extract_metadata_mobi (self, book)
+
+def extract_content (self, book_zip, book)
+
+def extract_cover_html (self, book_zip, book)
+
+def extract_cover_image (self, book_zip, book)
+
+def compare_shelf_current (self)
+
+def import_books (self, list=None, **kwargs)
+
+
+
+
Decodes book metadata for storage
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.__init__
+ (
+
+ self ,
+
+
+
+
+
+ config
+
+
+
+ )
+
+
+
+
+
+
+
+
+
+
◆ compare_shelf_current()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.compare_shelf_current
+ (
+
+ self )
+
+
+
+
+
Calls storage system, gets list of books stored and compares against files on disk
+
+
+
+
+
◆ extract_content()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.extract_content
+ (
+
+ self ,
+
+
+
+
+
+ book_zip ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Opens epub as zip file filters then stores as list any files matching opf_regx
+
+
+
+
+
◆ extract_cover_html()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.extract_cover_html
+ (
+
+ self ,
+
+
+
+
+
+ book_zip ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Opens epub as zip file filters then stores as list any files matching html_regx
+
+
+
+
+
◆ extract_cover_image()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.extract_cover_image
+ (
+
+ self ,
+
+
+
+
+
+ book_zip ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Opens epub as zip file filters then stores as list any files matching cover_regx
+
+
+
+
+
◆ extract_metadata_epub()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.extract_metadata_epub
+ (
+
+ self ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Return extracted metadata and cover picture
+book['path'] == Full path to ebook file
+book['files'] == list of files from self.process_book(book)
+
+
+
+
+
◆ extract_metadata_mobi()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.extract_metadata_mobi
+ (
+
+ self ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ extract_metadata_pdf()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.extract_metadata_pdf
+ (
+
+ self ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Return extracted metadata
+:NOTES: Retrieval of data has been problematic, some pdf's providing
+reliable titles that corespond with the actual, and others being
+nonsense.
+
+
+
+
+
◆ filter_books()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.filter_books
+ (
+
+ self )
+
+
+
+
+
+
+
◆ import_books()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.import_books
+ (
+
+ self ,
+
+
+
+
+
+ list = None,
+
+
+
+
+ **
+ kwargs
+
+
+
+ )
+
+
+
+
+
Main entry point for import operations.
+Gets a list of new files via compare_shelf_current.
+Iterates over list and inserts new books into database.
+
+
Reimplemented in src.backend.lib.overide.ACatalogue .
+
+
+
+
+
◆ process_by_filetype()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.process_by_filetype
+ (
+
+ self ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Determine books filetype and process.
+
+
+
+
◆ process_epub()
+
+
+
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.process_epub
+ (
+
+ book )
+
+
+
+
+
+static
+
+
+
+
Return dictionary of epub file contents
+
+
+
+
◆ scan_folder()
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.scan_folder
+ (
+
+ self ,
+
+
+
+
+
+ _path = None
+
+
+
+ )
+
+
+
+
+
+
+
◆ stripTags()
+
+
+
+
+
+
+
+
+ def src.backend.lib.library.Catalogue.stripTags
+ (
+
+ source )
+
+
+
+
+
+static
+
+
+
+
+
+
+
+
+
◆ book_folder
+
+
+
+
+
+ src.backend.lib.library.Catalogue.book_folder
+
+
+
+
+
+
+
+
◆ books
+
+
+
+
+
+ src.backend.lib.library.Catalogue.books
+
+
+
+
+
+
+
+
◆ config
+
+
+
+
+
+ src.backend.lib.library.Catalogue.config
+
+
+
+
+
+
+
+
◆ cover_regx
+
+
+
+
+
+ src.backend.lib.library.Catalogue.cover_regx
+
+
+
+
+
+
+
+
◆ db_pointer
+
+
+
+
+
+ src.backend.lib.library.Catalogue.db_pointer
+
+
+
+
+
+
+
+
◆ file_list
+
+
+
+
+
+ src.backend.lib.library.Catalogue.file_list
+
+
+
+
+
+
+
+
◆ html_regx
+
+
+
+
+
+ src.backend.lib.library.Catalogue.html_regx
+
+
+
+
+
+
+
+
◆ opf_regx
+
+
+
+
+
+ src.backend.lib.library.Catalogue.opf_regx
+
+
+
+
+
+
+
+
◆ root_dir
+
+
+
+
+
+ src.backend.lib.library.Catalogue.root_dir
+
+
+
+
+
+
+
+
◆ title_sanitization_dirs_regx
+
+
+
+
+
+ src.backend.lib.library.Catalogue.title_sanitization_dirs_regx
+
+
+
+
+
+
+
+
◆ title_sanitization_lvl2_regx
+
+
+
+
+
+ src.backend.lib.library.Catalogue.title_sanitization_lvl2_regx
+
+
+
+
+
+
+
+
◆ title_sanitization_regx
+
+
+
+
+
+ src.backend.lib.library.Catalogue.title_sanitization_regx
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.png b/docs/pyShelf/html/d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.png
new file mode 100644
index 0000000..84c2afe
Binary files /dev/null and b/docs/pyShelf/html/d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.png differ
diff --git a/docs/pyShelf/html/d5/ddf/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config-members.html b/docs/pyShelf/html/d5/ddf/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config-members.html
new file mode 100644
index 0000000..161a9a6
--- /dev/null
+++ b/docs/pyShelf/html/d5/ddf/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config-members.html
@@ -0,0 +1,67 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.tests.library_test.Test_Config , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d5/dee/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig-members.html b/docs/pyShelf/html/d5/dee/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig-members.html
new file mode 100644
index 0000000..4b98d4d
--- /dev/null
+++ b/docs/pyShelf/html/d5/dee/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig-members.html
@@ -0,0 +1,69 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.tests.config_test.TestConfig , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d6/d1d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin-members.html b/docs/pyShelf/html/d6/d1d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin-members.html
new file mode 100644
index 0000000..5140805
--- /dev/null
+++ b/docs/pyShelf/html/d6/d1d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin-members.html
@@ -0,0 +1,69 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+src backend lib pyShelf Admin
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.pyShelf.Admin , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/d6/d20/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue-members.html b/docs/pyShelf/html/d6/d20/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue-members.html
new file mode 100644
index 0000000..f37827e
--- /dev/null
+++ b/docs/pyShelf/html/d6/d20/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue-members.html
@@ -0,0 +1,91 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.library.Catalogue , including all inherited members.
+
+ __init__ (self, config) (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ book_folder (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ books (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ compare_shelf_current (self)src.backend.lib.library.Catalogue
+ config (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ cover_regx (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ db_pointer (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ extract_content (self, book_zip, book)src.backend.lib.library.Catalogue
+ extract_cover_html (self, book_zip, book)src.backend.lib.library.Catalogue
+ extract_cover_image (self, book_zip, book)src.backend.lib.library.Catalogue
+ extract_metadata_epub (self, book)src.backend.lib.library.Catalogue
+ extract_metadata_mobi (self, book) (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ extract_metadata_pdf (self, book)src.backend.lib.library.Catalogue
+ file_list (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ filter_books (self)src.backend.lib.library.Catalogue
+ html_regx (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ import_books (self, list=None, **kwargs)src.backend.lib.library.Catalogue
+ opf_regx (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ process_by_filetype (self, book)src.backend.lib.library.Catalogue
+ process_epub (book)src.backend.lib.library.Catalogue static
+ root_dir (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ scan_folder (self, _path=None)src.backend.lib.library.Catalogue
+ stripTags (source) (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue static
+ title_sanitization_dirs_regx (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ title_sanitization_lvl2_regx (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+ title_sanitization_regx (defined in src.backend.lib.library.Catalogue )src.backend.lib.library.Catalogue
+
+
+
+
+
diff --git a/docs/pyShelf/html/d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.html b/docs/pyShelf/html/d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.html
new file mode 100644
index 0000000..7cee714
--- /dev/null
+++ b/docs/pyShelf/html/d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.html
@@ -0,0 +1,419 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.models.Book Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
◆ author
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.author [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ book_id
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.book_id = mapped_column(primary_key=True, nullable=False)
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ categories
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.categories [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ cover
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.cover [Optional[bytes]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ date
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.date [timestamp]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ description
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.description [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ file_name
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.file_name [str]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ identifier
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.identifier [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ pages
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.pages [Optional[int]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ progress
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.progress [Optional[float]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ publisher
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.publisher [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ rights
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.rights [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ tags
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.tags [Optional[str]]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ title
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Book.title [str]
+
+
+
+
+static
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.png b/docs/pyShelf/html/d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.png
new file mode 100644
index 0000000..3083706
Binary files /dev/null and b/docs/pyShelf/html/d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.png differ
diff --git a/docs/pyShelf/html/d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.html b/docs/pyShelf/html/d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.html
new file mode 100644
index 0000000..cc02630
--- /dev/null
+++ b/docs/pyShelf/html/d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.html
@@ -0,0 +1,76 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.models.Base Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Base class for all models. The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.png b/docs/pyShelf/html/d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.png
new file mode 100644
index 0000000..793e09f
Binary files /dev/null and b/docs/pyShelf/html/d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.png differ
diff --git a/docs/pyShelf/html/d7/d40/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage.html b/docs/pyShelf/html/d7/d40/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage.html
new file mode 100644
index 0000000..a12bedc
--- /dev/null
+++ b/docs/pyShelf/html/d7/d40/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage.html
@@ -0,0 +1,493 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.storage.Storage Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Create a new Storage object.
+
+>>> db = Storage(config)
+
+Parameters
+----------
+config : Config()
+ Main program configuration.
+
+Attributes
+----------
+config : Stores configuration
+sql : Database Name
+user : Database User Name
+password : Database Password
+db_host : Database Host
+db_port : Database Port
+engine : sqlalchemy.create_engine(url, executor, kw)
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.__init__
+ (
+
+ self ,
+
+
+
+
+
+ config
+
+
+
+ )
+
+
+
+
+
Initialize storage object.
+
+
+
+
+
◆ book_paths_list()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.book_paths_list
+ (
+
+ self )
+
+
+
+
+
Get file paths from database for comparison to system files.
+
+Returns
+-------
+_result : ScalarResult Object
+
+
+
+
+
◆ create_tables()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.create_tables
+ (
+
+ self )
+
+
+
+
+
Create table structure.
+
+
+
+
◆ get_book()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.get_book
+ (
+
+ self ,
+
+
+
+
+
+ book_id
+
+
+
+ )
+
+
+
+
+
Get book from database.
+
+Parameters
+----------
+book_id : int
+ Book ID to filter by.
+
+Returns
+-------
+_result : ScalarResult Object
+
+
+
+
+
◆ get_books()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.get_books
+ (
+
+ self ,
+
+
+
+
+
+ collection = None,
+
+
+
+
+
+ skip = None,
+
+
+
+
+
+ limit = None
+
+
+
+ )
+
+
+
+
+
Get books from database.
+
+Parameters
+----------
+collection : str
+ Collection to filter by.
+
+Returns
+-------
+_result : ScalarResult Object
+
+
+
+
+
◆ get_collections()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.get_collections
+ (
+
+ self )
+
+
+
+
+
Get collections from database.
+
+Returns
+-------
+_result : ScalarResult Object
+
+
+
+
+
◆ get_connection_string()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.get_connection_string
+ (
+
+ self )
+
+
+
+
+
Get connection string.
+
+Engine type references config.json:DB_ENGINE.
+
+Returns
+-------
+str : sqlalchemy Connection String
+
+
+
+
+
◆ insert_book()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.insert_book
+ (
+
+ self ,
+
+
+
+
+
+ book
+
+
+
+ )
+
+
+
+
+
Insert a new book into the database.
+
+Parameters
+----------
+book: dict()
+ Book object to insert.
+
+Returns
+-------
+bool
+ True on success False on failure
+
+
+
+
+
◆ make_collections()
+
+
+
+
+
+ def src.backend.lib.storage.Storage.make_collections
+ (
+
+ self )
+
+
+
+
+
Parse book path's to determine common folder structure.
+
+Stores collections based on shared paths.
+
+
+
+
+
+
◆ config
+
+
+
+
+
+ src.backend.lib.storage.Storage.config
+
+
+
+
+
+
+
+
◆ db_host
+
+
+
+
+
+ src.backend.lib.storage.Storage.db_host
+
+
+
+
+
+
+
+
◆ db_port
+
+
+
+
+
+ src.backend.lib.storage.Storage.db_port
+
+
+
+
+
+
+
+
◆ engine
+
+
+
+
+
+ src.backend.lib.storage.Storage.engine
+
+
+
+
+
+
+
+
◆ password
+
+
+
+
+
+ src.backend.lib.storage.Storage.password
+
+
+
+
+
+
+
+
◆ sql
+
+
+
+
+
+ src.backend.lib.storage.Storage.sql
+
+
+
+
+
+
+
+
◆ user
+
+
+
+
+
+ src.backend.lib.storage.Storage.user
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d7/d88/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo.html b/docs/pyShelf/html/d7/d88/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo.html
new file mode 100644
index 0000000..ca708bb
--- /dev/null
+++ b/docs/pyShelf/html/d7/d88/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo.html
@@ -0,0 +1,194 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.api_hooks.DuckDuckGo Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
duckduckgo related searching
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.api_hooks.DuckDuckGo.__init__
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
+
◆ description_result()
+
+
+
+
+
+ def src.backend.lib.api_hooks.DuckDuckGo.description_result
+ (
+
+ self ,
+
+
+
+
+
+ query
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ image_result()
+
+
+
+
+
+ def src.backend.lib.api_hooks.DuckDuckGo.image_result
+ (
+
+ self ,
+
+
+
+
+
+ query
+
+
+
+ )
+
+
+
+
+
Returns json containing url to image
+:param _key: &t=h_&iar=images&iax=images&ia=images&format=json&pretty=1
+
+
+
+
+
+
◆ imageurl
+
+
+
+
+
+ src.backend.lib.api_hooks.DuckDuckGo.imageurl
+
+
+
+
+
+
+
+
◆ url
+
+
+
+
+
+ src.backend.lib.api_hooks.DuckDuckGo.url
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d7/db9/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server.html b/docs/pyShelf/html/d7/db9/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server.html
new file mode 100644
index 0000000..361dc60
--- /dev/null
+++ b/docs/pyShelf/html/d7/db9/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server.html
@@ -0,0 +1,344 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.pyShelf.Server Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Main Server Container
+:TODO: Document this
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.__init__
+ (
+
+ self ,
+
+
+
+
+
+ root
+
+
+
+ )
+
+
+
+
+
+
+
+
+
+
◆ __aexit__()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.__aexit__
+ (
+
+ self ,
+
+
+
+
+ *
+ args ,
+
+
+
+
+ **
+ kwargs
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ initialize_server()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.initialize_server
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
◆ pong()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.pong
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
◆ runImport()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.runImport
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
◆ socketio()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.socketio
+ (
+
+ self ,
+
+
+
+
+
+ websocket ,
+
+
+
+
+
+ path
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ start()
+
+
+
+
+
+ def src.backend.lib.pyShelf.Server.start
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
+
◆ config
+
+
+
+
+
+ src.backend.lib.pyShelf.Server.config
+
+
+
+
+
+
+
+
◆ host
+
+
+
+
+
+ src.backend.lib.pyShelf.Server.host
+
+
+
+
+
+
+
+
◆ loop
+
+
+
+
+
+ src.backend.lib.pyShelf.Server.loop
+
+
+
+
+
+
+
+
◆ root
+
+
+
+
+
+ src.backend.lib.pyShelf.Server.root
+
+
+
+
+
+
+
+
◆ serve
+
+
+
+
+
+ src.backend.lib.pyShelf.Server.serve
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d7/df9/namespaceDataHooks.html b/docs/pyShelf/html/d7/df9/namespaceDataHooks.html
new file mode 100644
index 0000000..e4d711b
--- /dev/null
+++ b/docs/pyShelf/html/d7/df9/namespaceDataHooks.html
@@ -0,0 +1,65 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: DataHooks Namespace Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
pyShelf's frontend database hooks.
+
+
+
+
diff --git a/docs/pyShelf/html/d8/d3d/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue.html b/docs/pyShelf/html/d8/d3d/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue.html
new file mode 100644
index 0000000..75964aa
--- /dev/null
+++ b/docs/pyShelf/html/d8/d3d/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue.html
@@ -0,0 +1,147 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.tests.library_test.TestCatalogue Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
◆ test_filter_books()
+
+
+
+
+
+ def src.backend.tests.library_test.TestCatalogue.test_filter_books
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
+
◆ config
+
+
+
+
◆ root
+
+
+
+
+
+
+
+
+ os src.backend.tests.library_test.TestCatalogue.root = os.path.abspath(os.path.curdir)
+
+
+
+
+static
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d8/def/classobjects_1_1JSInterface.html b/docs/pyShelf/html/d8/def/classobjects_1_1JSInterface.html
new file mode 100644
index 0000000..fc91435
--- /dev/null
+++ b/docs/pyShelf/html/d8/def/classobjects_1_1JSInterface.html
@@ -0,0 +1,125 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: objects.JSInterface Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
A class to interface with the JavaScript side of pyShelf.
+
+
◆ __init__()
+
+
+
+
+
+ def objects.JSInterface.__init__
+ (
+
+ self ,
+
+
+
+
+ Config
+ config
+
+
+
+ )
+
+
+
+
+
Initialize the JSInterface object.
+
+
+
+
+
◆ install()
+
+
+
+
+
+ def objects.JSInterface.install
+ (
+
+ self )
+
+
+
+
+
Install the JavaScript dependencies.
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.html b/docs/pyShelf/html/d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.html
new file mode 100644
index 0000000..05d1e26
--- /dev/null
+++ b/docs/pyShelf/html/d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.html
@@ -0,0 +1,601 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.config.Config Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Main System Configuration.
+
+>>> config = Config(root)
+
+Parameters
+----------
+root : File system root of program
+
+Attributes
+----------
+root : str() stores root.
+config_structure : dict() Default Configuration Structure.
+_fp : str() file pointer to main configuration.
+_cp : Path() object of configuration file.
+_data : dict() parsed json of _fp.
+logger : holds logging configuration from get_logger().
+book_path : directory pointer to main books folder.
+TITLE : str() Program title.
+VERSION : str() Program version.
+TITLE : str() Combines TITLE & VERSION.
+book_shelf : Deprecation TODO: Is this still in use?
+catalogue_db : str() Database Name.
+user : str() Database user name.
+password : str() Database password.
+db_host : str() Database host.
+db_port : int() Database port.
+file_array : list() copy of book_shelf TODO: See book_shelf
+auto_scan: bool() Do we auto scan on launch?
+allowed_hosts : list() Allowed host list.
+db_engine : str() Desired database engine type.
+db_user : str() Database user name. Duplication Warning.
+db_pass : str() Database password. Duplication Warning.
+build_mode : str() Production | Development mode.
+
+Methods
+-------
+get_logger : Setup loguru.
+open_file : Parse configuration file.
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.config.Config.__init__
+ (
+
+ self ,
+
+
+
+
+
+ root
+
+
+
+ )
+
+
+
+
+
+
+
+
◆ get_logger()
+
+
+
+
+
+ def src.backend.lib.config.Config.get_logger
+ (
+
+ self )
+
+
+
+
+
Instantiate logging system.
+
+
+
+
◆ init_config()
+
+
+
+
+
+ def src.backend.lib.config.Config.init_config
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
◆ open_file()
+
+
+
+
+
+ def src.backend.lib.config.Config.open_file
+ (
+
+ self )
+
+
+
+
+
+
+
+
◆ _cp
+
+
+
+
+
+
+
+
+ src.backend.lib.config.Config._cp
+
+
+
+
+protected
+
+
+
+
+
+
+
+
◆ _data
+
+
+
+
+
+
+
+
+ src.backend.lib.config.Config._data
+
+
+
+
+protected
+
+
+
+
+
+
+
+
◆ _fp
+
+
+
+
+
+
+
+
+ src.backend.lib.config.Config._fp
+
+
+
+
+protected
+
+
+
+
+
+
+
+
◆ allowed_hosts
+
+
+
+
+
+ src.backend.lib.config.Config.allowed_hosts
+
+
+
+
+
+
+
+
◆ auto_scan
+
+
+
+
+
+ src.backend.lib.config.Config.auto_scan
+
+
+
+
+
+
+
+
◆ book_path
+
+
+
+
+
+ src.backend.lib.config.Config.book_path
+
+
+
+
+
+
+
+
◆ book_shelf
+
+
+
+
+
+ src.backend.lib.config.Config.book_shelf
+
+
+
+
+
+
+
+
◆ build_mode
+
+
+
+
+
+ src.backend.lib.config.Config.build_mode
+
+
+
+
+
+
+
+
◆ catalogue_db
+
+
+
+
+
+ src.backend.lib.config.Config.catalogue_db
+
+
+
+
+
+
+
+
◆ config_structure
+
+
+
+
+
+ src.backend.lib.config.Config.config_structure
+
+
+
+
+
+
+
+
◆ db_engine
+
+
+
+
+
+ src.backend.lib.config.Config.db_engine
+
+
+
+
+
+
+
+
◆ db_host
+
+
+
+
+
+ src.backend.lib.config.Config.db_host
+
+
+
+
+
+
+
+
◆ db_pass
+
+
+
+
+
+ src.backend.lib.config.Config.db_pass
+
+
+
+
+
+
+
+
◆ db_port
+
+
+
+
+
+ src.backend.lib.config.Config.db_port
+
+
+
+
+
+
+
+
◆ db_user
+
+
+
+
+
+ src.backend.lib.config.Config.db_user
+
+
+
+
+
+
+
+
◆ file_array
+
+
+
+
+
+ src.backend.lib.config.Config.file_array
+
+
+
+
+
+
+
+
◆ logger
+
+
+
+
+
+ src.backend.lib.config.Config.logger
+
+
+
+
+
+
+
+
◆ password
+
+
+
+
+
+ src.backend.lib.config.Config.password
+
+
+
+
+
+
+
+
◆ root
+
+
+
+
+
+ src.backend.lib.config.Config.root
+
+
+
+
+
+
+
+
◆ TITLE
+
+
+
+
+
+ src.backend.lib.config.Config.TITLE
+
+
+
+
+
+
+
+
◆ user
+
+
+
+
+
+ src.backend.lib.config.Config.user
+
+
+
+
+
+
+
+
◆ VERSION
+
+
+
+
+
+ src.backend.lib.config.Config.VERSION
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.png b/docs/pyShelf/html/d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.png
new file mode 100644
index 0000000..c125235
Binary files /dev/null and b/docs/pyShelf/html/d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.png differ
diff --git a/docs/pyShelf/html/d9/d71/classDataHooks_1_1BookInterface.html b/docs/pyShelf/html/d9/d71/classDataHooks_1_1BookInterface.html
new file mode 100644
index 0000000..7de35ed
--- /dev/null
+++ b/docs/pyShelf/html/d9/d71/classDataHooks_1_1BookInterface.html
@@ -0,0 +1,106 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: DataHooks.BookInterface Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Access point for book database.
+
+
◆ config
+
+
+
+
+
+ DataHooks.BookInterface.config
+
+
+
+
+
+
+
+
◆ db
+
+
+
+
+
+ DataHooks.BookInterface.db
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/d9/d8b/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue-members.html b/docs/pyShelf/html/d9/d8b/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue-members.html
new file mode 100644
index 0000000..72fd565
--- /dev/null
+++ b/docs/pyShelf/html/d9/d8b/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue-members.html
@@ -0,0 +1,68 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.overide.ACatalogue , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/da/d3c/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles.html b/docs/pyShelf/html/da/d3c/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles.html
new file mode 100644
index 0000000..879722f
--- /dev/null
+++ b/docs/pyShelf/html/da/d3c/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles.html
@@ -0,0 +1,136 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.pyShelf.InitFiles Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
First run file creation operations
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.pyShelf.InitFiles.__init__
+ (
+
+ self ,
+
+
+
+
+
+ file_array
+
+
+
+ )
+
+
+
+
+
+
+
+
+
+
◆ CreateFile()
+
+
+
+
+
+ def src.backend.lib.pyShelf.InitFiles.CreateFile
+ (
+
+ self ,
+
+
+
+
+
+ _pointer
+
+
+
+ )
+
+
+
+
+
Checks if file exists and creates it if not
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/da/dc1/classsrc_1_1backend_1_1lib_1_1models_1_1Book-members.html b/docs/pyShelf/html/da/dc1/classsrc_1_1backend_1_1lib_1_1models_1_1Book-members.html
new file mode 100644
index 0000000..c86a9d8
--- /dev/null
+++ b/docs/pyShelf/html/da/dc1/classsrc_1_1backend_1_1lib_1_1models_1_1Book-members.html
@@ -0,0 +1,79 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.models.Book , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/db/d9a/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage-members.html b/docs/pyShelf/html/db/d9a/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage-members.html
new file mode 100644
index 0000000..c4ecebb
--- /dev/null
+++ b/docs/pyShelf/html/db/d9a/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage-members.html
@@ -0,0 +1,81 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.storage.Storage , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/db/dd5/namespaceFastAPIServer.html b/docs/pyShelf/html/db/dd5/namespaceFastAPIServer.html
new file mode 100644
index 0000000..ddbae11
--- /dev/null
+++ b/docs/pyShelf/html/db/dd5/namespaceFastAPIServer.html
@@ -0,0 +1,313 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: FastAPIServer Namespace Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
pyShelf's main frontend library.
+
+
◆ base64decode()
+
+
+
+
+
+ str FastAPIServer.base64decode
+ (
+
+ string )
+
+
+
+
+
Decode a base64 string.
+
+
+
+
◆ book_tojson()
+
+
+
+
+
+ dumps FastAPIServer.book_tojson
+ (
+
+ book )
+
+
+
+
+
Convert a book object to a json.
+
+
+
+
◆ books_tojson()
+
+
+
+
+
+ dumps FastAPIServer.books_tojson
+ (
+
+ obj )
+
+
+
+
+
Convert an object to a dictionary.
+
+
+
+
◆ collections_tojson()
+
+
+
+
+
+ dumps FastAPIServer.collections_tojson
+ (
+
+ collection )
+
+
+
+
+
Convert a collections object to json.
+
+
+
+
◆ convertDateTime()
+
+
+
+
+
+ str FastAPIServer.convertDateTime
+ (
+ datetime
+ timestamp )
+
+
+
+
+
Convert a datetime object to a string.
+
+
+
+
◆ summarize()
+
+
+
+
+
+ str FastAPIServer.summarize
+ (
+
+ string )
+
+
+
+
+
+
+
+
◆ allow_credentials
+
+
+
+
+
+ FastAPIServer.allow_credentials
+
+
+
+
+
+
+
+
◆ allow_headers
+
+
+
+
+
+ FastAPIServer.allow_headers
+
+
+
+
+
+
+
+
◆ allow_methods
+
+
+
+
+
+ FastAPIServer.allow_methods
+
+
+
+
+
+
+
+
◆ allow_origins
+
+
+
+
+
+ FastAPIServer.allow_origins
+
+
+
+
+
+
+
+
◆ app
+
+
+
+
+
+ FastAPI FastAPIServer.app = FastAPI()
+
+
+
+
+
+
+
+
◆ origins
+
+
+
+
+
+ list FastAPIServer.origins
+
+
+
+
Initial value: 1 = [
+
2 "http://localhost" ,
+
3 "http://localhost:8081" ,
+
4 "http://localhost:8080" ,
+
5 "*"
+
6 ]
+
+
+
+
+
◆ templates
+
+
+
+
+
+ Jinja2Templates FastAPIServer.templates = Jinja2Templates(directory="src/frontend/templates")
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.html b/docs/pyShelf/html/db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.html
new file mode 100644
index 0000000..ecbef76
--- /dev/null
+++ b/docs/pyShelf/html/db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.html
@@ -0,0 +1,153 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.tests.library_test.Test_Catalogue Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.tests.library_test.Test_Catalogue.__init__
+ (
+
+ self )
+
+
+
+
+
+
+
+
◆ filter_books()
+
+
+
+
+
+ def src.backend.tests.library_test.Test_Catalogue.filter_books
+ (
+
+ self )
+
+
+
+
+
Calls scan_folder and filters out book files.
+
+:returns self._book_list_expanded: json string containing
+all book metadata
+
+
Reimplemented from src.backend.lib.library.Catalogue .
+
+
+
+
+
+
◆ book_shelf
+
+
+
+
+
+ src.backend.tests.library_test.Test_Catalogue.book_shelf
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.png b/docs/pyShelf/html/db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.png
new file mode 100644
index 0000000..3fd03ee
Binary files /dev/null and b/docs/pyShelf/html/db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.png differ
diff --git a/docs/pyShelf/html/dc/d3e/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue-members.html b/docs/pyShelf/html/dc/d3e/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue-members.html
new file mode 100644
index 0000000..841856a
--- /dev/null
+++ b/docs/pyShelf/html/dc/d3e/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue-members.html
@@ -0,0 +1,68 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.tests.library_test.Test_Catalogue , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.html b/docs/pyShelf/html/dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.html
new file mode 100644
index 0000000..0a9c379
--- /dev/null
+++ b/docs/pyShelf/html/dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.html
@@ -0,0 +1,174 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.overide.ACatalogue Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Aynchronous overide of library.Catalogue,
+: TODO : Complete or discard this overide
+
+
+
◆ __init__()
+
+
+
+
+
+ def src.backend.lib.overide.ACatalogue.__init__
+ (
+
+ self )
+
+
+
+
+
+
+
+
◆ import_books()
+
+
+
+
+
+ def src.backend.lib.overide.ACatalogue.import_books
+ (
+
+ self ,
+
+
+
+
+ **
+ kwargs
+
+
+
+ )
+
+
+
+
+
+
+
◆ scan_folder()
+
+
+
+
+
+ def src.backend.lib.overide.ACatalogue.scan_folder
+ (
+
+ self ,
+
+
+
+
+
+ _path = None
+
+
+
+ )
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.png b/docs/pyShelf/html/dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.png
new file mode 100644
index 0000000..efb6c65
Binary files /dev/null and b/docs/pyShelf/html/dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.png differ
diff --git a/docs/pyShelf/html/dd/d22/classFastAPIServer_1_1FastAPIServer.html b/docs/pyShelf/html/dd/d22/classFastAPIServer_1_1FastAPIServer.html
new file mode 100644
index 0000000..ae2670b
--- /dev/null
+++ b/docs/pyShelf/html/dd/d22/classFastAPIServer_1_1FastAPIServer.html
@@ -0,0 +1,356 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: FastAPIServer.FastAPIServer Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Entry point for FastAPI server.
+
+
◆ __init__()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.__init__
+ (
+
+ self ,
+
+
+
+
+
+ config
+
+
+
+ )
+
+
+
+
+
Initialize FastAPIServer object parameters.
+
+
+
+
+
◆ book()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.book
+ (
+ Request
+ request ,
+
+
+
+
+ int
+ book_id
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ books()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.books
+ (
+ Request
+ request ,
+
+
+
+
+ int
+ skip = 0,
+
+
+
+
+ int
+ limit = 10,
+
+
+
+
+
+ collection = None
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ collections()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.collections
+ (
+ Request
+ request )
+
+
+
+
+
+
+
+
+
◆ compile_static_files()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.compile_static_files
+ (
+
+ self )
+
+
+
+
+
Compile static files for web frontend.
+
+
+
+
◆ index()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.index
+ (
+ Request
+ request ,
+
+
+
+
+ int
+ skip = 0,
+
+
+
+
+ int
+ limit = 10
+
+
+
+ )
+
+
+
+
+
+
+
+
+
◆ run()
+
+
+
+
+
+ def FastAPIServer.FastAPIServer.run
+ (
+
+ self )
+
+
+
+
+
Front end server entrypoint.
+
+
+
+
◆ use_route_names_as_operation_ids()
+
+
+
+
+
+ None FastAPIServer.FastAPIServer.use_route_names_as_operation_ids
+ (
+
+ self ,
+
+
+
+
+ FastAPI
+ app
+
+
+
+ )
+
+
+
+
+
Use route name as operation id.
+
+
+
+
+
◆ config
+
+
+
+
+
+ FastAPIServer.FastAPIServer.config
+
+
+
+
+
+
+
+
◆ fe_config
+
+
+
+
+
+ FastAPIServer.FastAPIServer.fe_config
+
+
+
+
+
+
+
+
◆ fe_server
+
+
+
+
+
+ FastAPIServer.FastAPIServer.fe_server
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/dd/de1/namespacesrc_1_1backend_1_1lib_1_1storage.html b/docs/pyShelf/html/dd/de1/namespacesrc_1_1backend_1_1lib_1_1storage.html
new file mode 100644
index 0000000..4affba5
--- /dev/null
+++ b/docs/pyShelf/html/dd/de1/namespacesrc_1_1backend_1_1lib_1_1storage.html
@@ -0,0 +1,69 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.storage Namespace Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Pyshelf's Main Storage Class.
+
+
+
+
diff --git a/docs/pyShelf/html/de/d11/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo-members.html b/docs/pyShelf/html/de/d11/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo-members.html
new file mode 100644
index 0000000..86b19f6
--- /dev/null
+++ b/docs/pyShelf/html/de/d11/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo-members.html
@@ -0,0 +1,70 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.api_hooks.DuckDuckGo , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/de/d6d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server-members.html b/docs/pyShelf/html/de/d6d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server-members.html
new file mode 100644
index 0000000..1cf062d
--- /dev/null
+++ b/docs/pyShelf/html/de/d6d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server-members.html
@@ -0,0 +1,77 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Member List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
This is the complete list of members for src.backend.lib.pyShelf.Server , including all inherited members.
+
+
+
+
+
diff --git a/docs/pyShelf/html/df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.html b/docs/pyShelf/html/df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.html
new file mode 100644
index 0000000..b945c7a
--- /dev/null
+++ b/docs/pyShelf/html/df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.html
@@ -0,0 +1,155 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.lib.models.Collection Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
◆ book_id
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Collection.book_id = mapped_column(ForeignKey(Book.book_id))
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ collection
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Collection.collection [str]
+
+
+
+
+static
+
+
+
+
+
+
+
+
◆ collection_id
+
+
+
+
+
+
+
+
+ Mapped src.backend.lib.models.Collection.collection_id = mapped_column(primary_key=True)
+
+
+
+
+static
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.png b/docs/pyShelf/html/df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.png
new file mode 100644
index 0000000..7475422
Binary files /dev/null and b/docs/pyShelf/html/df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.png differ
diff --git a/docs/pyShelf/html/df/d4c/namespaceobjects.html b/docs/pyShelf/html/df/d4c/namespaceobjects.html
new file mode 100644
index 0000000..f469771
--- /dev/null
+++ b/docs/pyShelf/html/df/d4c/namespaceobjects.html
@@ -0,0 +1,65 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: objects Namespace Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
pyShelf's Frontend Objects.
+
+
+
+
diff --git a/docs/pyShelf/html/df/d83/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig.html b/docs/pyShelf/html/df/d83/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig.html
new file mode 100644
index 0000000..7eba542
--- /dev/null
+++ b/docs/pyShelf/html/df/d83/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig.html
@@ -0,0 +1,163 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src.backend.tests.config_test.TestConfig Class Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
◆ test_book_dir()
+
+
+
+
+
+ def src.backend.tests.config_test.TestConfig.test_book_dir
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
◆ test_title()
+
+
+
+
+
+ def src.backend.tests.config_test.TestConfig.test_title
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
◆ test_version()
+
+
+
+
+
+ def src.backend.tests.config_test.TestConfig.test_version
+ (
+
+ self )
+
+
+
+
+
+
+
+
+
+
◆ config
+
+
+
+
+
+
+
+
+ Config src.backend.tests.config_test.TestConfig.config = Config (os.path.abspath(os.path.curdir))
+
+
+
+
+static
+
+
+
+
+
+
+
The documentation for this class was generated from the following file:
+
+
+
+
+
diff --git a/docs/pyShelf/html/dir_1c7294442ce83d360610449d04c9d7fc.html b/docs/pyShelf/html/dir_1c7294442ce83d360610449d04c9d7fc.html
new file mode 100644
index 0000000..5109080
--- /dev/null
+++ b/docs/pyShelf/html/dir_1c7294442ce83d360610449d04c9d7fc.html
@@ -0,0 +1,79 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src/backend/lib Directory Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/dir_3f615ade772d23cefe5e20dcb18424a2.html b/docs/pyShelf/html/dir_3f615ade772d23cefe5e20dcb18424a2.html
new file mode 100644
index 0000000..e9c02c9
--- /dev/null
+++ b/docs/pyShelf/html/dir_3f615ade772d23cefe5e20dcb18424a2.html
@@ -0,0 +1,76 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src/backend Directory Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+file __init__.py [code]
+
+file pyShelf_MakeCollections.py [code]
+
+file pyShelf_ScanLibrary.py [code]
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/dir_68267d1309a1af8e8297ef4c3efbcdba.html b/docs/pyShelf/html/dir_68267d1309a1af8e8297ef4c3efbcdba.html
new file mode 100644
index 0000000..4431a4e
--- /dev/null
+++ b/docs/pyShelf/html/dir_68267d1309a1af8e8297ef4c3efbcdba.html
@@ -0,0 +1,70 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src Directory Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+file __init__.py [code]
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/dir_8509b06bafa958d2011203a730bf5778.html b/docs/pyShelf/html/dir_8509b06bafa958d2011203a730bf5778.html
new file mode 100644
index 0000000..af62fcc
--- /dev/null
+++ b/docs/pyShelf/html/dir_8509b06bafa958d2011203a730bf5778.html
@@ -0,0 +1,65 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src/frontend Directory Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/dir_9020577e2da81c4cf5a5554c26431f94.html b/docs/pyShelf/html/dir_9020577e2da81c4cf5a5554c26431f94.html
new file mode 100644
index 0000000..7dee3c2
--- /dev/null
+++ b/docs/pyShelf/html/dir_9020577e2da81c4cf5a5554c26431f94.html
@@ -0,0 +1,71 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src/backend/tests Directory Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+file __init__.py [code]
+
+file config_test.py [code]
+
+file library_test.py [code]
+
+file server_test.py [code]
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/dir_d6d65651a873fbc3232a07f3f4a594e4.html b/docs/pyShelf/html/dir_d6d65651a873fbc3232a07f3f4a594e4.html
new file mode 100644
index 0000000..dc608e4
--- /dev/null
+++ b/docs/pyShelf/html/dir_d6d65651a873fbc3232a07f3f4a594e4.html
@@ -0,0 +1,69 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: src/frontend/lib Directory Reference
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/doc.png b/docs/pyShelf/html/doc.png
new file mode 100644
index 0000000..17edabf
Binary files /dev/null and b/docs/pyShelf/html/doc.png differ
diff --git a/docs/pyShelf/html/docd.png b/docs/pyShelf/html/docd.png
new file mode 100644
index 0000000..d7c94fd
Binary files /dev/null and b/docs/pyShelf/html/docd.png differ
diff --git a/docs/pyShelf/html/doxygen.svg b/docs/pyShelf/html/doxygen.svg
new file mode 100644
index 0000000..d42dad5
--- /dev/null
+++ b/docs/pyShelf/html/doxygen.svg
@@ -0,0 +1,26 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/dynsections.js b/docs/pyShelf/html/dynsections.js
new file mode 100644
index 0000000..1f4cd14
--- /dev/null
+++ b/docs/pyShelf/html/dynsections.js
@@ -0,0 +1,130 @@
+/*
+ @licstart The following is the entire license notice for the JavaScript code in this file.
+
+ The MIT License (MIT)
+
+ Copyright (C) 1997-2020 by Dimitri van Heesch
+
+ Permission is hereby granted, free of charge, to any person obtaining a copy of this software
+ and associated documentation files (the "Software"), to deal in the Software without restriction,
+ including without limitation the rights to use, copy, modify, merge, publish, distribute,
+ sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
+ furnished to do so, subject to the following conditions:
+
+ The above copyright notice and this permission notice shall be included in all copies or
+ substantial portions of the Software.
+
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
+ BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
+ DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+
+ @licend The above is the entire license notice for the JavaScript code in this file
+ */
+function toggleVisibility(linkObj)
+{
+ var base = $(linkObj).attr('id');
+ var summary = $('#'+base+'-summary');
+ var content = $('#'+base+'-content');
+ var trigger = $('#'+base+'-trigger');
+ var src=$(trigger).attr('src');
+ if (content.is(':visible')===true) {
+ content.hide();
+ summary.show();
+ $(linkObj).addClass('closed').removeClass('opened');
+ $(trigger).attr('src',src.substring(0,src.length-8)+'closed.png');
+ } else {
+ content.show();
+ summary.hide();
+ $(linkObj).removeClass('closed').addClass('opened');
+ $(trigger).attr('src',src.substring(0,src.length-10)+'open.png');
+ }
+ return false;
+}
+
+function updateStripes()
+{
+ $('table.directory tr').
+ removeClass('even').filter(':visible:even').addClass('even');
+ $('table.directory tr').
+ removeClass('odd').filter(':visible:odd').addClass('odd');
+}
+
+function toggleLevel(level)
+{
+ $('table.directory tr').each(function() {
+ var l = this.id.split('_').length-1;
+ var i = $('#img'+this.id.substring(3));
+ var a = $('#arr'+this.id.substring(3));
+ if (l
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: File List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
Here is a list of all documented files with brief descriptions:
+
[detail level 1 2 3 4 ]
+ ▼ src
+ ▼ backend
+ ▼ lib
+ __init__.py
+ api_hooks.py
+ config.py
+ library.py
+ models.py
+ overide.py
+ pyShelf.py
+ storage.py
+ ▼ tests
+ __init__.py
+ config_test.py
+ library_test.py
+ server_test.py
+ __init__.py
+ pyShelf_MakeCollections.py
+ pyShelf_ScanLibrary.py
+ ▼ frontend
+ ▼ lib
+ DataHooks.py
+ FastAPIServer.py
+ objects.py
+ __init__.py
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/folderclosed.png b/docs/pyShelf/html/folderclosed.png
new file mode 100644
index 0000000..bb8ab35
Binary files /dev/null and b/docs/pyShelf/html/folderclosed.png differ
diff --git a/docs/pyShelf/html/folderopen.png b/docs/pyShelf/html/folderopen.png
new file mode 100644
index 0000000..d6c7f67
Binary files /dev/null and b/docs/pyShelf/html/folderopen.png differ
diff --git a/docs/pyShelf/html/functions.html b/docs/pyShelf/html/functions.html
new file mode 100644
index 0000000..86bb94e
--- /dev/null
+++ b/docs/pyShelf/html/functions.html
@@ -0,0 +1,158 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Class Members
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Here is a list of all documented class members with links to the class documentation for each member:
+
+
- _ -
+
+
+
- b -
+
+
+
- c -
+
+
+
- e -
+
+
+
- f -
+
+
+
- g -
+
+
+
- i -
+
+
+
- m -
+
+
+
- o -
+
+
+
- p -
+
+
+
- r -
+
+
+
- s -
+
+
+
- u -
+
+
+
+
+
diff --git a/docs/pyShelf/html/functions_func.html b/docs/pyShelf/html/functions_func.html
new file mode 100644
index 0000000..c3ffc27
--- /dev/null
+++ b/docs/pyShelf/html/functions_func.html
@@ -0,0 +1,158 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Class Members - Functions
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
- _ -
+
+
+
- b -
+
+
+
- c -
+
+
+
- e -
+
+
+
- f -
+
+
+
- g -
+
+
+
- i -
+
+
+
- m -
+
+
+
- o -
+
+
+
- p -
+
+
+
- r -
+
+
+
- s -
+
+
+
- u -
+
+
+
+
+
diff --git a/docs/pyShelf/html/hierarchy.html b/docs/pyShelf/html/hierarchy.html
new file mode 100644
index 0000000..d593775
--- /dev/null
+++ b/docs/pyShelf/html/hierarchy.html
@@ -0,0 +1,81 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Class Hierarchy
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
This inheritance list is sorted roughly, but not completely, alphabetically:
+
+
+
+
+
diff --git a/docs/pyShelf/html/index.hhc b/docs/pyShelf/html/index.hhc
new file mode 100644
index 0000000..e77d1ad
--- /dev/null
+++ b/docs/pyShelf/html/index.hhc
@@ -0,0 +1,270 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/index.hhk b/docs/pyShelf/html/index.hhk
new file mode 100644
index 0000000..65dfd6e
--- /dev/null
+++ b/docs/pyShelf/html/index.hhk
@@ -0,0 +1,260 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/index.hhp b/docs/pyShelf/html/index.hhp
new file mode 100644
index 0000000..dd527ef
--- /dev/null
+++ b/docs/pyShelf/html/index.hhp
@@ -0,0 +1,111 @@
+[OPTIONS]
+Compiled file=pyShelf_docs
+Compatibility=1.1
+Full-text search=Yes
+Contents file=index.hhc
+Default Window=main
+Default topic=index.html
+Index file=index.hhk
+Language=0x409 English (United States)
+Binary TOC=YES
+Title=pyShelf Open Source Ebook Server
+
+[WINDOWS]
+main="pyShelf Open Source Ebook Server","index.hhc","index.hhk","index.html","index.html",,,,,0x23520,,0x70387e,,,,,,,,0
+
+[FILES]
+annotated.html
+classes.html
+d0/d49/classsrc_1_1backend_1_1lib_1_1models_1_1Collection-members.html
+d0/d93/namespacesrc_1_1backend_1_1lib_1_1config.html
+d1/dca/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles-members.html
+d2/dcb/classobjects_1_1JSInterface-members.html
+d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.html
+d3/d4a/classsrc_1_1backend_1_1lib_1_1config_1_1Config-members.html
+d3/d95/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue-members.html
+d4/d6b/classFastAPIServer_1_1FastAPIServer-members.html
+d4/d71/classDataHooks_1_1BookInterface-members.html
+d5/d80/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin.html
+d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.html
+d5/ddf/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config-members.html
+d5/dee/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig-members.html
+d6/d1d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Admin-members.html
+d6/d20/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue-members.html
+d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.html
+d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.html
+d7/d40/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage.html
+d7/d88/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo.html
+d7/db9/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server.html
+d7/df9/namespaceDataHooks.html
+d8/d3d/classsrc_1_1backend_1_1tests_1_1library__test_1_1TestCatalogue.html
+d8/def/classobjects_1_1JSInterface.html
+d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.html
+d9/d71/classDataHooks_1_1BookInterface.html
+d9/d8b/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue-members.html
+da/d3c/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1InitFiles.html
+da/dc1/classsrc_1_1backend_1_1lib_1_1models_1_1Book-members.html
+db/d9a/classsrc_1_1backend_1_1lib_1_1storage_1_1Storage-members.html
+db/dd5/namespaceFastAPIServer.html
+db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.html
+dc/d3e/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue-members.html
+dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.html
+dd/d22/classFastAPIServer_1_1FastAPIServer.html
+dd/de1/namespacesrc_1_1backend_1_1lib_1_1storage.html
+de/d11/classsrc_1_1backend_1_1lib_1_1api__hooks_1_1DuckDuckGo-members.html
+de/d6d/classsrc_1_1backend_1_1lib_1_1pyShelf_1_1Server-members.html
+df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.html
+df/d4c/namespaceobjects.html
+df/d83/classsrc_1_1backend_1_1tests_1_1config__test_1_1TestConfig.html
+dir_1c7294442ce83d360610449d04c9d7fc.html
+dir_3f615ade772d23cefe5e20dcb18424a2.html
+dir_68267d1309a1af8e8297ef4c3efbcdba.html
+dir_8509b06bafa958d2011203a730bf5778.html
+dir_9020577e2da81c4cf5a5554c26431f94.html
+dir_d6d65651a873fbc3232a07f3f4a594e4.html
+files.html
+functions.html
+functions_func.html
+hierarchy.html
+index.html
+namespacemembers.html
+namespacemembers_func.html
+namespaces.html
+bc_s.png
+bc_sd.png
+bdwn.png
+closed.png
+d3/d29/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Config.png
+d5/da5/classsrc_1_1backend_1_1lib_1_1library_1_1Catalogue.png
+d6/d4e/classsrc_1_1backend_1_1lib_1_1models_1_1Book.png
+d6/d90/classsrc_1_1backend_1_1lib_1_1models_1_1Base.png
+d8/dfe/classsrc_1_1backend_1_1lib_1_1config_1_1Config.png
+db/dda/classsrc_1_1backend_1_1tests_1_1library__test_1_1Test__Catalogue.png
+dc/ded/classsrc_1_1backend_1_1lib_1_1overide_1_1ACatalogue.png
+df/d03/classsrc_1_1backend_1_1lib_1_1models_1_1Collection.png
+doc.png
+docd.png
+doxygen.svg
+folderclosed.png
+folderopen.png
+nav_f.png
+nav_fd.png
+nav_g.png
+nav_h.png
+nav_hd.png
+open.png
+splitbar.png
+splitbard.png
+sync_off.png
+sync_on.png
+tab_a.png
+tab_ad.png
+tab_b.png
+tab_bd.png
+tab_h.png
+tab_hd.png
+tab_s.png
+tab_sd.png
+doxygen.css
+dynsections.js
+jquery.js
+tabs.css
diff --git a/docs/pyShelf/html/index.html b/docs/pyShelf/html/index.html
new file mode 100644
index 0000000..d142e85
--- /dev/null
+++ b/docs/pyShelf/html/index.html
@@ -0,0 +1,50 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Main Page
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/jquery.js b/docs/pyShelf/html/jquery.js
new file mode 100644
index 0000000..0e61ead
--- /dev/null
+++ b/docs/pyShelf/html/jquery.js
@@ -0,0 +1,34 @@
+/*! jQuery v3.6.0 | (c) OpenJS Foundation and other contributors | jquery.org/license */
+!function(e,t){"use strict";"object"==typeof module&&"object"==typeof module.exports?module.exports=e.document?t(e,!0):function(e){if(!e.document)throw new Error("jQuery requires a window with a document");return t(e)}:t(e)}("undefined"!=typeof window?window:this,function(C,e){"use strict";var t=[],r=Object.getPrototypeOf,s=t.slice,g=t.flat?function(e){return t.flat.call(e)}:function(e){return t.concat.apply([],e)},u=t.push,i=t.indexOf,n={},o=n.toString,v=n.hasOwnProperty,a=v.toString,l=a.call(Object),y={},m=function(e){return"function"==typeof e&&"number"!=typeof e.nodeType&&"function"!=typeof e.item},x=function(e){return null!=e&&e===e.window},E=C.document,c={type:!0,src:!0,nonce:!0,noModule:!0};function b(e,t,n){var r,i,o=(n=n||E).createElement("script");if(o.text=e,t)for(r in c)(i=t[r]||t.getAttribute&&t.getAttribute(r))&&o.setAttribute(r,i);n.head.appendChild(o).parentNode.removeChild(o)}function w(e){return null==e?e+"":"object"==typeof e||"function"==typeof e?n[o.call(e)]||"object":typeof e}var f="3.6.0",S=function(e,t){return new S.fn.init(e,t)};function p(e){var t=!!e&&"length"in e&&e.length,n=w(e);return!m(e)&&!x(e)&&("array"===n||0===t||"number"==typeof t&&0+~]|"+M+")"+M+"*"),U=new RegExp(M+"|>"),X=new RegExp(F),V=new RegExp("^"+I+"$"),G={ID:new RegExp("^#("+I+")"),CLASS:new RegExp("^\\.("+I+")"),TAG:new RegExp("^("+I+"|[*])"),ATTR:new RegExp("^"+W),PSEUDO:new RegExp("^"+F),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+M+"*(even|odd|(([+-]|)(\\d*)n|)"+M+"*(?:([+-]|)"+M+"*(\\d+)|))"+M+"*\\)|)","i"),bool:new RegExp("^(?:"+R+")$","i"),needsContext:new RegExp("^"+M+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+M+"*((?:-\\d)?\\d*)"+M+"*\\)|)(?=[^-]|$)","i")},Y=/HTML$/i,Q=/^(?:input|select|textarea|button)$/i,J=/^h\d$/i,K=/^[^{]+\{\s*\[native \w/,Z=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ee=/[+~]/,te=new RegExp("\\\\[\\da-fA-F]{1,6}"+M+"?|\\\\([^\\r\\n\\f])","g"),ne=function(e,t){var n="0x"+e.slice(1)-65536;return t||(n<0?String.fromCharCode(n+65536):String.fromCharCode(n>>10|55296,1023&n|56320))},re=/([\0-\x1f\x7f]|^-?\d)|^-$|[^\0-\x1f\x7f-\uFFFF\w-]/g,ie=function(e,t){return t?"\0"===e?"\ufffd":e.slice(0,-1)+"\\"+e.charCodeAt(e.length-1).toString(16)+" ":"\\"+e},oe=function(){T()},ae=be(function(e){return!0===e.disabled&&"fieldset"===e.nodeName.toLowerCase()},{dir:"parentNode",next:"legend"});try{H.apply(t=O.call(p.childNodes),p.childNodes),t[p.childNodes.length].nodeType}catch(e){H={apply:t.length?function(e,t){L.apply(e,O.call(t))}:function(e,t){var n=e.length,r=0;while(e[n++]=t[r++]);e.length=n-1}}}function se(t,e,n,r){var i,o,a,s,u,l,c,f=e&&e.ownerDocument,p=e?e.nodeType:9;if(n=n||[],"string"!=typeof t||!t||1!==p&&9!==p&&11!==p)return n;if(!r&&(T(e),e=e||C,E)){if(11!==p&&(u=Z.exec(t)))if(i=u[1]){if(9===p){if(!(a=e.getElementById(i)))return n;if(a.id===i)return n.push(a),n}else if(f&&(a=f.getElementById(i))&&y(e,a)&&a.id===i)return n.push(a),n}else{if(u[2])return H.apply(n,e.getElementsByTagName(t)),n;if((i=u[3])&&d.getElementsByClassName&&e.getElementsByClassName)return H.apply(n,e.getElementsByClassName(i)),n}if(d.qsa&&!N[t+" "]&&(!v||!v.test(t))&&(1!==p||"object"!==e.nodeName.toLowerCase())){if(c=t,f=e,1===p&&(U.test(t)||z.test(t))){(f=ee.test(t)&&ye(e.parentNode)||e)===e&&d.scope||((s=e.getAttribute("id"))?s=s.replace(re,ie):e.setAttribute("id",s=S)),o=(l=h(t)).length;while(o--)l[o]=(s?"#"+s:":scope")+" "+xe(l[o]);c=l.join(",")}try{return H.apply(n,f.querySelectorAll(c)),n}catch(e){N(t,!0)}finally{s===S&&e.removeAttribute("id")}}}return g(t.replace($,"$1"),e,n,r)}function ue(){var r=[];return function e(t,n){return r.push(t+" ")>b.cacheLength&&delete e[r.shift()],e[t+" "]=n}}function le(e){return e[S]=!0,e}function ce(e){var t=C.createElement("fieldset");try{return!!e(t)}catch(e){return!1}finally{t.parentNode&&t.parentNode.removeChild(t),t=null}}function fe(e,t){var n=e.split("|"),r=n.length;while(r--)b.attrHandle[n[r]]=t}function pe(e,t){var n=t&&e,r=n&&1===e.nodeType&&1===t.nodeType&&e.sourceIndex-t.sourceIndex;if(r)return r;if(n)while(n=n.nextSibling)if(n===t)return-1;return e?1:-1}function de(t){return function(e){return"input"===e.nodeName.toLowerCase()&&e.type===t}}function he(n){return function(e){var t=e.nodeName.toLowerCase();return("input"===t||"button"===t)&&e.type===n}}function ge(t){return function(e){return"form"in e?e.parentNode&&!1===e.disabled?"label"in e?"label"in e.parentNode?e.parentNode.disabled===t:e.disabled===t:e.isDisabled===t||e.isDisabled!==!t&&ae(e)===t:e.disabled===t:"label"in e&&e.disabled===t}}function ve(a){return le(function(o){return o=+o,le(function(e,t){var n,r=a([],e.length,o),i=r.length;while(i--)e[n=r[i]]&&(e[n]=!(t[n]=e[n]))})})}function ye(e){return e&&"undefined"!=typeof e.getElementsByTagName&&e}for(e in d=se.support={},i=se.isXML=function(e){var t=e&&e.namespaceURI,n=e&&(e.ownerDocument||e).documentElement;return!Y.test(t||n&&n.nodeName||"HTML")},T=se.setDocument=function(e){var t,n,r=e?e.ownerDocument||e:p;return r!=C&&9===r.nodeType&&r.documentElement&&(a=(C=r).documentElement,E=!i(C),p!=C&&(n=C.defaultView)&&n.top!==n&&(n.addEventListener?n.addEventListener("unload",oe,!1):n.attachEvent&&n.attachEvent("onunload",oe)),d.scope=ce(function(e){return a.appendChild(e).appendChild(C.createElement("div")),"undefined"!=typeof e.querySelectorAll&&!e.querySelectorAll(":scope fieldset div").length}),d.attributes=ce(function(e){return e.className="i",!e.getAttribute("className")}),d.getElementsByTagName=ce(function(e){return e.appendChild(C.createComment("")),!e.getElementsByTagName("*").length}),d.getElementsByClassName=K.test(C.getElementsByClassName),d.getById=ce(function(e){return a.appendChild(e).id=S,!C.getElementsByName||!C.getElementsByName(S).length}),d.getById?(b.filter.ID=function(e){var t=e.replace(te,ne);return function(e){return e.getAttribute("id")===t}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n=t.getElementById(e);return n?[n]:[]}}):(b.filter.ID=function(e){var n=e.replace(te,ne);return function(e){var t="undefined"!=typeof e.getAttributeNode&&e.getAttributeNode("id");return t&&t.value===n}},b.find.ID=function(e,t){if("undefined"!=typeof t.getElementById&&E){var n,r,i,o=t.getElementById(e);if(o){if((n=o.getAttributeNode("id"))&&n.value===e)return[o];i=t.getElementsByName(e),r=0;while(o=i[r++])if((n=o.getAttributeNode("id"))&&n.value===e)return[o]}return[]}}),b.find.TAG=d.getElementsByTagName?function(e,t){return"undefined"!=typeof t.getElementsByTagName?t.getElementsByTagName(e):d.qsa?t.querySelectorAll(e):void 0}:function(e,t){var n,r=[],i=0,o=t.getElementsByTagName(e);if("*"===e){while(n=o[i++])1===n.nodeType&&r.push(n);return r}return o},b.find.CLASS=d.getElementsByClassName&&function(e,t){if("undefined"!=typeof t.getElementsByClassName&&E)return t.getElementsByClassName(e)},s=[],v=[],(d.qsa=K.test(C.querySelectorAll))&&(ce(function(e){var t;a.appendChild(e).innerHTML=" ",e.querySelectorAll("[msallowcapture^='']").length&&v.push("[*^$]="+M+"*(?:''|\"\")"),e.querySelectorAll("[selected]").length||v.push("\\["+M+"*(?:value|"+R+")"),e.querySelectorAll("[id~="+S+"-]").length||v.push("~="),(t=C.createElement("input")).setAttribute("name",""),e.appendChild(t),e.querySelectorAll("[name='']").length||v.push("\\["+M+"*name"+M+"*="+M+"*(?:''|\"\")"),e.querySelectorAll(":checked").length||v.push(":checked"),e.querySelectorAll("a#"+S+"+*").length||v.push(".#.+[+~]"),e.querySelectorAll("\\\f"),v.push("[\\r\\n\\f]")}),ce(function(e){e.innerHTML=" ";var t=C.createElement("input");t.setAttribute("type","hidden"),e.appendChild(t).setAttribute("name","D"),e.querySelectorAll("[name=d]").length&&v.push("name"+M+"*[*^$|!~]?="),2!==e.querySelectorAll(":enabled").length&&v.push(":enabled",":disabled"),a.appendChild(e).disabled=!0,2!==e.querySelectorAll(":disabled").length&&v.push(":enabled",":disabled"),e.querySelectorAll("*,:x"),v.push(",.*:")})),(d.matchesSelector=K.test(c=a.matches||a.webkitMatchesSelector||a.mozMatchesSelector||a.oMatchesSelector||a.msMatchesSelector))&&ce(function(e){d.disconnectedMatch=c.call(e,"*"),c.call(e,"[s!='']:x"),s.push("!=",F)}),v=v.length&&new RegExp(v.join("|")),s=s.length&&new RegExp(s.join("|")),t=K.test(a.compareDocumentPosition),y=t||K.test(a.contains)?function(e,t){var n=9===e.nodeType?e.documentElement:e,r=t&&t.parentNode;return e===r||!(!r||1!==r.nodeType||!(n.contains?n.contains(r):e.compareDocumentPosition&&16&e.compareDocumentPosition(r)))}:function(e,t){if(t)while(t=t.parentNode)if(t===e)return!0;return!1},j=t?function(e,t){if(e===t)return l=!0,0;var n=!e.compareDocumentPosition-!t.compareDocumentPosition;return n||(1&(n=(e.ownerDocument||e)==(t.ownerDocument||t)?e.compareDocumentPosition(t):1)||!d.sortDetached&&t.compareDocumentPosition(e)===n?e==C||e.ownerDocument==p&&y(p,e)?-1:t==C||t.ownerDocument==p&&y(p,t)?1:u?P(u,e)-P(u,t):0:4&n?-1:1)}:function(e,t){if(e===t)return l=!0,0;var n,r=0,i=e.parentNode,o=t.parentNode,a=[e],s=[t];if(!i||!o)return e==C?-1:t==C?1:i?-1:o?1:u?P(u,e)-P(u,t):0;if(i===o)return pe(e,t);n=e;while(n=n.parentNode)a.unshift(n);n=t;while(n=n.parentNode)s.unshift(n);while(a[r]===s[r])r++;return r?pe(a[r],s[r]):a[r]==p?-1:s[r]==p?1:0}),C},se.matches=function(e,t){return se(e,null,null,t)},se.matchesSelector=function(e,t){if(T(e),d.matchesSelector&&E&&!N[t+" "]&&(!s||!s.test(t))&&(!v||!v.test(t)))try{var n=c.call(e,t);if(n||d.disconnectedMatch||e.document&&11!==e.document.nodeType)return n}catch(e){N(t,!0)}return 0":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(e){return e[1]=e[1].replace(te,ne),e[3]=(e[3]||e[4]||e[5]||"").replace(te,ne),"~="===e[2]&&(e[3]=" "+e[3]+" "),e.slice(0,4)},CHILD:function(e){return e[1]=e[1].toLowerCase(),"nth"===e[1].slice(0,3)?(e[3]||se.error(e[0]),e[4]=+(e[4]?e[5]+(e[6]||1):2*("even"===e[3]||"odd"===e[3])),e[5]=+(e[7]+e[8]||"odd"===e[3])):e[3]&&se.error(e[0]),e},PSEUDO:function(e){var t,n=!e[6]&&e[2];return G.CHILD.test(e[0])?null:(e[3]?e[2]=e[4]||e[5]||"":n&&X.test(n)&&(t=h(n,!0))&&(t=n.indexOf(")",n.length-t)-n.length)&&(e[0]=e[0].slice(0,t),e[2]=n.slice(0,t)),e.slice(0,3))}},filter:{TAG:function(e){var t=e.replace(te,ne).toLowerCase();return"*"===e?function(){return!0}:function(e){return e.nodeName&&e.nodeName.toLowerCase()===t}},CLASS:function(e){var t=m[e+" "];return t||(t=new RegExp("(^|"+M+")"+e+"("+M+"|$)"))&&m(e,function(e){return t.test("string"==typeof e.className&&e.className||"undefined"!=typeof e.getAttribute&&e.getAttribute("class")||"")})},ATTR:function(n,r,i){return function(e){var t=se.attr(e,n);return null==t?"!="===r:!r||(t+="","="===r?t===i:"!="===r?t!==i:"^="===r?i&&0===t.indexOf(i):"*="===r?i&&-1:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i;function j(e,n,r){return m(n)?S.grep(e,function(e,t){return!!n.call(e,t,e)!==r}):n.nodeType?S.grep(e,function(e){return e===n!==r}):"string"!=typeof n?S.grep(e,function(e){return-1)[^>]*|#([\w-]+))$/;(S.fn.init=function(e,t,n){var r,i;if(!e)return this;if(n=n||D,"string"==typeof e){if(!(r="<"===e[0]&&">"===e[e.length-1]&&3<=e.length?[null,e,null]:q.exec(e))||!r[1]&&t)return!t||t.jquery?(t||n).find(e):this.constructor(t).find(e);if(r[1]){if(t=t instanceof S?t[0]:t,S.merge(this,S.parseHTML(r[1],t&&t.nodeType?t.ownerDocument||t:E,!0)),N.test(r[1])&&S.isPlainObject(t))for(r in t)m(this[r])?this[r](t[r]):this.attr(r,t[r]);return this}return(i=E.getElementById(r[2]))&&(this[0]=i,this.length=1),this}return e.nodeType?(this[0]=e,this.length=1,this):m(e)?void 0!==n.ready?n.ready(e):e(S):S.makeArray(e,this)}).prototype=S.fn,D=S(E);var L=/^(?:parents|prev(?:Until|All))/,H={children:!0,contents:!0,next:!0,prev:!0};function O(e,t){while((e=e[t])&&1!==e.nodeType);return e}S.fn.extend({has:function(e){var t=S(e,this),n=t.length;return this.filter(function(){for(var e=0;e\x20\t\r\n\f]*)/i,he=/^$|^module$|\/(?:java|ecma)script/i;ce=E.createDocumentFragment().appendChild(E.createElement("div")),(fe=E.createElement("input")).setAttribute("type","radio"),fe.setAttribute("checked","checked"),fe.setAttribute("name","t"),ce.appendChild(fe),y.checkClone=ce.cloneNode(!0).cloneNode(!0).lastChild.checked,ce.innerHTML="",y.noCloneChecked=!!ce.cloneNode(!0).lastChild.defaultValue,ce.innerHTML=" ",y.option=!!ce.lastChild;var ge={thead:[1,""],col:[2,""],tr:[2,""],td:[3,""],_default:[0,"",""]};function ve(e,t){var n;return n="undefined"!=typeof e.getElementsByTagName?e.getElementsByTagName(t||"*"):"undefined"!=typeof e.querySelectorAll?e.querySelectorAll(t||"*"):[],void 0===t||t&&A(e,t)?S.merge([e],n):n}function ye(e,t){for(var n=0,r=e.length;n",""]);var me=/<|?\w+;/;function xe(e,t,n,r,i){for(var o,a,s,u,l,c,f=t.createDocumentFragment(),p=[],d=0,h=e.length;d\s*$/g;function je(e,t){return A(e,"table")&&A(11!==t.nodeType?t:t.firstChild,"tr")&&S(e).children("tbody")[0]||e}function De(e){return e.type=(null!==e.getAttribute("type"))+"/"+e.type,e}function qe(e){return"true/"===(e.type||"").slice(0,5)?e.type=e.type.slice(5):e.removeAttribute("type"),e}function Le(e,t){var n,r,i,o,a,s;if(1===t.nodeType){if(Y.hasData(e)&&(s=Y.get(e).events))for(i in Y.remove(t,"handle events"),s)for(n=0,r=s[i].length;n").attr(n.scriptAttrs||{}).prop({charset:n.scriptCharset,src:n.url}).on("load error",i=function(e){r.remove(),i=null,e&&t("error"===e.type?404:200,e.type)}),E.head.appendChild(r[0])},abort:function(){i&&i()}}});var _t,zt=[],Ut=/(=)\?(?=&|$)|\?\?/;S.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var e=zt.pop()||S.expando+"_"+wt.guid++;return this[e]=!0,e}}),S.ajaxPrefilter("json jsonp",function(e,t,n){var r,i,o,a=!1!==e.jsonp&&(Ut.test(e.url)?"url":"string"==typeof e.data&&0===(e.contentType||"").indexOf("application/x-www-form-urlencoded")&&Ut.test(e.data)&&"data");if(a||"jsonp"===e.dataTypes[0])return r=e.jsonpCallback=m(e.jsonpCallback)?e.jsonpCallback():e.jsonpCallback,a?e[a]=e[a].replace(Ut,"$1"+r):!1!==e.jsonp&&(e.url+=(Tt.test(e.url)?"&":"?")+e.jsonp+"="+r),e.converters["script json"]=function(){return o||S.error(r+" was not called"),o[0]},e.dataTypes[0]="json",i=C[r],C[r]=function(){o=arguments},n.always(function(){void 0===i?S(C).removeProp(r):C[r]=i,e[r]&&(e.jsonpCallback=t.jsonpCallback,zt.push(r)),o&&m(i)&&i(o[0]),o=i=void 0}),"script"}),y.createHTMLDocument=((_t=E.implementation.createHTMLDocument("").body).innerHTML="",2===_t.childNodes.length),S.parseHTML=function(e,t,n){return"string"!=typeof e?[]:("boolean"==typeof t&&(n=t,t=!1),t||(y.createHTMLDocument?((r=(t=E.implementation.createHTMLDocument("")).createElement("base")).href=E.location.href,t.head.appendChild(r)):t=E),o=!n&&[],(i=N.exec(e))?[t.createElement(i[1])]:(i=xe([e],t,o),o&&o.length&&S(o).remove(),S.merge([],i.childNodes)));var r,i,o},S.fn.load=function(e,t,n){var r,i,o,a=this,s=e.indexOf(" ");return-1").append(S.parseHTML(e)).find(r):e)}).always(n&&function(e,t){a.each(function(){n.apply(this,o||[e.responseText,t,e])})}),this},S.expr.pseudos.animated=function(t){return S.grep(S.timers,function(e){return t===e.elem}).length},S.offset={setOffset:function(e,t,n){var r,i,o,a,s,u,l=S.css(e,"position"),c=S(e),f={};"static"===l&&(e.style.position="relative"),s=c.offset(),o=S.css(e,"top"),u=S.css(e,"left"),("absolute"===l||"fixed"===l)&&-1<(o+u).indexOf("auto")?(a=(r=c.position()).top,i=r.left):(a=parseFloat(o)||0,i=parseFloat(u)||0),m(t)&&(t=t.call(e,n,S.extend({},s))),null!=t.top&&(f.top=t.top-s.top+a),null!=t.left&&(f.left=t.left-s.left+i),"using"in t?t.using.call(e,f):c.css(f)}},S.fn.extend({offset:function(t){if(arguments.length)return void 0===t?this:this.each(function(e){S.offset.setOffset(this,t,e)});var e,n,r=this[0];return r?r.getClientRects().length?(e=r.getBoundingClientRect(),n=r.ownerDocument.defaultView,{top:e.top+n.pageYOffset,left:e.left+n.pageXOffset}):{top:0,left:0}:void 0},position:function(){if(this[0]){var e,t,n,r=this[0],i={top:0,left:0};if("fixed"===S.css(r,"position"))t=r.getBoundingClientRect();else{t=this.offset(),n=r.ownerDocument,e=r.offsetParent||n.documentElement;while(e&&(e===n.body||e===n.documentElement)&&"static"===S.css(e,"position"))e=e.parentNode;e&&e!==r&&1===e.nodeType&&((i=S(e).offset()).top+=S.css(e,"borderTopWidth",!0),i.left+=S.css(e,"borderLeftWidth",!0))}return{top:t.top-i.top-S.css(r,"marginTop",!0),left:t.left-i.left-S.css(r,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var e=this.offsetParent;while(e&&"static"===S.css(e,"position"))e=e.offsetParent;return e||re})}}),S.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(t,i){var o="pageYOffset"===i;S.fn[t]=function(e){return $(this,function(e,t,n){var r;if(x(e)?r=e:9===e.nodeType&&(r=e.defaultView),void 0===n)return r?r[i]:e[t];r?r.scrollTo(o?r.pageXOffset:n,o?n:r.pageYOffset):e[t]=n},t,e,arguments.length)}}),S.each(["top","left"],function(e,n){S.cssHooks[n]=Fe(y.pixelPosition,function(e,t){if(t)return t=We(e,n),Pe.test(t)?S(e).position()[n]+"px":t})}),S.each({Height:"height",Width:"width"},function(a,s){S.each({padding:"inner"+a,content:s,"":"outer"+a},function(r,o){S.fn[o]=function(e,t){var n=arguments.length&&(r||"boolean"!=typeof e),i=r||(!0===e||!0===t?"margin":"border");return $(this,function(e,t,n){var r;return x(e)?0===o.indexOf("outer")?e["inner"+a]:e.document.documentElement["client"+a]:9===e.nodeType?(r=e.documentElement,Math.max(e.body["scroll"+a],r["scroll"+a],e.body["offset"+a],r["offset"+a],r["client"+a])):void 0===n?S.css(e,t,i):S.style(e,t,n,i)},s,n?e:void 0,n)}})}),S.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(e,t){S.fn[t]=function(e){return this.on(t,e)}}),S.fn.extend({bind:function(e,t,n){return this.on(e,null,t,n)},unbind:function(e,t){return this.off(e,null,t)},delegate:function(e,t,n,r){return this.on(t,e,n,r)},undelegate:function(e,t,n){return 1===arguments.length?this.off(e,"**"):this.off(t,e||"**",n)},hover:function(e,t){return this.mouseenter(e).mouseleave(t||e)}}),S.each("blur focus focusin focusout resize scroll click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup contextmenu".split(" "),function(e,n){S.fn[n]=function(e,t){return 0",options:{classes:{},disabled:!1,create:null},_createWidget:function(t,e){e=y(e||this.defaultElement||this)[0],this.element=y(e),this.uuid=i++,this.eventNamespace="."+this.widgetName+this.uuid,this.bindings=y(),this.hoverable=y(),this.focusable=y(),this.classesElementLookup={},e!==this&&(y.data(e,this.widgetFullName,this),this._on(!0,this.element,{remove:function(t){t.target===e&&this.destroy()}}),this.document=y(e.style?e.ownerDocument:e.document||e),this.window=y(this.document[0].defaultView||this.document[0].parentWindow)),this.options=y.widget.extend({},this.options,this._getCreateOptions(),t),this._create(),this.options.disabled&&this._setOptionDisabled(this.options.disabled),this._trigger("create",null,this._getCreateEventData()),this._init()},_getCreateOptions:function(){return{}},_getCreateEventData:y.noop,_create:y.noop,_init:y.noop,destroy:function(){var i=this;this._destroy(),y.each(this.classesElementLookup,function(t,e){i._removeClass(e,t)}),this.element.off(this.eventNamespace).removeData(this.widgetFullName),this.widget().off(this.eventNamespace).removeAttr("aria-disabled"),this.bindings.off(this.eventNamespace)},_destroy:y.noop,widget:function(){return this.element},option:function(t,e){var i,s,n,o=t;if(0===arguments.length)return y.widget.extend({},this.options);if("string"==typeof t)if(o={},t=(i=t.split(".")).shift(),i.length){for(s=o[t]=y.widget.extend({},this.options[t]),n=0;n
"),i=e.children()[0];return y("body").append(e),t=i.offsetWidth,e.css("overflow","scroll"),t===(i=i.offsetWidth)&&(i=e[0].clientWidth),e.remove(),s=t-i},getScrollInfo:function(t){var e=t.isWindow||t.isDocument?"":t.element.css("overflow-x"),i=t.isWindow||t.isDocument?"":t.element.css("overflow-y"),e="scroll"===e||"auto"===e&&t.widthx(D(s),D(n))?o.important="horizontal":o.important="vertical",p.using.call(this,t,o)}),h.offset(y.extend(l,{using:t}))})},y.ui.position={fit:{left:function(t,e){var i=e.within,s=i.isWindow?i.scrollLeft:i.offset.left,n=i.width,o=t.left-e.collisionPosition.marginLeft,h=s-o,a=o+e.collisionWidth-n-s;e.collisionWidth>n?0n?0=this.options.distance},_mouseDelayMet:function(){return this.mouseDelayMet},_mouseStart:function(){},_mouseDrag:function(){},_mouseStop:function(){},_mouseCapture:function(){return!0}}),y.ui.plugin={add:function(t,e,i){var s,n=y.ui[t].prototype;for(s in i)n.plugins[s]=n.plugins[s]||[],n.plugins[s].push([e,i[s]])},call:function(t,e,i,s){var n,o=t.plugins[e];if(o&&(s||t.element[0].parentNode&&11!==t.element[0].parentNode.nodeType))for(n=0;n").css({overflow:"hidden",position:this.element.css("position"),width:this.element.outerWidth(),height:this.element.outerHeight(),top:this.element.css("top"),left:this.element.css("left")})),this.element=this.element.parent().data("ui-resizable",this.element.resizable("instance")),this.elementIsWrapper=!0,t={marginTop:this.originalElement.css("marginTop"),marginRight:this.originalElement.css("marginRight"),marginBottom:this.originalElement.css("marginBottom"),marginLeft:this.originalElement.css("marginLeft")},this.element.css(t),this.originalElement.css("margin",0),this.originalResizeStyle=this.originalElement.css("resize"),this.originalElement.css("resize","none"),this._proportionallyResizeElements.push(this.originalElement.css({position:"static",zoom:1,display:"block"})),this.originalElement.css(t),this._proportionallyResize()),this._setupHandles(),e.autoHide&&y(this.element).on("mouseenter",function(){e.disabled||(i._removeClass("ui-resizable-autohide"),i._handles.show())}).on("mouseleave",function(){e.disabled||i.resizing||(i._addClass("ui-resizable-autohide"),i._handles.hide())}),this._mouseInit()},_destroy:function(){this._mouseDestroy(),this._addedHandles.remove();function t(t){y(t).removeData("resizable").removeData("ui-resizable").off(".resizable")}var e;return this.elementIsWrapper&&(t(this.element),e=this.element,this.originalElement.css({position:e.css("position"),width:e.outerWidth(),height:e.outerHeight(),top:e.css("top"),left:e.css("left")}).insertAfter(e),e.remove()),this.originalElement.css("resize",this.originalResizeStyle),t(this.originalElement),this},_setOption:function(t,e){switch(this._super(t,e),t){case"handles":this._removeHandles(),this._setupHandles();break;case"aspectRatio":this._aspectRatio=!!e}},_setupHandles:function(){var t,e,i,s,n,o=this.options,h=this;if(this.handles=o.handles||(y(".ui-resizable-handle",this.element).length?{n:".ui-resizable-n",e:".ui-resizable-e",s:".ui-resizable-s",w:".ui-resizable-w",se:".ui-resizable-se",sw:".ui-resizable-sw",ne:".ui-resizable-ne",nw:".ui-resizable-nw"}:"e,s,se"),this._handles=y(),this._addedHandles=y(),this.handles.constructor===String)for("all"===this.handles&&(this.handles="n,e,s,w,se,sw,ne,nw"),i=this.handles.split(","),this.handles={},e=0;e"),this._addClass(n,"ui-resizable-handle "+s),n.css({zIndex:o.zIndex}),this.handles[t]=".ui-resizable-"+t,this.element.children(this.handles[t]).length||(this.element.append(n),this._addedHandles=this._addedHandles.add(n));this._renderAxis=function(t){var e,i,s;for(e in t=t||this.element,this.handles)this.handles[e].constructor===String?this.handles[e]=this.element.children(this.handles[e]).first().show():(this.handles[e].jquery||this.handles[e].nodeType)&&(this.handles[e]=y(this.handles[e]),this._on(this.handles[e],{mousedown:h._mouseDown})),this.elementIsWrapper&&this.originalElement[0].nodeName.match(/^(textarea|input|select|button)$/i)&&(i=y(this.handles[e],this.element),s=/sw|ne|nw|se|n|s/.test(e)?i.outerHeight():i.outerWidth(),i=["padding",/ne|nw|n/.test(e)?"Top":/se|sw|s/.test(e)?"Bottom":/^e$/.test(e)?"Right":"Left"].join(""),t.css(i,s),this._proportionallyResize()),this._handles=this._handles.add(this.handles[e])},this._renderAxis(this.element),this._handles=this._handles.add(this.element.find(".ui-resizable-handle")),this._handles.disableSelection(),this._handles.on("mouseover",function(){h.resizing||(this.className&&(n=this.className.match(/ui-resizable-(se|sw|ne|nw|n|e|s|w)/i)),h.axis=n&&n[1]?n[1]:"se")}),o.autoHide&&(this._handles.hide(),this._addClass("ui-resizable-autohide"))},_removeHandles:function(){this._addedHandles.remove()},_mouseCapture:function(t){var e,i,s=!1;for(e in this.handles)(i=y(this.handles[e])[0])!==t.target&&!y.contains(i,t.target)||(s=!0);return!this.options.disabled&&s},_mouseStart:function(t){var e,i,s=this.options,n=this.element;return this.resizing=!0,this._renderProxy(),e=this._num(this.helper.css("left")),i=this._num(this.helper.css("top")),s.containment&&(e+=y(s.containment).scrollLeft()||0,i+=y(s.containment).scrollTop()||0),this.offset=this.helper.offset(),this.position={left:e,top:i},this.size=this._helper?{width:this.helper.width(),height:this.helper.height()}:{width:n.width(),height:n.height()},this.originalSize=this._helper?{width:n.outerWidth(),height:n.outerHeight()}:{width:n.width(),height:n.height()},this.sizeDiff={width:n.outerWidth()-n.width(),height:n.outerHeight()-n.height()},this.originalPosition={left:e,top:i},this.originalMousePosition={left:t.pageX,top:t.pageY},this.aspectRatio="number"==typeof s.aspectRatio?s.aspectRatio:this.originalSize.width/this.originalSize.height||1,s=y(".ui-resizable-"+this.axis).css("cursor"),y("body").css("cursor","auto"===s?this.axis+"-resize":s),this._addClass("ui-resizable-resizing"),this._propagate("start",t),!0},_mouseDrag:function(t){var e=this.originalMousePosition,i=this.axis,s=t.pageX-e.left||0,e=t.pageY-e.top||0,i=this._change[i];return this._updatePrevProperties(),i&&(e=i.apply(this,[t,s,e]),this._updateVirtualBoundaries(t.shiftKey),(this._aspectRatio||t.shiftKey)&&(e=this._updateRatio(e,t)),e=this._respectSize(e,t),this._updateCache(e),this._propagate("resize",t),e=this._applyChanges(),!this._helper&&this._proportionallyResizeElements.length&&this._proportionallyResize(),y.isEmptyObject(e)||(this._updatePrevProperties(),this._trigger("resize",t,this.ui()),this._applyChanges())),!1},_mouseStop:function(t){this.resizing=!1;var e,i,s,n=this.options,o=this;return this._helper&&(s=(e=(i=this._proportionallyResizeElements).length&&/textarea/i.test(i[0].nodeName))&&this._hasScroll(i[0],"left")?0:o.sizeDiff.height,i=e?0:o.sizeDiff.width,e={width:o.helper.width()-i,height:o.helper.height()-s},i=parseFloat(o.element.css("left"))+(o.position.left-o.originalPosition.left)||null,s=parseFloat(o.element.css("top"))+(o.position.top-o.originalPosition.top)||null,n.animate||this.element.css(y.extend(e,{top:s,left:i})),o.helper.height(o.size.height),o.helper.width(o.size.width),this._helper&&!n.animate&&this._proportionallyResize()),y("body").css("cursor","auto"),this._removeClass("ui-resizable-resizing"),this._propagate("stop",t),this._helper&&this.helper.remove(),!1},_updatePrevProperties:function(){this.prevPosition={top:this.position.top,left:this.position.left},this.prevSize={width:this.size.width,height:this.size.height}},_applyChanges:function(){var t={};return this.position.top!==this.prevPosition.top&&(t.top=this.position.top+"px"),this.position.left!==this.prevPosition.left&&(t.left=this.position.left+"px"),this.size.width!==this.prevSize.width&&(t.width=this.size.width+"px"),this.size.height!==this.prevSize.height&&(t.height=this.size.height+"px"),this.helper.css(t),t},_updateVirtualBoundaries:function(t){var e,i,s=this.options,n={minWidth:this._isNumber(s.minWidth)?s.minWidth:0,maxWidth:this._isNumber(s.maxWidth)?s.maxWidth:1/0,minHeight:this._isNumber(s.minHeight)?s.minHeight:0,maxHeight:this._isNumber(s.maxHeight)?s.maxHeight:1/0};(this._aspectRatio||t)&&(e=n.minHeight*this.aspectRatio,i=n.minWidth/this.aspectRatio,s=n.maxHeight*this.aspectRatio,t=n.maxWidth/this.aspectRatio,e>n.minWidth&&(n.minWidth=e),i>n.minHeight&&(n.minHeight=i),st.width,h=this._isNumber(t.height)&&e.minHeight&&e.minHeight>t.height,a=this.originalPosition.left+this.originalSize.width,r=this.originalPosition.top+this.originalSize.height,l=/sw|nw|w/.test(i),i=/nw|ne|n/.test(i);return o&&(t.width=e.minWidth),h&&(t.height=e.minHeight),s&&(t.width=e.maxWidth),n&&(t.height=e.maxHeight),o&&l&&(t.left=a-e.minWidth),s&&l&&(t.left=a-e.maxWidth),h&&i&&(t.top=r-e.minHeight),n&&i&&(t.top=r-e.maxHeight),t.width||t.height||t.left||!t.top?t.width||t.height||t.top||!t.left||(t.left=null):t.top=null,t},_getPaddingPlusBorderDimensions:function(t){for(var e=0,i=[],s=[t.css("borderTopWidth"),t.css("borderRightWidth"),t.css("borderBottomWidth"),t.css("borderLeftWidth")],n=[t.css("paddingTop"),t.css("paddingRight"),t.css("paddingBottom"),t.css("paddingLeft")];e<4;e++)i[e]=parseFloat(s[e])||0,i[e]+=parseFloat(n[e])||0;return{height:i[0]+i[2],width:i[1]+i[3]}},_proportionallyResize:function(){if(this._proportionallyResizeElements.length)for(var t,e=0,i=this.helper||this.element;e").css({overflow:"hidden"}),this._addClass(this.helper,this._helper),this.helper.css({width:this.element.outerWidth(),height:this.element.outerHeight(),position:"absolute",left:this.elementOffset.left+"px",top:this.elementOffset.top+"px",zIndex:++e.zIndex}),this.helper.appendTo("body").disableSelection()):this.helper=this.element},_change:{e:function(t,e){return{width:this.originalSize.width+e}},w:function(t,e){var i=this.originalSize;return{left:this.originalPosition.left+e,width:i.width-e}},n:function(t,e,i){var s=this.originalSize;return{top:this.originalPosition.top+i,height:s.height-i}},s:function(t,e,i){return{height:this.originalSize.height+i}},se:function(t,e,i){return y.extend(this._change.s.apply(this,arguments),this._change.e.apply(this,[t,e,i]))},sw:function(t,e,i){return y.extend(this._change.s.apply(this,arguments),this._change.w.apply(this,[t,e,i]))},ne:function(t,e,i){return y.extend(this._change.n.apply(this,arguments),this._change.e.apply(this,[t,e,i]))},nw:function(t,e,i){return y.extend(this._change.n.apply(this,arguments),this._change.w.apply(this,[t,e,i]))}},_propagate:function(t,e){y.ui.plugin.call(this,t,[e,this.ui()]),"resize"!==t&&this._trigger(t,e,this.ui())},plugins:{},ui:function(){return{originalElement:this.originalElement,element:this.element,helper:this.helper,position:this.position,size:this.size,originalSize:this.originalSize,originalPosition:this.originalPosition}}}),y.ui.plugin.add("resizable","animate",{stop:function(e){var i=y(this).resizable("instance"),t=i.options,s=i._proportionallyResizeElements,n=s.length&&/textarea/i.test(s[0].nodeName),o=n&&i._hasScroll(s[0],"left")?0:i.sizeDiff.height,h=n?0:i.sizeDiff.width,n={width:i.size.width-h,height:i.size.height-o},h=parseFloat(i.element.css("left"))+(i.position.left-i.originalPosition.left)||null,o=parseFloat(i.element.css("top"))+(i.position.top-i.originalPosition.top)||null;i.element.animate(y.extend(n,o&&h?{top:o,left:h}:{}),{duration:t.animateDuration,easing:t.animateEasing,step:function(){var t={width:parseFloat(i.element.css("width")),height:parseFloat(i.element.css("height")),top:parseFloat(i.element.css("top")),left:parseFloat(i.element.css("left"))};s&&s.length&&y(s[0]).css({width:t.width,height:t.height}),i._updateCache(t),i._propagate("resize",e)}})}}),y.ui.plugin.add("resizable","containment",{start:function(){var i,s,n=y(this).resizable("instance"),t=n.options,e=n.element,o=t.containment,h=o instanceof y?o.get(0):/parent/.test(o)?e.parent().get(0):o;h&&(n.containerElement=y(h),/document/.test(o)||o===document?(n.containerOffset={left:0,top:0},n.containerPosition={left:0,top:0},n.parentData={element:y(document),left:0,top:0,width:y(document).width(),height:y(document).height()||document.body.parentNode.scrollHeight}):(i=y(h),s=[],y(["Top","Right","Left","Bottom"]).each(function(t,e){s[t]=n._num(i.css("padding"+e))}),n.containerOffset=i.offset(),n.containerPosition=i.position(),n.containerSize={height:i.innerHeight()-s[3],width:i.innerWidth()-s[1]},t=n.containerOffset,e=n.containerSize.height,o=n.containerSize.width,o=n._hasScroll(h,"left")?h.scrollWidth:o,e=n._hasScroll(h)?h.scrollHeight:e,n.parentData={element:h,left:t.left,top:t.top,width:o,height:e}))},resize:function(t){var e=y(this).resizable("instance"),i=e.options,s=e.containerOffset,n=e.position,o=e._aspectRatio||t.shiftKey,h={top:0,left:0},a=e.containerElement,t=!0;a[0]!==document&&/static/.test(a.css("position"))&&(h=s),n.left<(e._helper?s.left:0)&&(e.size.width=e.size.width+(e._helper?e.position.left-s.left:e.position.left-h.left),o&&(e.size.height=e.size.width/e.aspectRatio,t=!1),e.position.left=i.helper?s.left:0),n.top<(e._helper?s.top:0)&&(e.size.height=e.size.height+(e._helper?e.position.top-s.top:e.position.top),o&&(e.size.width=e.size.height*e.aspectRatio,t=!1),e.position.top=e._helper?s.top:0),i=e.containerElement.get(0)===e.element.parent().get(0),n=/relative|absolute/.test(e.containerElement.css("position")),i&&n?(e.offset.left=e.parentData.left+e.position.left,e.offset.top=e.parentData.top+e.position.top):(e.offset.left=e.element.offset().left,e.offset.top=e.element.offset().top),n=Math.abs(e.sizeDiff.width+(e._helper?e.offset.left-h.left:e.offset.left-s.left)),s=Math.abs(e.sizeDiff.height+(e._helper?e.offset.top-h.top:e.offset.top-s.top)),n+e.size.width>=e.parentData.width&&(e.size.width=e.parentData.width-n,o&&(e.size.height=e.size.width/e.aspectRatio,t=!1)),s+e.size.height>=e.parentData.height&&(e.size.height=e.parentData.height-s,o&&(e.size.width=e.size.height*e.aspectRatio,t=!1)),t||(e.position.left=e.prevPosition.left,e.position.top=e.prevPosition.top,e.size.width=e.prevSize.width,e.size.height=e.prevSize.height)},stop:function(){var t=y(this).resizable("instance"),e=t.options,i=t.containerOffset,s=t.containerPosition,n=t.containerElement,o=y(t.helper),h=o.offset(),a=o.outerWidth()-t.sizeDiff.width,o=o.outerHeight()-t.sizeDiff.height;t._helper&&!e.animate&&/relative/.test(n.css("position"))&&y(this).css({left:h.left-s.left-i.left,width:a,height:o}),t._helper&&!e.animate&&/static/.test(n.css("position"))&&y(this).css({left:h.left-s.left-i.left,width:a,height:o})}}),y.ui.plugin.add("resizable","alsoResize",{start:function(){var t=y(this).resizable("instance").options;y(t.alsoResize).each(function(){var t=y(this);t.data("ui-resizable-alsoresize",{width:parseFloat(t.width()),height:parseFloat(t.height()),left:parseFloat(t.css("left")),top:parseFloat(t.css("top"))})})},resize:function(t,i){var e=y(this).resizable("instance"),s=e.options,n=e.originalSize,o=e.originalPosition,h={height:e.size.height-n.height||0,width:e.size.width-n.width||0,top:e.position.top-o.top||0,left:e.position.left-o.left||0};y(s.alsoResize).each(function(){var t=y(this),s=y(this).data("ui-resizable-alsoresize"),n={},e=t.parents(i.originalElement[0]).length?["width","height"]:["width","height","top","left"];y.each(e,function(t,e){var i=(s[e]||0)+(h[e]||0);i&&0<=i&&(n[e]=i||null)}),t.css(n)})},stop:function(){y(this).removeData("ui-resizable-alsoresize")}}),y.ui.plugin.add("resizable","ghost",{start:function(){var t=y(this).resizable("instance"),e=t.size;t.ghost=t.originalElement.clone(),t.ghost.css({opacity:.25,display:"block",position:"relative",height:e.height,width:e.width,margin:0,left:0,top:0}),t._addClass(t.ghost,"ui-resizable-ghost"),!1!==y.uiBackCompat&&"string"==typeof t.options.ghost&&t.ghost.addClass(this.options.ghost),t.ghost.appendTo(t.helper)},resize:function(){var t=y(this).resizable("instance");t.ghost&&t.ghost.css({position:"relative",height:t.size.height,width:t.size.width})},stop:function(){var t=y(this).resizable("instance");t.ghost&&t.helper&&t.helper.get(0).removeChild(t.ghost.get(0))}}),y.ui.plugin.add("resizable","grid",{resize:function(){var t,e=y(this).resizable("instance"),i=e.options,s=e.size,n=e.originalSize,o=e.originalPosition,h=e.axis,a="number"==typeof i.grid?[i.grid,i.grid]:i.grid,r=a[0]||1,l=a[1]||1,u=Math.round((s.width-n.width)/r)*r,p=Math.round((s.height-n.height)/l)*l,d=n.width+u,c=n.height+p,f=i.maxWidth&&i.maxWidthd,s=i.minHeight&&i.minHeight>c;i.grid=a,m&&(d+=r),s&&(c+=l),f&&(d-=r),g&&(c-=l),/^(se|s|e)$/.test(h)?(e.size.width=d,e.size.height=c):/^(ne)$/.test(h)?(e.size.width=d,e.size.height=c,e.position.top=o.top-p):/^(sw)$/.test(h)?(e.size.width=d,e.size.height=c,e.position.left=o.left-u):((c-l<=0||d-r<=0)&&(t=e._getPaddingPlusBorderDimensions(this)),0=f[g]?0:Math.min(f[g],n));!a&&1-1){targetElements.on(evt+EVENT_NAMESPACE,function elementToggle(event){$.powerTip.toggle(this,event)})}else{targetElements.on(evt+EVENT_NAMESPACE,function elementOpen(event){$.powerTip.show(this,event)})}});$.each(options.closeEvents,function(idx,evt){if($.inArray(evt,options.openEvents)<0){targetElements.on(evt+EVENT_NAMESPACE,function elementClose(event){$.powerTip.hide(this,!isMouseEvent(event))})}});targetElements.on("keydown"+EVENT_NAMESPACE,function elementKeyDown(event){if(event.keyCode===27){$.powerTip.hide(this,true)}})}return targetElements};$.fn.powerTip.defaults={fadeInTime:200,fadeOutTime:100,followMouse:false,popupId:"powerTip",popupClass:null,intentSensitivity:7,intentPollInterval:100,closeDelay:100,placement:"n",smartPlacement:false,offset:10,mouseOnToPopup:false,manual:false,openEvents:["mouseenter","focus"],closeEvents:["mouseleave","blur"]};$.fn.powerTip.smartPlacementLists={n:["n","ne","nw","s"],e:["e","ne","se","w","nw","sw","n","s","e"],s:["s","se","sw","n"],w:["w","nw","sw","e","ne","se","n","s","w"],nw:["nw","w","sw","n","s","se","nw"],ne:["ne","e","se","n","s","sw","ne"],sw:["sw","w","nw","s","n","ne","sw"],se:["se","e","ne","s","n","nw","se"],"nw-alt":["nw-alt","n","ne-alt","sw-alt","s","se-alt","w","e"],"ne-alt":["ne-alt","n","nw-alt","se-alt","s","sw-alt","e","w"],"sw-alt":["sw-alt","s","se-alt","nw-alt","n","ne-alt","w","e"],"se-alt":["se-alt","s","sw-alt","ne-alt","n","nw-alt","e","w"]};$.powerTip={show:function apiShowTip(element,event){if(isMouseEvent(event)){trackMouse(event);session.previousX=event.pageX;session.previousY=event.pageY;$(element).data(DATA_DISPLAYCONTROLLER).show()}else{$(element).first().data(DATA_DISPLAYCONTROLLER).show(true,true)}return element},reposition:function apiResetPosition(element){$(element).first().data(DATA_DISPLAYCONTROLLER).resetPosition();return element},hide:function apiCloseTip(element,immediate){var displayController;immediate=element?immediate:true;if(element){displayController=$(element).first().data(DATA_DISPLAYCONTROLLER)}else if(session.activeHover){displayController=session.activeHover.data(DATA_DISPLAYCONTROLLER)}if(displayController){displayController.hide(immediate)}return element},toggle:function apiToggle(element,event){if(session.activeHover&&session.activeHover.is(element)){$.powerTip.hide(element,!isMouseEvent(event))}else{$.powerTip.show(element,event)}return element}};$.powerTip.showTip=$.powerTip.show;$.powerTip.closeTip=$.powerTip.hide;function CSSCoordinates(){var me=this;me.top="auto";me.left="auto";me.right="auto";me.bottom="auto";me.set=function(property,value){if($.isNumeric(value)){me[property]=Math.round(value)}}}function DisplayController(element,options,tipController){var hoverTimer=null,myCloseDelay=null;function openTooltip(immediate,forceOpen){cancelTimer();if(!element.data(DATA_HASACTIVEHOVER)){if(!immediate){session.tipOpenImminent=true;hoverTimer=setTimeout(function intentDelay(){hoverTimer=null;checkForIntent()},options.intentPollInterval)}else{if(forceOpen){element.data(DATA_FORCEDOPEN,true)}closeAnyDelayed();tipController.showTip(element)}}else{cancelClose()}}function closeTooltip(disableDelay){if(myCloseDelay){myCloseDelay=session.closeDelayTimeout=clearTimeout(myCloseDelay);session.delayInProgress=false}cancelTimer();session.tipOpenImminent=false;if(element.data(DATA_HASACTIVEHOVER)){element.data(DATA_FORCEDOPEN,false);if(!disableDelay){session.delayInProgress=true;session.closeDelayTimeout=setTimeout(function closeDelay(){session.closeDelayTimeout=null;tipController.hideTip(element);session.delayInProgress=false;myCloseDelay=null},options.closeDelay);myCloseDelay=session.closeDelayTimeout}else{tipController.hideTip(element)}}}function checkForIntent(){var xDifference=Math.abs(session.previousX-session.currentX),yDifference=Math.abs(session.previousY-session.currentY),totalDifference=xDifference+yDifference;if(totalDifference ",{id:options.popupId});if($body.length===0){$body=$("body")}$body.append(tipElement);session.tooltips=session.tooltips?session.tooltips.add(tipElement):tipElement}if(options.followMouse){if(!tipElement.data(DATA_HASMOUSEMOVE)){$document.on("mousemove"+EVENT_NAMESPACE,positionTipOnCursor);$window.on("scroll"+EVENT_NAMESPACE,positionTipOnCursor);tipElement.data(DATA_HASMOUSEMOVE,true)}}function beginShowTip(element){element.data(DATA_HASACTIVEHOVER,true);tipElement.queue(function queueTipInit(next){showTip(element);next()})}function showTip(element){var tipContent;if(!element.data(DATA_HASACTIVEHOVER)){return}if(session.isTipOpen){if(!session.isClosing){hideTip(session.activeHover)}tipElement.delay(100).queue(function queueTipAgain(next){showTip(element);next()});return}element.trigger("powerTipPreRender");tipContent=getTooltipContent(element);if(tipContent){tipElement.empty().append(tipContent)}else{return}element.trigger("powerTipRender");session.activeHover=element;session.isTipOpen=true;tipElement.data(DATA_MOUSEONTOTIP,options.mouseOnToPopup);tipElement.addClass(options.popupClass);if(!options.followMouse||element.data(DATA_FORCEDOPEN)){positionTipOnElement(element);session.isFixedTipOpen=true}else{positionTipOnCursor()}if(!element.data(DATA_FORCEDOPEN)&&!options.followMouse){$document.on("click"+EVENT_NAMESPACE,function documentClick(event){var target=event.target;if(target!==element[0]){if(options.mouseOnToPopup){if(target!==tipElement[0]&&!$.contains(tipElement[0],target)){$.powerTip.hide()}}else{$.powerTip.hide()}}})}if(options.mouseOnToPopup&&!options.manual){tipElement.on("mouseenter"+EVENT_NAMESPACE,function tipMouseEnter(){if(session.activeHover){session.activeHover.data(DATA_DISPLAYCONTROLLER).cancel()}});tipElement.on("mouseleave"+EVENT_NAMESPACE,function tipMouseLeave(){if(session.activeHover){session.activeHover.data(DATA_DISPLAYCONTROLLER).hide()}})}tipElement.fadeIn(options.fadeInTime,function fadeInCallback(){if(!session.desyncTimeout){session.desyncTimeout=setInterval(closeDesyncedTip,500)}element.trigger("powerTipOpen")})}function hideTip(element){session.isClosing=true;session.isTipOpen=false;session.desyncTimeout=clearInterval(session.desyncTimeout);element.data(DATA_HASACTIVEHOVER,false);element.data(DATA_FORCEDOPEN,false);$document.off("click"+EVENT_NAMESPACE);tipElement.off(EVENT_NAMESPACE);tipElement.fadeOut(options.fadeOutTime,function fadeOutCallback(){var coords=new CSSCoordinates;session.activeHover=null;session.isClosing=false;session.isFixedTipOpen=false;tipElement.removeClass();coords.set("top",session.currentY+options.offset);coords.set("left",session.currentX+options.offset);tipElement.css(coords);element.trigger("powerTipClose")})}function positionTipOnCursor(){var tipWidth,tipHeight,coords,collisions,collisionCount;if(!session.isFixedTipOpen&&(session.isTipOpen||session.tipOpenImminent&&tipElement.data(DATA_HASMOUSEMOVE))){tipWidth=tipElement.outerWidth();tipHeight=tipElement.outerHeight();coords=new CSSCoordinates;coords.set("top",session.currentY+options.offset);coords.set("left",session.currentX+options.offset);collisions=getViewportCollisions(coords,tipWidth,tipHeight);if(collisions!==Collision.none){collisionCount=countFlags(collisions);if(collisionCount===1){if(collisions===Collision.right){coords.set("left",session.scrollLeft+session.windowWidth-tipWidth)}else if(collisions===Collision.bottom){coords.set("top",session.scrollTop+session.windowHeight-tipHeight)}}else{coords.set("left",session.currentX-tipWidth-options.offset);coords.set("top",session.currentY-tipHeight-options.offset)}}tipElement.css(coords)}}function positionTipOnElement(element){var priorityList,finalPlacement;if(options.smartPlacement||options.followMouse&&element.data(DATA_FORCEDOPEN)){priorityList=$.fn.powerTip.smartPlacementLists[options.placement];$.each(priorityList,function(idx,pos){var collisions=getViewportCollisions(placeTooltip(element,pos),tipElement.outerWidth(),tipElement.outerHeight());finalPlacement=pos;return collisions!==Collision.none})}else{placeTooltip(element,options.placement);finalPlacement=options.placement}tipElement.removeClass("w nw sw e ne se n s w se-alt sw-alt ne-alt nw-alt");tipElement.addClass(finalPlacement)}function placeTooltip(element,placement){var iterationCount=0,tipWidth,tipHeight,coords=new CSSCoordinates;coords.set("top",0);coords.set("left",0);tipElement.css(coords);do{tipWidth=tipElement.outerWidth();tipHeight=tipElement.outerHeight();coords=placementCalculator.compute(element,placement,tipWidth,tipHeight,options.offset);tipElement.css(coords)}while(++iterationCount<=5&&(tipWidth!==tipElement.outerWidth()||tipHeight!==tipElement.outerHeight()));return coords}function closeDesyncedTip(){var isDesynced=false,hasDesyncableCloseEvent=$.grep(["mouseleave","mouseout","blur","focusout"],function(eventType){return $.inArray(eventType,options.closeEvents)!==-1}).length>0;if(session.isTipOpen&&!session.isClosing&&!session.delayInProgress&&hasDesyncableCloseEvent){if(session.activeHover.data(DATA_HASACTIVEHOVER)===false||session.activeHover.is(":disabled")){isDesynced=true}else if(!isMouseOver(session.activeHover)&&!session.activeHover.is(":focus")&&!session.activeHover.data(DATA_FORCEDOPEN)){if(tipElement.data(DATA_MOUSEONTOTIP)){if(!isMouseOver(tipElement)){isDesynced=true}}else{isDesynced=true}}if(isDesynced){hideTip(session.activeHover)}}}this.showTip=beginShowTip;this.hideTip=hideTip;this.resetPosition=positionTipOnElement}function isSvgElement(element){return Boolean(window.SVGElement&&element[0]instanceof SVGElement)}function isMouseEvent(event){return Boolean(event&&$.inArray(event.type,MOUSE_EVENTS)>-1&&typeof event.pageX==="number")}function initTracking(){if(!session.mouseTrackingActive){session.mouseTrackingActive=true;getViewportDimensions();$(getViewportDimensions);$document.on("mousemove"+EVENT_NAMESPACE,trackMouse);$window.on("resize"+EVENT_NAMESPACE,trackResize);$window.on("scroll"+EVENT_NAMESPACE,trackScroll)}}function getViewportDimensions(){session.scrollLeft=$window.scrollLeft();session.scrollTop=$window.scrollTop();session.windowWidth=$window.width();session.windowHeight=$window.height()}function trackResize(){session.windowWidth=$window.width();session.windowHeight=$window.height()}function trackScroll(){var x=$window.scrollLeft(),y=$window.scrollTop();if(x!==session.scrollLeft){session.currentX+=x-session.scrollLeft;session.scrollLeft=x}if(y!==session.scrollTop){session.currentY+=y-session.scrollTop;session.scrollTop=y}}function trackMouse(event){session.currentX=event.pageX;session.currentY=event.pageY}function isMouseOver(element){var elementPosition=element.offset(),elementBox=element[0].getBoundingClientRect(),elementWidth=elementBox.right-elementBox.left,elementHeight=elementBox.bottom-elementBox.top;return session.currentX>=elementPosition.left&&session.currentX<=elementPosition.left+elementWidth&&session.currentY>=elementPosition.top&&session.currentY<=elementPosition.top+elementHeight}function getTooltipContent(element){var tipText=element.data(DATA_POWERTIP),tipObject=element.data(DATA_POWERTIPJQ),tipTarget=element.data(DATA_POWERTIPTARGET),targetElement,content;if(tipText){if($.isFunction(tipText)){tipText=tipText.call(element[0])}content=tipText}else if(tipObject){if($.isFunction(tipObject)){tipObject=tipObject.call(element[0])}if(tipObject.length>0){content=tipObject.clone(true,true)}}else if(tipTarget){targetElement=$("#"+tipTarget);if(targetElement.length>0){content=targetElement.html()}}return content}function getViewportCollisions(coords,elementWidth,elementHeight){var viewportTop=session.scrollTop,viewportLeft=session.scrollLeft,viewportBottom=viewportTop+session.windowHeight,viewportRight=viewportLeft+session.windowWidth,collisions=Collision.none;if(coords.topviewportBottom||Math.abs(coords.bottom-session.windowHeight)>viewportBottom){collisions|=Collision.bottom}if(coords.leftviewportRight){collisions|=Collision.left}if(coords.left+elementWidth>viewportRight||coords.right1)){a.preventDefault();var c=a.originalEvent.changedTouches[0],d=document.createEvent("MouseEvents");d.initMouseEvent(b,!0,!0,window,1,c.screenX,c.screenY,c.clientX,c.clientY,!1,!1,!1,!1,0,null),a.target.dispatchEvent(d)}}if(a.support.touch="ontouchend"in document,a.support.touch){var e,b=a.ui.mouse.prototype,c=b._mouseInit,d=b._mouseDestroy;b._touchStart=function(a){var b=this;!e&&b._mouseCapture(a.originalEvent.changedTouches[0])&&(e=!0,b._touchMoved=!1,f(a,"mouseover"),f(a,"mousemove"),f(a,"mousedown"))},b._touchMove=function(a){e&&(this._touchMoved=!0,f(a,"mousemove"))},b._touchEnd=function(a){e&&(f(a,"mouseup"),f(a,"mouseout"),this._touchMoved||f(a,"click"),e=!1)},b._mouseInit=function(){var b=this;b.element.bind({touchstart:a.proxy(b,"_touchStart"),touchmove:a.proxy(b,"_touchMove"),touchend:a.proxy(b,"_touchEnd")}),c.call(b)},b._mouseDestroy=function(){var b=this;b.element.unbind({touchstart:a.proxy(b,"_touchStart"),touchmove:a.proxy(b,"_touchMove"),touchend:a.proxy(b,"_touchEnd")}),d.call(b)}}}(jQuery);/*! SmartMenus jQuery Plugin - v1.1.0 - September 17, 2017
+ * http://www.smartmenus.org/
+ * Copyright Vasil Dinkov, Vadikom Web Ltd. http://vadikom.com; Licensed MIT */(function(t){"function"==typeof define&&define.amd?define(["jquery"],t):"object"==typeof module&&"object"==typeof module.exports?module.exports=t(require("jquery")):t(jQuery)})(function($){function initMouseDetection(t){var e=".smartmenus_mouse";if(mouseDetectionEnabled||t)mouseDetectionEnabled&&t&&($(document).off(e),mouseDetectionEnabled=!1);else{var i=!0,s=null,o={mousemove:function(t){var e={x:t.pageX,y:t.pageY,timeStamp:(new Date).getTime()};if(s){var o=Math.abs(s.x-e.x),a=Math.abs(s.y-e.y);if((o>0||a>0)&&2>=o&&2>=a&&300>=e.timeStamp-s.timeStamp&&(mouse=!0,i)){var n=$(t.target).closest("a");n.is("a")&&$.each(menuTrees,function(){return $.contains(this.$root[0],n[0])?(this.itemEnter({currentTarget:n[0]}),!1):void 0}),i=!1}}s=e}};o[touchEvents?"touchstart":"pointerover pointermove pointerout MSPointerOver MSPointerMove MSPointerOut"]=function(t){isTouchEvent(t.originalEvent)&&(mouse=!1)},$(document).on(getEventsNS(o,e)),mouseDetectionEnabled=!0}}function isTouchEvent(t){return!/^(4|mouse)$/.test(t.pointerType)}function getEventsNS(t,e){e||(e="");var i={};for(var s in t)i[s.split(" ").join(e+" ")+e]=t[s];return i}var menuTrees=[],mouse=!1,touchEvents="ontouchstart"in window,mouseDetectionEnabled=!1,requestAnimationFrame=window.requestAnimationFrame||function(t){return setTimeout(t,1e3/60)},cancelAnimationFrame=window.cancelAnimationFrame||function(t){clearTimeout(t)},canAnimate=!!$.fn.animate;return $.SmartMenus=function(t,e){this.$root=$(t),this.opts=e,this.rootId="",this.accessIdPrefix="",this.$subArrow=null,this.activatedItems=[],this.visibleSubMenus=[],this.showTimeout=0,this.hideTimeout=0,this.scrollTimeout=0,this.clickActivated=!1,this.focusActivated=!1,this.zIndexInc=0,this.idInc=0,this.$firstLink=null,this.$firstSub=null,this.disabled=!1,this.$disableOverlay=null,this.$touchScrollingSub=null,this.cssTransforms3d="perspective"in t.style||"webkitPerspective"in t.style,this.wasCollapsible=!1,this.init()},$.extend($.SmartMenus,{hideAll:function(){$.each(menuTrees,function(){this.menuHideAll()})},destroy:function(){for(;menuTrees.length;)menuTrees[0].destroy();initMouseDetection(!0)},prototype:{init:function(t){var e=this;if(!t){menuTrees.push(this),this.rootId=((new Date).getTime()+Math.random()+"").replace(/\D/g,""),this.accessIdPrefix="sm-"+this.rootId+"-",this.$root.hasClass("sm-rtl")&&(this.opts.rightToLeftSubMenus=!0);var i=".smartmenus";this.$root.data("smartmenus",this).attr("data-smartmenus-id",this.rootId).dataSM("level",1).on(getEventsNS({"mouseover focusin":$.proxy(this.rootOver,this),"mouseout focusout":$.proxy(this.rootOut,this),keydown:$.proxy(this.rootKeyDown,this)},i)).on(getEventsNS({mouseenter:$.proxy(this.itemEnter,this),mouseleave:$.proxy(this.itemLeave,this),mousedown:$.proxy(this.itemDown,this),focus:$.proxy(this.itemFocus,this),blur:$.proxy(this.itemBlur,this),click:$.proxy(this.itemClick,this)},i),"a"),i+=this.rootId,this.opts.hideOnClick&&$(document).on(getEventsNS({touchstart:$.proxy(this.docTouchStart,this),touchmove:$.proxy(this.docTouchMove,this),touchend:$.proxy(this.docTouchEnd,this),click:$.proxy(this.docClick,this)},i)),$(window).on(getEventsNS({"resize orientationchange":$.proxy(this.winResize,this)},i)),this.opts.subIndicators&&(this.$subArrow=$(" ").addClass("sub-arrow"),this.opts.subIndicatorsText&&this.$subArrow.html(this.opts.subIndicatorsText)),initMouseDetection()}if(this.$firstSub=this.$root.find("ul").each(function(){e.menuInit($(this))}).eq(0),this.$firstLink=this.$root.find("a").eq(0),this.opts.markCurrentItem){var s=/(index|default)\.[^#\?\/]*/i,o=/#.*/,a=window.location.href.replace(s,""),n=a.replace(o,"");this.$root.find("a").each(function(){var t=this.href.replace(s,""),i=$(this);(t==a||t==n)&&(i.addClass("current"),e.opts.markCurrentTree&&i.parentsUntil("[data-smartmenus-id]","ul").each(function(){$(this).dataSM("parent-a").addClass("current")}))})}this.wasCollapsible=this.isCollapsible()},destroy:function(t){if(!t){var e=".smartmenus";this.$root.removeData("smartmenus").removeAttr("data-smartmenus-id").removeDataSM("level").off(e),e+=this.rootId,$(document).off(e),$(window).off(e),this.opts.subIndicators&&(this.$subArrow=null)}this.menuHideAll();var i=this;this.$root.find("ul").each(function(){var t=$(this);t.dataSM("scroll-arrows")&&t.dataSM("scroll-arrows").remove(),t.dataSM("shown-before")&&((i.opts.subMenusMinWidth||i.opts.subMenusMaxWidth)&&t.css({width:"",minWidth:"",maxWidth:""}).removeClass("sm-nowrap"),t.dataSM("scroll-arrows")&&t.dataSM("scroll-arrows").remove(),t.css({zIndex:"",top:"",left:"",marginLeft:"",marginTop:"",display:""})),0==(t.attr("id")||"").indexOf(i.accessIdPrefix)&&t.removeAttr("id")}).removeDataSM("in-mega").removeDataSM("shown-before").removeDataSM("scroll-arrows").removeDataSM("parent-a").removeDataSM("level").removeDataSM("beforefirstshowfired").removeAttr("role").removeAttr("aria-hidden").removeAttr("aria-labelledby").removeAttr("aria-expanded"),this.$root.find("a.has-submenu").each(function(){var t=$(this);0==t.attr("id").indexOf(i.accessIdPrefix)&&t.removeAttr("id")}).removeClass("has-submenu").removeDataSM("sub").removeAttr("aria-haspopup").removeAttr("aria-controls").removeAttr("aria-expanded").closest("li").removeDataSM("sub"),this.opts.subIndicators&&this.$root.find("span.sub-arrow").remove(),this.opts.markCurrentItem&&this.$root.find("a.current").removeClass("current"),t||(this.$root=null,this.$firstLink=null,this.$firstSub=null,this.$disableOverlay&&(this.$disableOverlay.remove(),this.$disableOverlay=null),menuTrees.splice($.inArray(this,menuTrees),1))},disable:function(t){if(!this.disabled){if(this.menuHideAll(),!t&&!this.opts.isPopup&&this.$root.is(":visible")){var e=this.$root.offset();this.$disableOverlay=$('
').css({position:"absolute",top:e.top,left:e.left,width:this.$root.outerWidth(),height:this.$root.outerHeight(),zIndex:this.getStartZIndex(!0),opacity:0}).appendTo(document.body)}this.disabled=!0}},docClick:function(t){return this.$touchScrollingSub?(this.$touchScrollingSub=null,void 0):((this.visibleSubMenus.length&&!$.contains(this.$root[0],t.target)||$(t.target).closest("a").length)&&this.menuHideAll(),void 0)},docTouchEnd:function(){if(this.lastTouch){if(!(!this.visibleSubMenus.length||void 0!==this.lastTouch.x2&&this.lastTouch.x1!=this.lastTouch.x2||void 0!==this.lastTouch.y2&&this.lastTouch.y1!=this.lastTouch.y2||this.lastTouch.target&&$.contains(this.$root[0],this.lastTouch.target))){this.hideTimeout&&(clearTimeout(this.hideTimeout),this.hideTimeout=0);var t=this;this.hideTimeout=setTimeout(function(){t.menuHideAll()},350)}this.lastTouch=null}},docTouchMove:function(t){if(this.lastTouch){var e=t.originalEvent.touches[0];this.lastTouch.x2=e.pageX,this.lastTouch.y2=e.pageY}},docTouchStart:function(t){var e=t.originalEvent.touches[0];this.lastTouch={x1:e.pageX,y1:e.pageY,target:e.target}},enable:function(){this.disabled&&(this.$disableOverlay&&(this.$disableOverlay.remove(),this.$disableOverlay=null),this.disabled=!1)},getClosestMenu:function(t){for(var e=$(t).closest("ul");e.dataSM("in-mega");)e=e.parent().closest("ul");return e[0]||null},getHeight:function(t){return this.getOffset(t,!0)},getOffset:function(t,e){var i;"none"==t.css("display")&&(i={position:t[0].style.position,visibility:t[0].style.visibility},t.css({position:"absolute",visibility:"hidden"}).show());var s=t[0].getBoundingClientRect&&t[0].getBoundingClientRect(),o=s&&(e?s.height||s.bottom-s.top:s.width||s.right-s.left);return o||0===o||(o=e?t[0].offsetHeight:t[0].offsetWidth),i&&t.hide().css(i),o},getStartZIndex:function(t){var e=parseInt(this[t?"$root":"$firstSub"].css("z-index"));return!t&&isNaN(e)&&(e=parseInt(this.$root.css("z-index"))),isNaN(e)?1:e},getTouchPoint:function(t){return t.touches&&t.touches[0]||t.changedTouches&&t.changedTouches[0]||t},getViewport:function(t){var e=t?"Height":"Width",i=document.documentElement["client"+e],s=window["inner"+e];return s&&(i=Math.min(i,s)),i},getViewportHeight:function(){return this.getViewport(!0)},getViewportWidth:function(){return this.getViewport()},getWidth:function(t){return this.getOffset(t)},handleEvents:function(){return!this.disabled&&this.isCSSOn()},handleItemEvents:function(t){return this.handleEvents()&&!this.isLinkInMegaMenu(t)},isCollapsible:function(){return"static"==this.$firstSub.css("position")},isCSSOn:function(){return"inline"!=this.$firstLink.css("display")},isFixed:function(){var t="fixed"==this.$root.css("position");return t||this.$root.parentsUntil("body").each(function(){return"fixed"==$(this).css("position")?(t=!0,!1):void 0}),t},isLinkInMegaMenu:function(t){return $(this.getClosestMenu(t[0])).hasClass("mega-menu")},isTouchMode:function(){return!mouse||this.opts.noMouseOver||this.isCollapsible()},itemActivate:function(t,e){var i=t.closest("ul"),s=i.dataSM("level");if(s>1&&(!this.activatedItems[s-2]||this.activatedItems[s-2][0]!=i.dataSM("parent-a")[0])){var o=this;$(i.parentsUntil("[data-smartmenus-id]","ul").get().reverse()).add(i).each(function(){o.itemActivate($(this).dataSM("parent-a"))})}if((!this.isCollapsible()||e)&&this.menuHideSubMenus(this.activatedItems[s-1]&&this.activatedItems[s-1][0]==t[0]?s:s-1),this.activatedItems[s-1]=t,this.$root.triggerHandler("activate.smapi",t[0])!==!1){var a=t.dataSM("sub");a&&(this.isTouchMode()||!this.opts.showOnClick||this.clickActivated)&&this.menuShow(a)}},itemBlur:function(t){var e=$(t.currentTarget);this.handleItemEvents(e)&&this.$root.triggerHandler("blur.smapi",e[0])},itemClick:function(t){var e=$(t.currentTarget);if(this.handleItemEvents(e)){if(this.$touchScrollingSub&&this.$touchScrollingSub[0]==e.closest("ul")[0])return this.$touchScrollingSub=null,t.stopPropagation(),!1;if(this.$root.triggerHandler("click.smapi",e[0])===!1)return!1;var i=$(t.target).is(".sub-arrow"),s=e.dataSM("sub"),o=s?2==s.dataSM("level"):!1,a=this.isCollapsible(),n=/toggle$/.test(this.opts.collapsibleBehavior),r=/link$/.test(this.opts.collapsibleBehavior),h=/^accordion/.test(this.opts.collapsibleBehavior);if(s&&!s.is(":visible")){if((!r||!a||i)&&(this.opts.showOnClick&&o&&(this.clickActivated=!0),this.itemActivate(e,h),s.is(":visible")))return this.focusActivated=!0,!1}else if(a&&(n||i))return this.itemActivate(e,h),this.menuHide(s),n&&(this.focusActivated=!1),!1;return this.opts.showOnClick&&o||e.hasClass("disabled")||this.$root.triggerHandler("select.smapi",e[0])===!1?!1:void 0}},itemDown:function(t){var e=$(t.currentTarget);this.handleItemEvents(e)&&e.dataSM("mousedown",!0)},itemEnter:function(t){var e=$(t.currentTarget);if(this.handleItemEvents(e)){if(!this.isTouchMode()){this.showTimeout&&(clearTimeout(this.showTimeout),this.showTimeout=0);var i=this;this.showTimeout=setTimeout(function(){i.itemActivate(e)},this.opts.showOnClick&&1==e.closest("ul").dataSM("level")?1:this.opts.showTimeout)}this.$root.triggerHandler("mouseenter.smapi",e[0])}},itemFocus:function(t){var e=$(t.currentTarget);this.handleItemEvents(e)&&(!this.focusActivated||this.isTouchMode()&&e.dataSM("mousedown")||this.activatedItems.length&&this.activatedItems[this.activatedItems.length-1][0]==e[0]||this.itemActivate(e,!0),this.$root.triggerHandler("focus.smapi",e[0]))},itemLeave:function(t){var e=$(t.currentTarget);this.handleItemEvents(e)&&(this.isTouchMode()||(e[0].blur(),this.showTimeout&&(clearTimeout(this.showTimeout),this.showTimeout=0)),e.removeDataSM("mousedown"),this.$root.triggerHandler("mouseleave.smapi",e[0]))},menuHide:function(t){if(this.$root.triggerHandler("beforehide.smapi",t[0])!==!1&&(canAnimate&&t.stop(!0,!0),"none"!=t.css("display"))){var e=function(){t.css("z-index","")};this.isCollapsible()?canAnimate&&this.opts.collapsibleHideFunction?this.opts.collapsibleHideFunction.call(this,t,e):t.hide(this.opts.collapsibleHideDuration,e):canAnimate&&this.opts.hideFunction?this.opts.hideFunction.call(this,t,e):t.hide(this.opts.hideDuration,e),t.dataSM("scroll")&&(this.menuScrollStop(t),t.css({"touch-action":"","-ms-touch-action":"","-webkit-transform":"",transform:""}).off(".smartmenus_scroll").removeDataSM("scroll").dataSM("scroll-arrows").hide()),t.dataSM("parent-a").removeClass("highlighted").attr("aria-expanded","false"),t.attr({"aria-expanded":"false","aria-hidden":"true"});var i=t.dataSM("level");this.activatedItems.splice(i-1,1),this.visibleSubMenus.splice($.inArray(t,this.visibleSubMenus),1),this.$root.triggerHandler("hide.smapi",t[0])}},menuHideAll:function(){this.showTimeout&&(clearTimeout(this.showTimeout),this.showTimeout=0);for(var t=this.opts.isPopup?1:0,e=this.visibleSubMenus.length-1;e>=t;e--)this.menuHide(this.visibleSubMenus[e]);this.opts.isPopup&&(canAnimate&&this.$root.stop(!0,!0),this.$root.is(":visible")&&(canAnimate&&this.opts.hideFunction?this.opts.hideFunction.call(this,this.$root):this.$root.hide(this.opts.hideDuration))),this.activatedItems=[],this.visibleSubMenus=[],this.clickActivated=!1,this.focusActivated=!1,this.zIndexInc=0,this.$root.triggerHandler("hideAll.smapi")},menuHideSubMenus:function(t){for(var e=this.activatedItems.length-1;e>=t;e--){var i=this.activatedItems[e].dataSM("sub");i&&this.menuHide(i)}},menuInit:function(t){if(!t.dataSM("in-mega")){t.hasClass("mega-menu")&&t.find("ul").dataSM("in-mega",!0);for(var e=2,i=t[0];(i=i.parentNode.parentNode)!=this.$root[0];)e++;var s=t.prevAll("a").eq(-1);s.length||(s=t.prevAll().find("a").eq(-1)),s.addClass("has-submenu").dataSM("sub",t),t.dataSM("parent-a",s).dataSM("level",e).parent().dataSM("sub",t);var o=s.attr("id")||this.accessIdPrefix+ ++this.idInc,a=t.attr("id")||this.accessIdPrefix+ ++this.idInc;s.attr({id:o,"aria-haspopup":"true","aria-controls":a,"aria-expanded":"false"}),t.attr({id:a,role:"group","aria-hidden":"true","aria-labelledby":o,"aria-expanded":"false"}),this.opts.subIndicators&&s[this.opts.subIndicatorsPos](this.$subArrow.clone())}},menuPosition:function(t){var e,i,s=t.dataSM("parent-a"),o=s.closest("li"),a=o.parent(),n=t.dataSM("level"),r=this.getWidth(t),h=this.getHeight(t),u=s.offset(),l=u.left,c=u.top,d=this.getWidth(s),m=this.getHeight(s),p=$(window),f=p.scrollLeft(),v=p.scrollTop(),b=this.getViewportWidth(),S=this.getViewportHeight(),g=a.parent().is("[data-sm-horizontal-sub]")||2==n&&!a.hasClass("sm-vertical"),M=this.opts.rightToLeftSubMenus&&!o.is("[data-sm-reverse]")||!this.opts.rightToLeftSubMenus&&o.is("[data-sm-reverse]"),w=2==n?this.opts.mainMenuSubOffsetX:this.opts.subMenusSubOffsetX,T=2==n?this.opts.mainMenuSubOffsetY:this.opts.subMenusSubOffsetY;if(g?(e=M?d-r-w:w,i=this.opts.bottomToTopSubMenus?-h-T:m+T):(e=M?w-r:d-w,i=this.opts.bottomToTopSubMenus?m-T-h:T),this.opts.keepInViewport){var y=l+e,I=c+i;if(M&&f>y?e=g?f-y+e:d-w:!M&&y+r>f+b&&(e=g?f+b-r-y+e:w-r),g||(S>h&&I+h>v+S?i+=v+S-h-I:(h>=S||v>I)&&(i+=v-I)),g&&(I+h>v+S+.49||v>I)||!g&&h>S+.49){var x=this;t.dataSM("scroll-arrows")||t.dataSM("scroll-arrows",$([$(' ')[0],$(' ')[0]]).on({mouseenter:function(){t.dataSM("scroll").up=$(this).hasClass("scroll-up"),x.menuScroll(t)},mouseleave:function(e){x.menuScrollStop(t),x.menuScrollOut(t,e)},"mousewheel DOMMouseScroll":function(t){t.preventDefault()}}).insertAfter(t));var A=".smartmenus_scroll";if(t.dataSM("scroll",{y:this.cssTransforms3d?0:i-m,step:1,itemH:m,subH:h,arrowDownH:this.getHeight(t.dataSM("scroll-arrows").eq(1))}).on(getEventsNS({mouseover:function(e){x.menuScrollOver(t,e)},mouseout:function(e){x.menuScrollOut(t,e)},"mousewheel DOMMouseScroll":function(e){x.menuScrollMousewheel(t,e)}},A)).dataSM("scroll-arrows").css({top:"auto",left:"0",marginLeft:e+(parseInt(t.css("border-left-width"))||0),width:r-(parseInt(t.css("border-left-width"))||0)-(parseInt(t.css("border-right-width"))||0),zIndex:t.css("z-index")}).eq(g&&this.opts.bottomToTopSubMenus?0:1).show(),this.isFixed()){var C={};C[touchEvents?"touchstart touchmove touchend":"pointerdown pointermove pointerup MSPointerDown MSPointerMove MSPointerUp"]=function(e){x.menuScrollTouch(t,e)},t.css({"touch-action":"none","-ms-touch-action":"none"}).on(getEventsNS(C,A))}}}t.css({top:"auto",left:"0",marginLeft:e,marginTop:i-m})},menuScroll:function(t,e,i){var s,o=t.dataSM("scroll"),a=t.dataSM("scroll-arrows"),n=o.up?o.upEnd:o.downEnd;if(!e&&o.momentum){if(o.momentum*=.92,s=o.momentum,.5>s)return this.menuScrollStop(t),void 0}else s=i||(e||!this.opts.scrollAccelerate?this.opts.scrollStep:Math.floor(o.step));var r=t.dataSM("level");if(this.activatedItems[r-1]&&this.activatedItems[r-1].dataSM("sub")&&this.activatedItems[r-1].dataSM("sub").is(":visible")&&this.menuHideSubMenus(r-1),o.y=o.up&&o.y>=n||!o.up&&n>=o.y?o.y:Math.abs(n-o.y)>s?o.y+(o.up?s:-s):n,t.css(this.cssTransforms3d?{"-webkit-transform":"translate3d(0, "+o.y+"px, 0)",transform:"translate3d(0, "+o.y+"px, 0)"}:{marginTop:o.y}),mouse&&(o.up&&o.y>o.downEnd||!o.up&&o.y0;t.dataSM("scroll-arrows").eq(i?0:1).is(":visible")&&(t.dataSM("scroll").up=i,this.menuScroll(t,!0))}e.preventDefault()},menuScrollOut:function(t,e){mouse&&(/^scroll-(up|down)/.test((e.relatedTarget||"").className)||(t[0]==e.relatedTarget||$.contains(t[0],e.relatedTarget))&&this.getClosestMenu(e.relatedTarget)==t[0]||t.dataSM("scroll-arrows").css("visibility","hidden"))},menuScrollOver:function(t,e){if(mouse&&!/^scroll-(up|down)/.test(e.target.className)&&this.getClosestMenu(e.target)==t[0]){this.menuScrollRefreshData(t);var i=t.dataSM("scroll"),s=$(window).scrollTop()-t.dataSM("parent-a").offset().top-i.itemH;t.dataSM("scroll-arrows").eq(0).css("margin-top",s).end().eq(1).css("margin-top",s+this.getViewportHeight()-i.arrowDownH).end().css("visibility","visible")}},menuScrollRefreshData:function(t){var e=t.dataSM("scroll"),i=$(window).scrollTop()-t.dataSM("parent-a").offset().top-e.itemH;this.cssTransforms3d&&(i=-(parseFloat(t.css("margin-top"))-i)),$.extend(e,{upEnd:i,downEnd:i+this.getViewportHeight()-e.subH})},menuScrollStop:function(t){return this.scrollTimeout?(cancelAnimationFrame(this.scrollTimeout),this.scrollTimeout=0,t.dataSM("scroll").step=1,!0):void 0},menuScrollTouch:function(t,e){if(e=e.originalEvent,isTouchEvent(e)){var i=this.getTouchPoint(e);if(this.getClosestMenu(i.target)==t[0]){var s=t.dataSM("scroll");if(/(start|down)$/i.test(e.type))this.menuScrollStop(t)?(e.preventDefault(),this.$touchScrollingSub=t):this.$touchScrollingSub=null,this.menuScrollRefreshData(t),$.extend(s,{touchStartY:i.pageY,touchStartTime:e.timeStamp});else if(/move$/i.test(e.type)){var o=void 0!==s.touchY?s.touchY:s.touchStartY;if(void 0!==o&&o!=i.pageY){this.$touchScrollingSub=t;var a=i.pageY>o;void 0!==s.up&&s.up!=a&&$.extend(s,{touchStartY:i.pageY,touchStartTime:e.timeStamp}),$.extend(s,{up:a,touchY:i.pageY}),this.menuScroll(t,!0,Math.abs(i.pageY-o))}e.preventDefault()}else void 0!==s.touchY&&((s.momentum=15*Math.pow(Math.abs(i.pageY-s.touchStartY)/(e.timeStamp-s.touchStartTime),2))&&(this.menuScrollStop(t),this.menuScroll(t),e.preventDefault()),delete s.touchY)}}},menuShow:function(t){if((t.dataSM("beforefirstshowfired")||(t.dataSM("beforefirstshowfired",!0),this.$root.triggerHandler("beforefirstshow.smapi",t[0])!==!1))&&this.$root.triggerHandler("beforeshow.smapi",t[0])!==!1&&(t.dataSM("shown-before",!0),canAnimate&&t.stop(!0,!0),!t.is(":visible"))){var e=t.dataSM("parent-a"),i=this.isCollapsible();if((this.opts.keepHighlighted||i)&&e.addClass("highlighted"),i)t.removeClass("sm-nowrap").css({zIndex:"",width:"auto",minWidth:"",maxWidth:"",top:"",left:"",marginLeft:"",marginTop:""});else{if(t.css("z-index",this.zIndexInc=(this.zIndexInc||this.getStartZIndex())+1),(this.opts.subMenusMinWidth||this.opts.subMenusMaxWidth)&&(t.css({width:"auto",minWidth:"",maxWidth:""}).addClass("sm-nowrap"),this.opts.subMenusMinWidth&&t.css("min-width",this.opts.subMenusMinWidth),this.opts.subMenusMaxWidth)){var s=this.getWidth(t);t.css("max-width",this.opts.subMenusMaxWidth),s>this.getWidth(t)&&t.removeClass("sm-nowrap").css("width",this.opts.subMenusMaxWidth)}this.menuPosition(t)}var o=function(){t.css("overflow","")};i?canAnimate&&this.opts.collapsibleShowFunction?this.opts.collapsibleShowFunction.call(this,t,o):t.show(this.opts.collapsibleShowDuration,o):canAnimate&&this.opts.showFunction?this.opts.showFunction.call(this,t,o):t.show(this.opts.showDuration,o),e.attr("aria-expanded","true"),t.attr({"aria-expanded":"true","aria-hidden":"false"}),this.visibleSubMenus.push(t),this.$root.triggerHandler("show.smapi",t[0])}},popupHide:function(t){this.hideTimeout&&(clearTimeout(this.hideTimeout),this.hideTimeout=0);var e=this;this.hideTimeout=setTimeout(function(){e.menuHideAll()},t?1:this.opts.hideTimeout)},popupShow:function(t,e){if(!this.opts.isPopup)return alert('SmartMenus jQuery Error:\n\nIf you want to show this menu via the "popupShow" method, set the isPopup:true option.'),void 0;if(this.hideTimeout&&(clearTimeout(this.hideTimeout),this.hideTimeout=0),this.$root.dataSM("shown-before",!0),canAnimate&&this.$root.stop(!0,!0),!this.$root.is(":visible")){this.$root.css({left:t,top:e});var i=this,s=function(){i.$root.css("overflow","")};canAnimate&&this.opts.showFunction?this.opts.showFunction.call(this,this.$root,s):this.$root.show(this.opts.showDuration,s),this.visibleSubMenus[0]=this.$root}},refresh:function(){this.destroy(!0),this.init(!0)},rootKeyDown:function(t){if(this.handleEvents())switch(t.keyCode){case 27:var e=this.activatedItems[0];if(e){this.menuHideAll(),e[0].focus();var i=e.dataSM("sub");i&&this.menuHide(i)}break;case 32:var s=$(t.target);if(s.is("a")&&this.handleItemEvents(s)){var i=s.dataSM("sub");i&&!i.is(":visible")&&(this.itemClick({currentTarget:t.target}),t.preventDefault())}}},rootOut:function(t){if(this.handleEvents()&&!this.isTouchMode()&&t.target!=this.$root[0]&&(this.hideTimeout&&(clearTimeout(this.hideTimeout),this.hideTimeout=0),!this.opts.showOnClick||!this.opts.hideOnClick)){var e=this;this.hideTimeout=setTimeout(function(){e.menuHideAll()},this.opts.hideTimeout)}},rootOver:function(t){this.handleEvents()&&!this.isTouchMode()&&t.target!=this.$root[0]&&this.hideTimeout&&(clearTimeout(this.hideTimeout),this.hideTimeout=0)},winResize:function(t){if(this.handleEvents()){if(!("onorientationchange"in window)||"orientationchange"==t.type){var e=this.isCollapsible();this.wasCollapsible&&e||(this.activatedItems.length&&this.activatedItems[this.activatedItems.length-1][0].blur(),this.menuHideAll()),this.wasCollapsible=e}}else if(this.$disableOverlay){var i=this.$root.offset();this.$disableOverlay.css({top:i.top,left:i.left,width:this.$root.outerWidth(),height:this.$root.outerHeight()})}}}}),$.fn.dataSM=function(t,e){return e?this.data(t+"_smartmenus",e):this.data(t+"_smartmenus")},$.fn.removeDataSM=function(t){return this.removeData(t+"_smartmenus")},$.fn.smartmenus=function(options){if("string"==typeof options){var args=arguments,method=options;return Array.prototype.shift.call(args),this.each(function(){var t=$(this).data("smartmenus");t&&t[method]&&t[method].apply(t,args)})}return this.each(function(){var dataOpts=$(this).data("sm-options")||null;if(dataOpts)try{dataOpts=eval("("+dataOpts+")")}catch(e){dataOpts=null,alert('ERROR\n\nSmartMenus jQuery init:\nInvalid "data-sm-options" attribute value syntax.')}new $.SmartMenus(this,$.extend({},$.fn.smartmenus.defaults,options,dataOpts))})},$.fn.smartmenus.defaults={isPopup:!1,mainMenuSubOffsetX:0,mainMenuSubOffsetY:0,subMenusSubOffsetX:0,subMenusSubOffsetY:0,subMenusMinWidth:"10em",subMenusMaxWidth:"20em",subIndicators:!0,subIndicatorsPos:"append",subIndicatorsText:"",scrollStep:30,scrollAccelerate:!0,showTimeout:250,hideTimeout:500,showDuration:0,showFunction:null,hideDuration:0,hideFunction:function(t,e){t.fadeOut(200,e)},collapsibleShowDuration:0,collapsibleShowFunction:function(t,e){t.slideDown(200,e)},collapsibleHideDuration:0,collapsibleHideFunction:function(t,e){t.slideUp(200,e)},showOnClick:!1,hideOnClick:!0,noMouseOver:!1,keepInViewport:!0,keepHighlighted:!0,markCurrentItem:!1,markCurrentTree:!0,rightToLeftSubMenus:!1,bottomToTopSubMenus:!1,collapsibleBehavior:"default"},$});
diff --git a/docs/pyShelf/html/menudata.js b/docs/pyShelf/html/menudata.js
new file mode 100644
index 0000000..8006205
--- /dev/null
+++ b/docs/pyShelf/html/menudata.js
@@ -0,0 +1,66 @@
+/*
+ @licstart The following is the entire license notice for the JavaScript code in this file.
+
+ The MIT License (MIT)
+
+ Copyright (C) 1997-2020 by Dimitri van Heesch
+
+ Permission is hereby granted, free of charge, to any person obtaining a copy of this software
+ and associated documentation files (the "Software"), to deal in the Software without restriction,
+ including without limitation the rights to use, copy, modify, merge, publish, distribute,
+ sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is
+ furnished to do so, subject to the following conditions:
+
+ The above copyright notice and this permission notice shall be included in all copies or
+ substantial portions of the Software.
+
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
+ BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
+ DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+
+ @licend The above is the entire license notice for the JavaScript code in this file
+*/
+var menudata={children:[
+{text:"Main Page",url:"index.html"},
+{text:"Namespaces",url:"namespaces.html",children:[
+{text:"Namespace List",url:"namespaces.html"},
+{text:"Namespace Members",url:"namespacemembers.html",children:[
+{text:"All",url:"namespacemembers.html"},
+{text:"Functions",url:"namespacemembers_func.html"}]}]},
+{text:"Classes",url:"annotated.html",children:[
+{text:"Class List",url:"annotated.html"},
+{text:"Class Index",url:"classes.html"},
+{text:"Class Hierarchy",url:"hierarchy.html"},
+{text:"Class Members",url:"functions.html",children:[
+{text:"All",url:"functions.html",children:[
+{text:"_",url:"functions.html#index__5F"},
+{text:"b",url:"functions.html#index_b"},
+{text:"c",url:"functions.html#index_c"},
+{text:"e",url:"functions.html#index_e"},
+{text:"f",url:"functions.html#index_f"},
+{text:"g",url:"functions.html#index_g"},
+{text:"i",url:"functions.html#index_i"},
+{text:"m",url:"functions.html#index_m"},
+{text:"o",url:"functions.html#index_o"},
+{text:"p",url:"functions.html#index_p"},
+{text:"r",url:"functions.html#index_r"},
+{text:"s",url:"functions.html#index_s"},
+{text:"u",url:"functions.html#index_u"}]},
+{text:"Functions",url:"functions_func.html",children:[
+{text:"_",url:"functions_func.html#index__5F"},
+{text:"b",url:"functions_func.html#index_b"},
+{text:"c",url:"functions_func.html#index_c"},
+{text:"e",url:"functions_func.html#index_e"},
+{text:"f",url:"functions_func.html#index_f"},
+{text:"g",url:"functions_func.html#index_g"},
+{text:"i",url:"functions_func.html#index_i"},
+{text:"m",url:"functions_func.html#index_m"},
+{text:"o",url:"functions_func.html#index_o"},
+{text:"p",url:"functions_func.html#index_p"},
+{text:"r",url:"functions_func.html#index_r"},
+{text:"s",url:"functions_func.html#index_s"},
+{text:"u",url:"functions_func.html#index_u"}]}]}]},
+{text:"Files",url:"files.html",children:[
+{text:"File List",url:"files.html"}]}]}
diff --git a/docs/pyShelf/html/namespacemembers.html b/docs/pyShelf/html/namespacemembers.html
new file mode 100644
index 0000000..2c225d4
--- /dev/null
+++ b/docs/pyShelf/html/namespacemembers.html
@@ -0,0 +1,67 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Namespace Members
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
Here is a list of all documented namespace members with links to the namespaces they belong to:
+
+
+
+
+
diff --git a/docs/pyShelf/html/namespacemembers_func.html b/docs/pyShelf/html/namespacemembers_func.html
new file mode 100644
index 0000000..4bcfa7c
--- /dev/null
+++ b/docs/pyShelf/html/namespacemembers_func.html
@@ -0,0 +1,67 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Namespace Members
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/docs/pyShelf/html/namespaces.html b/docs/pyShelf/html/namespaces.html
new file mode 100644
index 0000000..6ce007e
--- /dev/null
+++ b/docs/pyShelf/html/namespaces.html
@@ -0,0 +1,66 @@
+
+
+
+
+
+
+
+pyShelf Open Source Ebook Server: Namespace List
+
+
+
+
+
+
+
+
+
+
+
+
+ pyShelf Open Source Ebook Server 0.8.0
+
+ FOSS E-Book Server, https://pyshelf.com
+
+
+
+
+
+
+
+
+
+
+
+
+
Here is a list of all documented namespaces with brief descriptions:
+
+
+
+
+
diff --git a/docs/pyShelf/html/nav_f.png b/docs/pyShelf/html/nav_f.png
new file mode 100644
index 0000000..72a58a5
Binary files /dev/null and b/docs/pyShelf/html/nav_f.png differ
diff --git a/docs/pyShelf/html/nav_fd.png b/docs/pyShelf/html/nav_fd.png
new file mode 100644
index 0000000..032fbdd
Binary files /dev/null and b/docs/pyShelf/html/nav_fd.png differ
diff --git a/docs/pyShelf/html/nav_g.png b/docs/pyShelf/html/nav_g.png
new file mode 100644
index 0000000..2093a23
Binary files /dev/null and b/docs/pyShelf/html/nav_g.png differ
diff --git a/docs/pyShelf/html/nav_h.png b/docs/pyShelf/html/nav_h.png
new file mode 100644
index 0000000..33389b1
Binary files /dev/null and b/docs/pyShelf/html/nav_h.png differ
diff --git a/docs/pyShelf/html/nav_hd.png b/docs/pyShelf/html/nav_hd.png
new file mode 100644
index 0000000..de80f18
Binary files /dev/null and b/docs/pyShelf/html/nav_hd.png differ
diff --git a/docs/pyShelf/html/open.png b/docs/pyShelf/html/open.png
new file mode 100644
index 0000000..30f75c7
Binary files /dev/null and b/docs/pyShelf/html/open.png differ
diff --git a/docs/pyShelf/html/splitbar.png b/docs/pyShelf/html/splitbar.png
new file mode 100644
index 0000000..fe895f2
Binary files /dev/null and b/docs/pyShelf/html/splitbar.png differ
diff --git a/docs/pyShelf/html/splitbard.png b/docs/pyShelf/html/splitbard.png
new file mode 100644
index 0000000..8367416
Binary files /dev/null and b/docs/pyShelf/html/splitbard.png differ
diff --git a/docs/pyShelf/html/sync_off.png b/docs/pyShelf/html/sync_off.png
new file mode 100644
index 0000000..3b443fc
Binary files /dev/null and b/docs/pyShelf/html/sync_off.png differ
diff --git a/docs/pyShelf/html/sync_on.png b/docs/pyShelf/html/sync_on.png
new file mode 100644
index 0000000..e08320f
Binary files /dev/null and b/docs/pyShelf/html/sync_on.png differ
diff --git a/docs/pyShelf/html/tab_a.png b/docs/pyShelf/html/tab_a.png
new file mode 100644
index 0000000..3b725c4
Binary files /dev/null and b/docs/pyShelf/html/tab_a.png differ
diff --git a/docs/pyShelf/html/tab_ad.png b/docs/pyShelf/html/tab_ad.png
new file mode 100644
index 0000000..e34850a
Binary files /dev/null and b/docs/pyShelf/html/tab_ad.png differ
diff --git a/docs/pyShelf/html/tab_b.png b/docs/pyShelf/html/tab_b.png
new file mode 100644
index 0000000..e2b4a86
Binary files /dev/null and b/docs/pyShelf/html/tab_b.png differ
diff --git a/docs/pyShelf/html/tab_bd.png b/docs/pyShelf/html/tab_bd.png
new file mode 100644
index 0000000..91c2524
Binary files /dev/null and b/docs/pyShelf/html/tab_bd.png differ
diff --git a/docs/pyShelf/html/tab_h.png b/docs/pyShelf/html/tab_h.png
new file mode 100644
index 0000000..fd5cb70
Binary files /dev/null and b/docs/pyShelf/html/tab_h.png differ
diff --git a/docs/pyShelf/html/tab_hd.png b/docs/pyShelf/html/tab_hd.png
new file mode 100644
index 0000000..2489273
Binary files /dev/null and b/docs/pyShelf/html/tab_hd.png differ
diff --git a/docs/pyShelf/html/tab_s.png b/docs/pyShelf/html/tab_s.png
new file mode 100644
index 0000000..ab478c9
Binary files /dev/null and b/docs/pyShelf/html/tab_s.png differ
diff --git a/docs/pyShelf/html/tab_sd.png b/docs/pyShelf/html/tab_sd.png
new file mode 100644
index 0000000..757a565
Binary files /dev/null and b/docs/pyShelf/html/tab_sd.png differ
diff --git a/docs/pyShelf/man/man3/DataHooks.3 b/docs/pyShelf/man/man3/DataHooks.3
new file mode 100644
index 0000000..819ce33
--- /dev/null
+++ b/docs/pyShelf/man/man3/DataHooks.3
@@ -0,0 +1,27 @@
+.TH "DataHooks" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+DataHooks
+.SH SYNOPSIS
+.br
+.PP
+.SS "Classes"
+
+.in +1c
+.ti -1c
+.RI "class \fBBookInterface\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+pyShelf's frontend database hooks\&.
+.fi
+.PP
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/DataHooks_BookInterface.3 b/docs/pyShelf/man/man3/DataHooks_BookInterface.3
new file mode 100644
index 0000000..64dcd83
--- /dev/null
+++ b/docs/pyShelf/man/man3/DataHooks_BookInterface.3
@@ -0,0 +1,37 @@
+.TH "DataHooks.BookInterface" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+DataHooks.BookInterface
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBconfig\fP"
+.br
+.ti -1c
+.RI "\fBdb\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Access point for book database\&.
+.fi
+.PP
+
+.SH "Member Data Documentation"
+.PP
+.SS "DataHooks\&.BookInterface\&.config"
+
+.SS "DataHooks\&.BookInterface\&.db"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/FastAPIServer.3 b/docs/pyShelf/man/man3/FastAPIServer.3
new file mode 100644
index 0000000..cc14024
--- /dev/null
+++ b/docs/pyShelf/man/man3/FastAPIServer.3
@@ -0,0 +1,149 @@
+.TH "FastAPIServer" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+FastAPIServer
+.SH SYNOPSIS
+.br
+.PP
+.SS "Classes"
+
+.in +1c
+.ti -1c
+.RI "class \fBFastAPIServer\fP"
+.br
+.in -1c
+.SS "Functions"
+
+.in +1c
+.ti -1c
+.RI "str \fBbase64decode\fP (string)"
+.br
+.ti -1c
+.RI "str \fBsummarize\fP (string)"
+.br
+.ti -1c
+.RI "str \fBconvertDateTime\fP (datetime timestamp)"
+.br
+.ti -1c
+.RI "dumps \fBbooks_tojson\fP (obj)"
+.br
+.ti -1c
+.RI "dumps \fBbook_tojson\fP (book)"
+.br
+.ti -1c
+.RI "dumps \fBcollections_tojson\fP (collection)"
+.br
+.in -1c
+.SS "Variables"
+
+.in +1c
+.ti -1c
+.RI "FastAPI \fBapp\fP = FastAPI()"
+.br
+.ti -1c
+.RI "Jinja2Templates \fBtemplates\fP = Jinja2Templates(directory='src/frontend/templates')"
+.br
+.ti -1c
+.RI "list \fBorigins\fP"
+.br
+.ti -1c
+.RI "\fBallow_origins\fP"
+.br
+.ti -1c
+.RI "\fBallow_credentials\fP"
+.br
+.ti -1c
+.RI "\fBallow_methods\fP"
+.br
+.ti -1c
+.RI "\fBallow_headers\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+pyShelf's main frontend library\&.
+.fi
+.PP
+
+.SH "Function Documentation"
+.PP
+.SS " str FastAPIServer\&.base64decode ( string)"
+
+.PP
+.nf
+Decode a base64 string\&.
+.fi
+.PP
+
+.SS " dumps FastAPIServer\&.book_tojson ( book)"
+
+.PP
+.nf
+Convert a book object to a json\&.
+.fi
+.PP
+
+.SS " dumps FastAPIServer\&.books_tojson ( obj)"
+
+.PP
+.nf
+Convert an object to a dictionary\&.
+.fi
+.PP
+
+.SS " dumps FastAPIServer\&.collections_tojson ( collection)"
+
+.PP
+.nf
+Convert a collections object to json\&.
+.fi
+.PP
+
+.SS " str FastAPIServer\&.convertDateTime (datetime timestamp)"
+
+.PP
+.nf
+Convert a datetime object to a string\&.
+.fi
+.PP
+
+.SS " str FastAPIServer\&.summarize ( string)"
+
+.PP
+.nf
+Summarize a string\&.
+.fi
+.PP
+
+.SH "Variable Documentation"
+.PP
+.SS "FastAPIServer\&.allow_credentials"
+
+.SS "FastAPIServer\&.allow_headers"
+
+.SS "FastAPIServer\&.allow_methods"
+
+.SS "FastAPIServer\&.allow_origins"
+
+.SS "FastAPI FastAPIServer\&.app = FastAPI()"
+
+.SS "list FastAPIServer\&.origins"
+\fBInitial value:\fP.PP
+.nf
+1 = [
+2 "http://localhost",
+3 "http://localhost:8081",
+4 "http://localhost:8080",
+5 "*"
+6 ]
+.fi
+
+.SS "Jinja2Templates FastAPIServer\&.templates = Jinja2Templates(directory='src/frontend/templates')"
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/FastAPIServer_FastAPIServer.3 b/docs/pyShelf/man/man3/FastAPIServer_FastAPIServer.3
new file mode 100644
index 0000000..14035fa
--- /dev/null
+++ b/docs/pyShelf/man/man3/FastAPIServer_FastAPIServer.3
@@ -0,0 +1,114 @@
+.TH "FastAPIServer.FastAPIServer" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+FastAPIServer.FastAPIServer
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, config)"
+.br
+.ti -1c
+.RI "def \fBcompile_static_files\fP (self)"
+.br
+.ti -1c
+.RI "None \fBuse_route_names_as_operation_ids\fP (self, FastAPI app)"
+.br
+.ti -1c
+.RI "def \fBindex\fP (Request request, int skip=0, int limit=10)"
+.br
+.ti -1c
+.RI "def \fBbooks\fP (Request request, int skip=0, int limit=10, collection=None)"
+.br
+.ti -1c
+.RI "def \fBbook\fP (Request request, int book_id)"
+.br
+.ti -1c
+.RI "def \fBcollections\fP (Request request)"
+.br
+.ti -1c
+.RI "def \fBrun\fP (self)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBconfig\fP"
+.br
+.ti -1c
+.RI "\fBfe_config\fP"
+.br
+.ti -1c
+.RI "\fBfe_server\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Entry point for FastAPI server\&.
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def FastAPIServer\&.FastAPIServer\&.__init__ ( self, config)"
+
+.PP
+.nf
+Initialize FastAPIServer object parameters\&.
+.fi
+.PP
+
+.SH "Member Function Documentation"
+.PP
+.SS "def FastAPIServer\&.FastAPIServer\&.book (Request request, int book_id)"
+
+.SS "def FastAPIServer\&.FastAPIServer\&.books (Request request, int skip = \fC0\fP, int limit = \fC10\fP, collection = \fCNone\fP)"
+
+.SS "def FastAPIServer\&.FastAPIServer\&.collections (Request request)"
+
+.SS "def FastAPIServer\&.FastAPIServer\&.compile_static_files ( self)"
+
+.PP
+.nf
+Compile static files for web frontend\&.
+.fi
+.PP
+
+.SS "def FastAPIServer\&.FastAPIServer\&.index (Request request, int skip = \fC0\fP, int limit = \fC10\fP)"
+
+.SS "def FastAPIServer\&.FastAPIServer\&.run ( self)"
+
+.PP
+.nf
+Front end server entrypoint\&.
+.fi
+.PP
+
+.SS " None FastAPIServer\&.FastAPIServer\&.use_route_names_as_operation_ids ( self, FastAPI app)"
+
+.PP
+.nf
+Use route name as operation id\&.
+.fi
+.PP
+
+.SH "Member Data Documentation"
+.PP
+.SS "FastAPIServer\&.FastAPIServer\&.config"
+
+.SS "FastAPIServer\&.FastAPIServer\&.fe_config"
+
+.SS "FastAPIServer\&.FastAPIServer\&.fe_server"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/objects.3 b/docs/pyShelf/man/man3/objects.3
new file mode 100644
index 0000000..7282962
--- /dev/null
+++ b/docs/pyShelf/man/man3/objects.3
@@ -0,0 +1,27 @@
+.TH "objects" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+objects
+.SH SYNOPSIS
+.br
+.PP
+.SS "Classes"
+
+.in +1c
+.ti -1c
+.RI "class \fBJSInterface\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+pyShelf's Frontend Objects\&.
+.fi
+.PP
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/objects_JSInterface.3 b/docs/pyShelf/man/man3/objects_JSInterface.3
new file mode 100644
index 0000000..5cb1805
--- /dev/null
+++ b/docs/pyShelf/man/man3/objects_JSInterface.3
@@ -0,0 +1,51 @@
+.TH "objects.JSInterface" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+objects.JSInterface
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, Config config)"
+.br
+.ti -1c
+.RI "def \fBinstall\fP (self)"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+A class to interface with the JavaScript side of pyShelf\&.
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def objects\&.JSInterface\&.__init__ ( self, Config config)"
+
+.PP
+.nf
+Initialize the JSInterface object\&.
+.fi
+.PP
+
+.SH "Member Function Documentation"
+.PP
+.SS "def objects\&.JSInterface\&.install ( self)"
+
+.PP
+.nf
+Install the JavaScript dependencies\&.
+.fi
+.PP
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_api_hooks_DuckDuckGo.3 b/docs/pyShelf/man/man3/src_backend_lib_api_hooks_DuckDuckGo.3
new file mode 100644
index 0000000..5b1ca50
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_api_hooks_DuckDuckGo.3
@@ -0,0 +1,68 @@
+.TH "src.backend.lib.api_hooks.DuckDuckGo" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.api_hooks.DuckDuckGo
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self)"
+.br
+.ti -1c
+.RI "def \fBimage_result\fP (self, query)"
+.br
+.ti -1c
+.RI "def \fBdescription_result\fP (self, query)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBurl\fP"
+.br
+.ti -1c
+.RI "\fBimageurl\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+duckduckgo related searching
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.api_hooks\&.DuckDuckGo\&.__init__ ( self)"
+
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.api_hooks\&.DuckDuckGo\&.description_result ( self, query)"
+
+.SS "def src\&.backend\&.lib\&.api_hooks\&.DuckDuckGo\&.image_result ( self, query)"
+
+.PP
+.nf
+Returns json containing url to image
+:param _key: &t=h_&iar=images&iax=images&ia=images&format=json&pretty=1
+
+.fi
+.PP
+
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.lib\&.api_hooks\&.DuckDuckGo\&.imageurl"
+
+.SS "src\&.backend\&.lib\&.api_hooks\&.DuckDuckGo\&.url"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_config.3 b/docs/pyShelf/man/man3/src_backend_lib_config.3
new file mode 100644
index 0000000..44a12a7
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_config.3
@@ -0,0 +1,27 @@
+.TH "src.backend.lib.config" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.config
+.SH SYNOPSIS
+.br
+.PP
+.SS "Classes"
+
+.in +1c
+.ti -1c
+.RI "class \fBConfig\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Pyshelf's Configuration Object\&.
+.fi
+.PP
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_config_Config.3 b/docs/pyShelf/man/man3/src_backend_lib_config_Config.3
new file mode 100644
index 0000000..ed4be25
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_config_Config.3
@@ -0,0 +1,231 @@
+.TH "src.backend.lib.config.Config" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.config.Config
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherited by \fBsrc\&.backend\&.tests\&.library_test\&.Test_Config\fP\&.
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, root)"
+.br
+.ti -1c
+.RI "def \fBinit_config\fP (self)"
+.br
+.ti -1c
+.RI "def \fBget_logger\fP (self)"
+.br
+.ti -1c
+.RI "def \fBopen_file\fP (self)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBroot\fP"
+.br
+.ti -1c
+.RI "\fBconfig_structure\fP"
+.br
+.ti -1c
+.RI "\fBlogger\fP"
+.br
+.ti -1c
+.RI "\fBbook_path\fP"
+.br
+.ti -1c
+.RI "\fBTITLE\fP"
+.br
+.ti -1c
+.RI "\fBVERSION\fP"
+.br
+.ti -1c
+.RI "\fBbook_shelf\fP"
+.br
+.ti -1c
+.RI "\fBcatalogue_db\fP"
+.br
+.ti -1c
+.RI "\fBuser\fP"
+.br
+.ti -1c
+.RI "\fBpassword\fP"
+.br
+.ti -1c
+.RI "\fBdb_host\fP"
+.br
+.ti -1c
+.RI "\fBdb_port\fP"
+.br
+.ti -1c
+.RI "\fBfile_array\fP"
+.br
+.ti -1c
+.RI "\fBauto_scan\fP"
+.br
+.ti -1c
+.RI "\fBallowed_hosts\fP"
+.br
+.ti -1c
+.RI "\fBdb_engine\fP"
+.br
+.ti -1c
+.RI "\fBdb_user\fP"
+.br
+.ti -1c
+.RI "\fBdb_pass\fP"
+.br
+.ti -1c
+.RI "\fBbuild_mode\fP"
+.br
+.in -1c
+.SS "Protected Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fB_fp\fP"
+.br
+.ti -1c
+.RI "\fB_cp\fP"
+.br
+.ti -1c
+.RI "\fB_data\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Main System Configuration\&.
+
+>>> config = Config(root)
+
+Parameters
+----------
+root : File system root of program
+
+Attributes
+----------
+root : str() stores root\&.
+config_structure : dict() Default Configuration Structure\&.
+_fp : str() file pointer to main configuration\&.
+_cp : Path() object of configuration file\&.
+_data : dict() parsed json of _fp\&.
+logger : holds logging configuration from get_logger()\&.
+book_path : directory pointer to main books folder\&.
+TITLE : str() Program title\&.
+VERSION : str() Program version\&.
+TITLE : str() Combines TITLE & VERSION\&.
+book_shelf : Deprecation TODO: Is this still in use?
+catalogue_db : str() Database Name\&.
+user : str() Database user name\&.
+password : str() Database password\&.
+db_host : str() Database host\&.
+db_port : int() Database port\&.
+file_array : list() copy of book_shelf TODO: See book_shelf
+auto_scan: bool() Do we auto scan on launch?
+allowed_hosts : list() Allowed host list\&.
+db_engine : str() Desired database engine type\&.
+db_user : str() Database user name\&. Duplication Warning\&.
+db_pass : str() Database password\&. Duplication Warning\&.
+build_mode : str() Production | Development mode\&.
+
+Methods
+-------
+get_logger : Setup loguru\&.
+open_file : Parse configuration file\&.
+
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.config\&.Config\&.__init__ ( self, root)"
+
+.PP
+.nf
+Initialize main configuration options\&.
+.fi
+.PP
+
+.PP
+Reimplemented in \fBsrc\&.backend\&.tests\&.library_test\&.Test_Config\fP\&.
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.config\&.Config\&.get_logger ( self)"
+
+.PP
+.nf
+Instantiate logging system\&.
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.config\&.Config\&.init_config ( self)"
+
+.SS "def src\&.backend\&.lib\&.config\&.Config\&.open_file ( self)"
+
+.PP
+.nf
+Open config\&.json and reads in configuration options\&.
+.fi
+.PP
+
+.PP
+Reimplemented in \fBsrc\&.backend\&.tests\&.library_test\&.Test_Config\fP\&.
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.lib\&.config\&.Config\&._cp\fC [protected]\fP"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&._data\fC [protected]\fP"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&._fp\fC [protected]\fP"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.allowed_hosts"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.auto_scan"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.book_path"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.book_shelf"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.build_mode"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.catalogue_db"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.config_structure"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.db_engine"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.db_host"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.db_pass"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.db_port"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.db_user"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.file_array"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.logger"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.password"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.root"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.TITLE"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.user"
+
+.SS "src\&.backend\&.lib\&.config\&.Config\&.VERSION"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_library_Catalogue.3 b/docs/pyShelf/man/man3/src_backend_lib_library_Catalogue.3
new file mode 100644
index 0000000..044a637
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_library_Catalogue.3
@@ -0,0 +1,263 @@
+.TH "src.backend.lib.library.Catalogue" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.library.Catalogue
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherited by \fBsrc\&.backend\&.lib\&.overide\&.ACatalogue\fP, and \fBsrc\&.backend\&.tests\&.library_test\&.Test_Catalogue\fP\&.
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, config)"
+.br
+.ti -1c
+.RI "def \fBscan_folder\fP (self, _path=None)"
+.br
+.ti -1c
+.RI "def \fBfilter_books\fP (self)"
+.br
+.ti -1c
+.RI "def \fBprocess_by_filetype\fP (self, book)"
+.br
+.ti -1c
+.RI "def \fBextract_metadata_epub\fP (self, book)"
+.br
+.ti -1c
+.RI "def \fBextract_metadata_pdf\fP (self, book)"
+.br
+.ti -1c
+.RI "def \fBextract_metadata_mobi\fP (self, book)"
+.br
+.ti -1c
+.RI "def \fBextract_content\fP (self, book_zip, book)"
+.br
+.ti -1c
+.RI "def \fBextract_cover_html\fP (self, book_zip, book)"
+.br
+.ti -1c
+.RI "def \fBextract_cover_image\fP (self, book_zip, book)"
+.br
+.ti -1c
+.RI "def \fBcompare_shelf_current\fP (self)"
+.br
+.ti -1c
+.RI "def \fBimport_books\fP (self, list=None, **kwargs)"
+.br
+.in -1c
+.SS "Static Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fBprocess_epub\fP (book)"
+.br
+.ti -1c
+.RI "def \fBstripTags\fP (source)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBfile_list\fP"
+.br
+.ti -1c
+.RI "\fBopf_regx\fP"
+.br
+.ti -1c
+.RI "\fBcover_regx\fP"
+.br
+.ti -1c
+.RI "\fBhtml_regx\fP"
+.br
+.ti -1c
+.RI "\fBtitle_sanitization_regx\fP"
+.br
+.ti -1c
+.RI "\fBtitle_sanitization_lvl2_regx\fP"
+.br
+.ti -1c
+.RI "\fBtitle_sanitization_dirs_regx\fP"
+.br
+.ti -1c
+.RI "\fBroot_dir\fP"
+.br
+.ti -1c
+.RI "\fBbook_folder\fP"
+.br
+.ti -1c
+.RI "\fBbooks\fP"
+.br
+.ti -1c
+.RI "\fBdb_pointer\fP"
+.br
+.ti -1c
+.RI "\fBconfig\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Decodes book metadata for storage
+
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.__init__ ( self, config)"
+
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.compare_shelf_current ( self)"
+
+.PP
+.nf
+Calls storage system, gets list of books stored and compares against files on disk
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.extract_content ( self, book_zip, book)"
+
+.PP
+.nf
+Opens epub as zip file filters then stores as list any files matching opf_regx
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.extract_cover_html ( self, book_zip, book)"
+
+.PP
+.nf
+Opens epub as zip file filters then stores as list any files matching html_regx
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.extract_cover_image ( self, book_zip, book)"
+
+.PP
+.nf
+Opens epub as zip file filters then stores as list any files matching cover_regx
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.extract_metadata_epub ( self, book)"
+
+.PP
+.nf
+Return extracted metadata and cover picture
+book['path'] == Full path to ebook file
+book['files'] == list of files from self\&.process_book(book)
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.extract_metadata_mobi ( self, book)"
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.extract_metadata_pdf ( self, book)"
+
+.PP
+.nf
+Return extracted metadata
+:NOTES: Retrieval of data has been problematic, some pdf's providing
+reliable titles that corespond with the actual, and others being
+nonsense\&.
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.filter_books ( self)"
+
+.PP
+.nf
+Calls scan_folder and filters out book files\&.
+
+:returns self\&._book_list_expanded: json string containing
+all book metadata
+
+.fi
+.PP
+
+.PP
+Reimplemented in \fBsrc\&.backend\&.tests\&.library_test\&.Test_Catalogue\fP\&.
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.import_books ( self, list = \fCNone\fP, ** kwargs)"
+
+.PP
+.nf
+Main entry point for import operations\&.
+Gets a list of new files via compare_shelf_current\&.
+Iterates over list and inserts new books into database\&.
+
+.fi
+.PP
+
+.PP
+Reimplemented in \fBsrc\&.backend\&.lib\&.overide\&.ACatalogue\fP\&.
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.process_by_filetype ( self, book)"
+
+.PP
+.nf
+Determine books filetype and process\&.
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.process_epub ( book)\fC [static]\fP"
+
+.PP
+.nf
+Return dictionary of epub file contents
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.scan_folder ( self, _path = \fCNone\fP)"
+
+.PP
+.nf
+Scan folder by _path, allows recurisive scanning
+
+.fi
+.PP
+
+.PP
+Reimplemented in \fBsrc\&.backend\&.lib\&.overide\&.ACatalogue\fP\&.
+.SS "def src\&.backend\&.lib\&.library\&.Catalogue\&.stripTags ( source)\fC [static]\fP"
+
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.book_folder"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.books"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.config"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.cover_regx"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.db_pointer"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.file_list"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.html_regx"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.opf_regx"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.root_dir"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.title_sanitization_dirs_regx"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.title_sanitization_lvl2_regx"
+
+.SS "src\&.backend\&.lib\&.library\&.Catalogue\&.title_sanitization_regx"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_models_Base.3 b/docs/pyShelf/man/man3/src_backend_lib_models_Base.3
new file mode 100644
index 0000000..2fd17d4
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_models_Base.3
@@ -0,0 +1,25 @@
+.TH "src.backend.lib.models.Base" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.models.Base
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherits DeclarativeBase\&.
+.PP
+Inherited by \fBsrc\&.backend\&.lib\&.models\&.Book\fP, and \fBsrc\&.backend\&.lib\&.models\&.Collection\fP\&.
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Base class for all models\&.
+.fi
+.PP
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_models_Book.3 b/docs/pyShelf/man/man3/src_backend_lib_models_Book.3
new file mode 100644
index 0000000..4c473fd
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_models_Book.3
@@ -0,0 +1,99 @@
+.TH "src.backend.lib.models.Book" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.models.Book
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherits \fBsrc\&.backend\&.lib\&.models\&.Base\fP\&.
+.SS "Static Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "Mapped \fBbook_id\fP = mapped_column(primary_key=True, nullable=False)"
+.br
+.ti -1c
+.RI "Mapped \fBtitle\fP [str]"
+.br
+.ti -1c
+.RI "Mapped \fBauthor\fP [Optional[str]]"
+.br
+.ti -1c
+.RI "Mapped \fBcategories\fP [Optional[str]]"
+.br
+.ti -1c
+.RI "Mapped \fBcover\fP [Optional[bytes]]"
+.br
+.ti -1c
+.RI "Mapped \fBpages\fP [Optional[int]]"
+.br
+.ti -1c
+.RI "Mapped \fBprogress\fP [Optional[float]]"
+.br
+.ti -1c
+.RI "Mapped \fBfile_name\fP [str]"
+.br
+.ti -1c
+.RI "Mapped \fBdescription\fP [Optional[str]]"
+.br
+.ti -1c
+.RI "Mapped \fBdate\fP [timestamp]"
+.br
+.ti -1c
+.RI "Mapped \fBrights\fP [Optional[str]]"
+.br
+.ti -1c
+.RI "Mapped \fBtags\fP [Optional[str]]"
+.br
+.ti -1c
+.RI "Mapped \fBidentifier\fP [Optional[str]]"
+.br
+.ti -1c
+.RI "Mapped \fBpublisher\fP [Optional[str]]"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Book model\&.
+.fi
+.PP
+
+.SH "Member Data Documentation"
+.PP
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.author [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.book_id = mapped_column(primary_key=True, nullable=False)\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.categories [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.cover [Optional[bytes]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.date [timestamp]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.description [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.file_name [str]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.identifier [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.pages [Optional[int]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.progress [Optional[float]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.publisher [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.rights [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.tags [Optional[str]]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Book\&.title [str]\fC [static]\fP"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_models_Collection.3 b/docs/pyShelf/man/man3/src_backend_lib_models_Collection.3
new file mode 100644
index 0000000..57bd0ea
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_models_Collection.3
@@ -0,0 +1,44 @@
+.TH "src.backend.lib.models.Collection" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.models.Collection
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherits \fBsrc\&.backend\&.lib\&.models\&.Base\fP\&.
+.SS "Static Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "Mapped \fBcollection\fP [str]"
+.br
+.ti -1c
+.RI "Mapped \fBbook_id\fP = mapped_column(ForeignKey(Book\&.book_id))"
+.br
+.ti -1c
+.RI "Mapped \fBcollection_id\fP = mapped_column(primary_key=True)"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Collection model\&.
+.fi
+.PP
+
+.SH "Member Data Documentation"
+.PP
+.SS "Mapped src\&.backend\&.lib\&.models\&.Collection\&.book_id = mapped_column(ForeignKey(Book\&.book_id))\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Collection\&.collection [str]\fC [static]\fP"
+
+.SS "Mapped src\&.backend\&.lib\&.models\&.Collection\&.collection_id = mapped_column(primary_key=True)\fC [static]\fP"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_overide_ACatalogue.3 b/docs/pyShelf/man/man3/src_backend_lib_overide_ACatalogue.3
new file mode 100644
index 0000000..013ce15
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_overide_ACatalogue.3
@@ -0,0 +1,68 @@
+.TH "src.backend.lib.overide.ACatalogue" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.overide.ACatalogue
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherits \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self)"
+.br
+.ti -1c
+.RI "def \fBscan_folder\fP (self, _path=None)"
+.br
+.ti -1c
+.RI "def \fBimport_books\fP (self, **kwargs)"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Aynchronous overide of library\&.Catalogue,
+: TODO : Complete or discard this overide
+
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.overide\&.ACatalogue\&.__init__ ( self)"
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.overide\&.ACatalogue\&.import_books ( self, ** kwargs)"
+
+.PP
+.nf
+Async overide of import_books
+
+.fi
+.PP
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+.SS "def src\&.backend\&.lib\&.overide\&.ACatalogue\&.scan_folder ( self, _path = \fCNone\fP)"
+
+.PP
+.nf
+Scan folder by _path, allows recurisive scanning
+
+.fi
+.PP
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_pyShelf_Admin.3 b/docs/pyShelf/man/man3/src_backend_lib_pyShelf_Admin.3
new file mode 100644
index 0000000..b0228ea
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_pyShelf_Admin.3
@@ -0,0 +1,48 @@
+.TH "src.backend.lib.pyShelf.Admin" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.pyShelf.Admin
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, root)"
+.br
+.ti -1c
+.RI "def \fBcreatesuperuser\fP (self)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBconfig\fP"
+.br
+.ti -1c
+.RI "\fBdb\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Admin\&.__init__ ( self, root)"
+
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Admin\&.createsuperuser ( self)"
+
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.lib\&.pyShelf\&.Admin\&.config"
+
+.SS "src\&.backend\&.lib\&.pyShelf\&.Admin\&.db"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_pyShelf_InitFiles.3 b/docs/pyShelf/man/man3/src_backend_lib_pyShelf_InitFiles.3
new file mode 100644
index 0000000..a129f8f
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_pyShelf_InitFiles.3
@@ -0,0 +1,46 @@
+.TH "src.backend.lib.pyShelf.InitFiles" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.pyShelf.InitFiles
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, file_array)"
+.br
+.ti -1c
+.RI "def \fBCreateFile\fP (self, _pointer)"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+First run file creation operations
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.pyShelf\&.InitFiles\&.__init__ ( self, file_array)"
+
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.pyShelf\&.InitFiles\&.CreateFile ( self, _pointer)"
+
+.PP
+.nf
+Checks if file exists and creates it if not
+
+.fi
+.PP
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_pyShelf_Server.3 b/docs/pyShelf/man/man3/src_backend_lib_pyShelf_Server.3
new file mode 100644
index 0000000..e583892
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_pyShelf_Server.3
@@ -0,0 +1,97 @@
+.TH "src.backend.lib.pyShelf.Server" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.pyShelf.Server
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, root)"
+.br
+.ti -1c
+.RI "def \fB__aexit__\fP (self, *args, **kwargs)"
+.br
+.ti -1c
+.RI "def \fBinitialize_server\fP (self)"
+.br
+.ti -1c
+.RI "def \fBrunImport\fP (self)"
+.br
+.ti -1c
+.RI "def \fBsocketio\fP (self, websocket, path)"
+.br
+.ti -1c
+.RI "def \fBpong\fP (self)"
+.br
+.ti -1c
+.RI "def \fBstart\fP (self)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBroot\fP"
+.br
+.ti -1c
+.RI "\fBhost\fP"
+.br
+.ti -1c
+.RI "\fBconfig\fP"
+.br
+.ti -1c
+.RI "\fBloop\fP"
+.br
+.ti -1c
+.RI "\fBserve\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Main Server Container
+:TODO: Document this
+
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.__init__ ( self, root)"
+
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.__aexit__ ( self, * args, ** kwargs)"
+
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.initialize_server ( self)"
+
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.pong ( self)"
+
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.runImport ( self)"
+
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.socketio ( self, websocket, path)"
+
+.SS "def src\&.backend\&.lib\&.pyShelf\&.Server\&.start ( self)"
+
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.lib\&.pyShelf\&.Server\&.config"
+
+.SS "src\&.backend\&.lib\&.pyShelf\&.Server\&.host"
+
+.SS "src\&.backend\&.lib\&.pyShelf\&.Server\&.loop"
+
+.SS "src\&.backend\&.lib\&.pyShelf\&.Server\&.root"
+
+.SS "src\&.backend\&.lib\&.pyShelf\&.Server\&.serve"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_storage.3 b/docs/pyShelf/man/man3/src_backend_lib_storage.3
new file mode 100644
index 0000000..32a7410
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_storage.3
@@ -0,0 +1,27 @@
+.TH "src.backend.lib.storage" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.storage
+.SH SYNOPSIS
+.br
+.PP
+.SS "Classes"
+
+.in +1c
+.ti -1c
+.RI "class \fBStorage\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Pyshelf's Main Storage Class\&.
+.fi
+.PP
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_lib_storage_Storage.3 b/docs/pyShelf/man/man3/src_backend_lib_storage_Storage.3
new file mode 100644
index 0000000..fa439a6
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_lib_storage_Storage.3
@@ -0,0 +1,238 @@
+.TH "src.backend.lib.storage.Storage" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.lib.storage.Storage
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self, config)"
+.br
+.ti -1c
+.RI "def \fBget_connection_string\fP (self)"
+.br
+.ti -1c
+.RI "def \fBcreate_tables\fP (self)"
+.br
+.ti -1c
+.RI "def \fBinsert_book\fP (self, book)"
+.br
+.ti -1c
+.RI "def \fBbook_paths_list\fP (self)"
+.br
+.ti -1c
+.RI "def \fBmake_collections\fP (self)"
+.br
+.ti -1c
+.RI "def \fBget_books\fP (self, collection=None, skip=None, limit=None)"
+.br
+.ti -1c
+.RI "def \fBget_book\fP (self, book_id)"
+.br
+.ti -1c
+.RI "def \fBget_collections\fP (self)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBconfig\fP"
+.br
+.ti -1c
+.RI "\fBsql\fP"
+.br
+.ti -1c
+.RI "\fBuser\fP"
+.br
+.ti -1c
+.RI "\fBpassword\fP"
+.br
+.ti -1c
+.RI "\fBdb_host\fP"
+.br
+.ti -1c
+.RI "\fBdb_port\fP"
+.br
+.ti -1c
+.RI "\fBengine\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+
+.PP
+.nf
+Create a new Storage object\&.
+
+>>> db = Storage(config)
+
+Parameters
+----------
+config : Config()
+ Main program configuration\&.
+
+Attributes
+----------
+config : Stores configuration
+sql : Database Name
+user : Database User Name
+password : Database Password
+db_host : Database Host
+db_port : Database Port
+engine : sqlalchemy\&.create_engine(url, executor, kw)
+
+.fi
+.PP
+
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.__init__ ( self, config)"
+
+.PP
+.nf
+Initialize storage object\&.
+.fi
+.PP
+
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.book_paths_list ( self)"
+
+.PP
+.nf
+Get file paths from database for comparison to system files\&.
+
+Returns
+-------
+_result : ScalarResult Object
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.create_tables ( self)"
+
+.PP
+.nf
+Create table structure\&.
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.get_book ( self, book_id)"
+
+.PP
+.nf
+Get book from database\&.
+
+Parameters
+----------
+book_id : int
+ Book ID to filter by\&.
+
+Returns
+-------
+_result : ScalarResult Object
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.get_books ( self, collection = \fCNone\fP, skip = \fCNone\fP, limit = \fCNone\fP)"
+
+.PP
+.nf
+Get books from database\&.
+
+Parameters
+----------
+collection : str
+ Collection to filter by\&.
+
+Returns
+-------
+_result : ScalarResult Object
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.get_collections ( self)"
+
+.PP
+.nf
+Get collections from database\&.
+
+Returns
+-------
+_result : ScalarResult Object
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.get_connection_string ( self)"
+
+.PP
+.nf
+Get connection string\&.
+
+Engine type references config\&.json:DB_ENGINE\&.
+
+Returns
+-------
+str : sqlalchemy Connection String
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.insert_book ( self, book)"
+
+.PP
+.nf
+Insert a new book into the database\&.
+
+Parameters
+----------
+book: dict()
+ Book object to insert\&.
+
+Returns
+-------
+bool
+ True on success False on failure
+
+.fi
+.PP
+
+.SS "def src\&.backend\&.lib\&.storage\&.Storage\&.make_collections ( self)"
+
+.PP
+.nf
+Parse book path's to determine common folder structure\&.
+
+Stores collections based on shared paths\&.
+
+.fi
+.PP
+
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.config"
+
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.db_host"
+
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.db_port"
+
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.engine"
+
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.password"
+
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.sql"
+
+.SS "src\&.backend\&.lib\&.storage\&.Storage\&.user"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_tests_config_test_TestConfig.3 b/docs/pyShelf/man/man3/src_backend_tests_config_test_TestConfig.3
new file mode 100644
index 0000000..7b2e608
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_tests_config_test_TestConfig.3
@@ -0,0 +1,46 @@
+.TH "src.backend.tests.config_test.TestConfig" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.tests.config_test.TestConfig
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fBtest_book_dir\fP (self)"
+.br
+.ti -1c
+.RI "def \fBtest_title\fP (self)"
+.br
+.ti -1c
+.RI "def \fBtest_version\fP (self)"
+.br
+.in -1c
+.SS "Static Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBConfig\fP \fBconfig\fP = \fBConfig\fP(os\&.path\&.abspath(os\&.path\&.curdir))"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.tests\&.config_test\&.TestConfig\&.test_book_dir ( self)"
+
+.SS "def src\&.backend\&.tests\&.config_test\&.TestConfig\&.test_title ( self)"
+
+.SS "def src\&.backend\&.tests\&.config_test\&.TestConfig\&.test_version ( self)"
+
+.SH "Member Data Documentation"
+.PP
+.SS "\fBConfig\fP src\&.backend\&.tests\&.config_test\&.TestConfig\&.config = \fBConfig\fP(os\&.path\&.abspath(os\&.path\&.curdir))\fC [static]\fP"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_tests_library_test_TestCatalogue.3 b/docs/pyShelf/man/man3/src_backend_tests_library_test_TestCatalogue.3
new file mode 100644
index 0000000..2f70127
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_tests_library_test_TestCatalogue.3
@@ -0,0 +1,41 @@
+.TH "src.backend.tests.library_test.TestCatalogue" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.tests.library_test.TestCatalogue
+.SH SYNOPSIS
+.br
+.PP
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fBtest_filter_books\fP (self)"
+.br
+.in -1c
+.SS "Static Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "os \fBroot\fP = os\&.path\&.abspath(os\&.path\&.curdir)"
+.br
+.ti -1c
+.RI "\fBTest_Config\fP \fBconfig\fP = \fBTest_Config\fP()"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.tests\&.library_test\&.TestCatalogue\&.test_filter_books ( self)"
+
+.SH "Member Data Documentation"
+.PP
+.SS "\fBTest_Config\fP src\&.backend\&.tests\&.library_test\&.TestCatalogue\&.config = \fBTest_Config\fP()\fC [static]\fP"
+
+.SS "os src\&.backend\&.tests\&.library_test\&.TestCatalogue\&.root = os\&.path\&.abspath(os\&.path\&.curdir)\fC [static]\fP"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_tests_library_test_Test_Catalogue.3 b/docs/pyShelf/man/man3/src_backend_tests_library_test_Test_Catalogue.3
new file mode 100644
index 0000000..0711761
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_tests_library_test_Test_Catalogue.3
@@ -0,0 +1,59 @@
+.TH "src.backend.tests.library_test.Test_Catalogue" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.tests.library_test.Test_Catalogue
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherits \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self)"
+.br
+.ti -1c
+.RI "def \fBfilter_books\fP (self)"
+.br
+.in -1c
+.SS "Public Attributes"
+
+.in +1c
+.ti -1c
+.RI "\fBbook_shelf\fP"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.tests\&.library_test\&.Test_Catalogue\&.__init__ ( self)"
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.tests\&.library_test\&.Test_Catalogue\&.filter_books ( self)"
+
+.PP
+.nf
+Calls scan_folder and filters out book files\&.
+
+:returns self\&._book_list_expanded: json string containing
+all book metadata
+
+.fi
+.PP
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.library\&.Catalogue\fP\&.
+.SH "Member Data Documentation"
+.PP
+.SS "src\&.backend\&.tests\&.library_test\&.Test_Catalogue\&.book_shelf"
+
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/docs/pyShelf/man/man3/src_backend_tests_library_test_Test_Config.3 b/docs/pyShelf/man/man3/src_backend_tests_library_test_Test_Config.3
new file mode 100644
index 0000000..011d5a1
--- /dev/null
+++ b/docs/pyShelf/man/man3/src_backend_tests_library_test_Test_Config.3
@@ -0,0 +1,50 @@
+.TH "src.backend.tests.library_test.Test_Config" 3 "Sat Mar 18 2023" "Version 0.8.0" "pyShelf Open Source Ebook Server" \" -*- nroff -*-
+.ad l
+.nh
+.SH NAME
+src.backend.tests.library_test.Test_Config
+.SH SYNOPSIS
+.br
+.PP
+.PP
+Inherits \fBsrc\&.backend\&.lib\&.config\&.Config\fP\&.
+.SS "Public Member Functions"
+
+.in +1c
+.ti -1c
+.RI "def \fB__init__\fP (self)"
+.br
+.ti -1c
+.RI "def \fBopen_file\fP (self, root='config\&.json')"
+.br
+.in -1c
+.SH "Detailed Description"
+.PP
+.SH "Constructor & Destructor Documentation"
+.PP
+.SS "def src\&.backend\&.tests\&.library_test\&.Test_Config\&.__init__ ( self)"
+
+.PP
+.nf
+Initialize main configuration options\&.
+.fi
+.PP
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.config\&.Config\fP\&.
+.SH "Member Function Documentation"
+.PP
+.SS "def src\&.backend\&.tests\&.library_test\&.Test_Config\&.open_file ( self, root = \fC'config\&.json'\fP)"
+
+.PP
+.nf
+Open config\&.json and reads in configuration options\&.
+.fi
+.PP
+
+.PP
+Reimplemented from \fBsrc\&.backend\&.lib\&.config\&.Config\fP\&.
+
+.SH "Author"
+.PP
+Generated automatically by Doxygen for pyShelf Open Source Ebook Server from the source code\&.
diff --git a/doxygen.conf b/doxygen.conf
index e846ee0..41d3278 100755
--- a/doxygen.conf
+++ b/doxygen.conf
@@ -38,7 +38,7 @@ PROJECT_NAME = "pyShelf Open Source Ebook Server"
# could be handy for archiving the generated documentation or if some version
# control system is used.
-PROJECT_NUMBER = 0.7.0
+PROJECT_NUMBER = 0.8.0
# Using the PROJECT_BRIEF tag one can provide an optional one line description
# for a project that appears at the top of each page and should give viewer a
@@ -58,7 +58,7 @@ PROJECT_LOGO =
# entered, it will be relative to the location where doxygen was started. If
# left blank the current directory will be used.
-OUTPUT_DIRECTORY = "docs/"
+OUTPUT_DIRECTORY = "docs/pyShelf"
# If the CREATE_SUBDIRS tag is set to YES then doxygen will create 4096 sub-
# directories (in 2 levels) under the output directory of each output format and
@@ -829,7 +829,7 @@ WARN_LOGFILE = "docs/warn.log"
# spaces. See also FILE_PATTERNS and EXTENSION_MAPPING
# Note: If this tag is empty the current directory is searched.
-INPUT = "src/backend"
+INPUT = "src/"
# This tag can be used to specify the character encoding of the source files
# that doxygen parses. Internally doxygen uses the UTF-8 encoding. Doxygen uses
@@ -854,10 +854,7 @@ INPUT_ENCODING = UTF-8
# *.m, *.markdown, *.md, *.mm, *.dox, *.py, *.pyw, *.f90, *.f95, *.f03, *.f08,
# *.f, *.for, *.tcl, *.vhd, *.vhdl, *.ucf, *.qsf and *.ice.
-FILE_PATTERNS = *.markdown \
- *.md \
- *.py \
- *.pyw \
+FILE_PATTERNS = *.py \
# The RECURSIVE tag can be used to specify whether or not subdirectories should
# be searched for input files as well.