chore: add resources

This commit is contained in:
Jesús Pérez 2023-08-14 23:33:21 +00:00
parent 3ed622da54
commit 8882e5cf81
473 changed files with 40905 additions and 0 deletions

View File

@ -0,0 +1,44 @@
[en]
id = "en"
[en.itms]
html_title = "Doc Server"
html_header = ""
html_footer = ""
req_expired = "Request expired"
expired_in = "Expired in"
seconds = "seconds"
try_again = "try again !"
principal = "Doc Server"
log_req = "Log Req"
log_data = "Log Data"
sign_in = "Sign In"
list = "List"
profiles = "Profiles"
reload = "Reload"
search = "Search"
done = "Done"
info = "Info"
log = "Log"
trace = "Trace"
debug = "Debug"
error = "Error"
warn = "Warn"
rust = "Rust"
unknown = "Unknown"
all = "All"
view_settings = "View Conf."
documentation = "Documentation"
logfrmt_sel_enter = "Sel.Enter"
logfrmt_rev_data = "Last"
log_timestamp = "Time"
log_loglevel = "Level"
log_where = "Where"
log_source = "Source"
log_line = "Line"
log_note = "Note"
view_other_options = "More-Ops."
main = "Main"
ogs_list = "Logs List"
curr_env = "Environment"
curr_env_aut = "Env. Auth"
howitwasmade = "How was made"

View File

@ -0,0 +1,35 @@
[[web_menu_items]]
# ~home text part will be removed, it is to indicate it is a home link
text = "~home"
url = "/"
roles = []
[[web_menu_items]]
typ = "icon"
srctyp = "mdbook"
text = "DocServer"
url = "/docserver"
roles = ["dev"]
[[web_menu_items.items]]
typ = "icon"
srctyp = "mdbook"
text = "Documentation"
url = "/docserver"
roles = ["dev"]
[[web_menu_items.items]]
typ = "icon"
srctyp = "source"
text = "Source Docs"
url = "/srcdocserver"
roles = ["dev"]
[[web_menu_items.items]]
typ = "icon"
srctyp = "git"
text = "Git Repo"
url = "docserver"
roles = ["dev"]
[[web_menu_items]]
text = "Users"
url = "/users"
roles = ["admin"]

View File

@ -0,0 +1,45 @@
[[serv_paths]]
src_path = "assets"
url_path = "/assets"
not_found = ""
not_auth = "/"
is_restricted = false
[[serv_paths]]
src_path = "html"
url_path = "/html"
not_found = ""
not_auth = "/"
is_restricted = false
[[serv_paths]]
src_path = "docs"
url_path = "/docs"
not_found = ""
not_auth = "/login"
is_restricted = true
[[serv_paths]]
src_path = "dist"
url_path = "/dist"
not_found = ""
not_auth = "/dist"
is_restricted = false
[[serv_paths]]
src_path = "docserver"
url_path = "/docserver"
not_found = ""
roles = "dev"
redirect_to = "/login"
not_auth = "/login"
is_restricted = true
[[serv_paths]]
src_path = "srcdocserver"
url_path = "/srcdocserver"
not_found = ""
roles = "dev,cl"
redirect_to = "/login"
not_auth = "/login"
is_restricted = true

View File

@ -0,0 +1,155 @@
## Root path for filea and relative path settings
root_path = "sitehome"
# Sever name
name = "docserver"
org = "org"
# Show verbose info from 0 to 2
verbose = 0
# Main root path for all rest
home_path = ""
prefix = ""
resources_path = "resources"
## Certs for SSL only for protocol = "https"
cert_file = "sitehome/certs/fullchain.pem"
key_file = "sitehome/certs/privkey.pem"
## Templates path
templates_path = "templates"
defaults_path = "resources/defaults"
## Main URLs
html_url = "/html"
assets_url = "/assets"
## Origin allowed for CORS requests
allow_origin = ["http://localhost:3000"]
## HOST ip or name for Web service
hostport = "localhost:3000"
## IP address to bind web service
bind = "0.0.0.0"
## PORT for Web service
port = 8800
## PROTOCOL for Web service: "http" or "https" (will do bind_rustls with cert_file and key_file)
protocol = "http"
## Availables Langs
langs = ["en","es"]
dflt_lang = "en"
## Setting path for locales
path_locales_config = "config/locales.toml"
## How to signup by "invitation" or "open"
signup_mode = "open"
## Exporation time for invitations in seconds
invite_expire = 900 # in seconds
## Use of TOTP Mode: "mandatory", "optional", "no" (default)
totp_mode = "optional"
## Number of TOTP digits: 6 (default) or 8
totp_digits = 6
## TOTP Algorithm: "sha1" (default), "sha256", "sha512""
totp_algorithm = "sha256"
## Password score value from 0 to 3
password_score = 0
## Defaul ADMIN fields
admin_fields = "roles,otp_base32,status"
## Token use: true or false
use_token = false
## Main use for notifications: true or false
use_mail = false
## SMTP host
smtp = "mailserver.example"
## SMTP auth can be text as: "user@domain|password" but for safety is much better to generate via "tools/get_mail_token.sh" after setting "sitehome/srv" token keys
smtp_auth = "v4.public...."
## Mails from address
mail_from ="docserver@mydomain.com"
## Users Storage URI
#users_store_uri = "file:///data/users"
users_store_uri = "sqlite:sitehome/data/users.db"
## Log user store access
user_store_access = "access"
## CASBIN authorizations and access control files settings
authz_model_path = "srvc/model.conf"
authz_policy_path = "srvc/policy.csv"
## Defaul auth roles
auth_roles = ["dev"]
## Store trace values:
trace_store_uri = "file:///logs/trace"
trace_level = 1
## Sessions store URI
#session_store_uri = "file:///data/sessions"
session_store_uri = "sqlite:sitehome/data/sessions.db"
## Sessions store file
session_store_file = "session"
## Expiration session time in seconds if sessions is not used
session_expire = 300
## Menu Items settings path
path_menu_items = "config/menu_items.toml"
## Serv path settings file, where path are defined: url, auth, is_restricted ...
path_serv_paths = "config/serv_paths.toml"
## Templates path and settings
[tpls]
url = "http://localhost:8800/"
main = "home.html.j2"
info = "info.html.j2"
help = "help.html.j2"
notfound = "notfound.html.j2"
notauth = "notauth.html.j2"
login = "login.html.j2"
logout = "logout.html.j2"
signup = "signup.html.j2"
user_settings = "user_settings.html.j2"
invite_create = "invite_create.html.j2"
invite_output = "invite_output.html.j2"
invite_mail_html = "invite_mail.html.j2"
invite_mail_txt = "invite_mail.txt.j2"
reset_password_mail_html = "reset_password_mail.html.j2"
reset_password_mail_txt = "reset_password_mail.txt.j2"
logs = "logs.html.j2"
users = "users.html.j2"
trylater = "trylater.j2"
session = "session.j2"
js = "js"
css = ""
## Paseto Tokens settings
[paseto]
public_path = "srvc/public.ky"
secret_path = "srvc/secret.ky"
is_bin = false
assert_val = ""
expire = true
[paseto.map_footer]
[paseto.data]
## UI settings: css and links paths
[ui]
main_name = "<h3>Doc Server</h3>"
title = "Doc Libraries"
#title = "<div>A Cryptographic Library</div><div> You can TRust</div>"
#subtitle = "<p>Fast, Secure, Agile</p> <p> Customizable to contexts and needs</p>"
subtitle = "Documentations"
css_link = "https://cdn.jsdelivr.net/npm/@unocss/reset/tailwind.min.css"
js_link = "https://cdn.jsdelivr.net/npm/@unocss/runtime"
main_js_link = "/assets/js/main.js"
utils_js_link = "/assets/js/utils.js"
other_css_link = "https://cdnjs.cloudflare.com/ajax/libs/flowbite/1.6.5/flowbite.min.css"
other_js_link="https://cdnjs.cloudflare.com/ajax/libs/flowbite/1.6.5/flowbite.min.js"

Binary file not shown.

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,4 @@
.blue-radial {
background-image: url('data:image/svg+xml,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 4247.55 1243"><defs><radialGradient id="a" cx="2162.71" cy="6893.27" r="1809.44" fx="2162.71" fy="6893.27" gradientTransform="matrix(1.14 0 0 .51 -338.95 -2864.82)" gradientUnits="userSpaceOnUse"><stop offset="0" stop-color="%23003860"/><stop offset=".18" stop-color="%2300345a"/><stop offset=".4" stop-color="%23002b4b"/><stop offset=".65" stop-color="%23001c31"/><stop offset=".91" stop-color="%2300080d"/><stop offset="1"/></radialGradient></defs><path d="M0 0h4247.55v1243H0z" data-name="Layer 1" style="fill:url(%23a)"/></svg>');
background-position: center; background-repeat: no-repeat; background-size: cover;
}

View File

@ -0,0 +1 @@
<?xml version="1.0" encoding="UTF-8"?><svg id="b" xmlns="http://www.w3.org/2000/svg" width="226.88mm" height="156.73mm" viewBox="0 0 643.13 444.26"><g id="c"><g><path d="M24.39,378.75c3.89-10.9,80.78-223.49,84.44-234.5,3.67-11.01-3.67-15.97-6.88-5.5-3.21,10.46-94.08,249.36-99.13,263.12-5.04,13.77-5.04,23.12,14.69,23.12H289.66v19.27h64.25v-20.92h275.82c9.64,0,18.36-2.2,10.09-23.12-8.26-20.92-93.62-247.16-97.76-259.27-4.13-12.11-11.93-7.15-8.72,2.21s82.61,226.79,85.82,237.26c3.21,10.46,7.34,26.97-29.83,26.97-50.02,0-137.68-41.83-175.78-41.83s-73.87,26.97-92.01,55.59c-18.14-20.37-41.08-55.6-90.19-55.6s-111.52,41.29-175.31,41.29c-23.86,0-40.74-2.62-31.67-28.07Z" style="fill:#9d9d9c;"/><path d="M595.33,339.85c-7.44-28.26-80.51-299.1-80.51-299.1,0,0-6.98,3.29-21.4,3.29-47,0-100.05-44.04-120.99-44.04s-50.48,34.18-50.48,34.18C321.94,34.18,288.66,0,272.84,0,243.52,0,205.37,43.39,148.13,43.39c-12.56,0-20.01-2.63-20.01-2.63,0,0-77.72,290.55-83.3,310.27-5.58,19.72-5.59,32.87,15.36,32.87,50.26,0,141-64.42,169.39-64.42,52.12,0,92.38,82.83,92.38,82.83,0,0,44.43-82.17,91.9-82.17s111.68,65.08,170.78,65.08c18.15,0,18.15-17.09,10.7-45.36Zm-286.19-5.92s-25.13-43.39-74.45-43.39-127.5,63.11-148.91,63.11c-12.1,0-19.08-3.94-13.96-28.26,5.12-24.32,66.55-256.37,66.55-256.37,0,0,8.85,1.32,21.4,1.32,57.24,0,78.18-37.47,111.68-37.47,22.8,0,34.9,18.4,37.69,31.55V333.94Zm249.84,19.72c-21.4,0-97.72-63.11-148.91-63.11s-74.45,43.39-74.45,43.39V64.42c2.79-13.15,14.89-31.55,37.69-31.55,33.51,0,54.45,37.47,111.68,37.47,12.55,0,21.4-1.32,21.4-1.32,0,0,61.43,232.05,66.55,256.37,5.12,24.33-1.86,28.27-13.96,28.27Z" style="fill:#9d9d9c;"/></g></g></svg>

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

@ -0,0 +1,271 @@
<svg width="1440" height="462" viewBox="0 0 1440 462" fill="none" xmlns="http://www.w3.org/2000/svg">
<g opacity="0.3" clip-path="url(#clip0_302_24807)">
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 58.7206H48V0H47V57.4972H0V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 58.7206H96V0H95V57.4972H48V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 58.7206H144V0H143V57.4972H96V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 58.7206H192V0H191V57.4972H144V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 58.7206H240V0H239V57.4972H192V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 58.7206H288V0H287V57.4972H240V58.7206Z" fill="#1C2735"/>
<path d="M288 0H336V58.7206H288V0Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 58.7206H336V0H335V57.4972H288V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 58.7206H384V0H383V57.4972H336V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 58.7206H432V0H431V57.4972H384V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 58.7206H480V0H479V57.4972H432V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 58.7206H528V0H527V57.4972H480V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 58.7206H576V0H575V57.4972H528V58.7206Z" fill="#1C2735"/>
<path d="M576 0H624V58.7206H576V0Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 58.7206H624V0H623V57.4972H576V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 58.7206H672V0H671V57.4972H624V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 58.7206H720V0H719V57.4972H672V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 58.7206H768V0H767V57.4972H720V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 58.7206H816V0H815V57.4972H768V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 58.7206H864V0H863V57.4972H816V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 58.7206H912V0H911V57.4972H864V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 58.7206H960V0H959V57.4972H912V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 58.7206H1008V0H1007V57.4972H960V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 58.7206H1056V0H1055V57.4972H1008V58.7206Z" fill="#1C2735"/>
<path d="M1056 0H1104V58.7206H1056V0Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 58.7206H1104V0H1103V57.4972H1056V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 58.7206H1152V0H1151V57.4972H1104V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 58.7206H1200V0H1199V57.4972H1152V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 58.7206H1248V0H1247V57.4972H1200V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 58.7206H1296V0H1295V57.4972H1248V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 58.7206H1344V0H1343V57.4972H1296V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 58.7206H1392V0H1391V57.4972H1344V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 58.7206H1440V0H1439V57.4972H1392V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 117.441H48V58.7206L47 58.7206V116.218H0V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 117.441H96V58.7206L95 58.7206V116.218H48V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 117.441H144V58.7206L143 58.7206V116.218H96V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 117.441H192V58.7206L191 58.7206V116.218H144V117.441Z" fill="#1C2735"/>
<path d="M192 58.7206H240V117.441H192V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 117.441H240V58.7206L239 58.7206V116.218H192V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 117.441H288V58.7206L287 58.7206V116.218H240V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 117.441H336V58.7206L335 58.7206V116.218H288V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 117.441H384V58.7206L383 58.7206V116.218H336V117.441Z" fill="#1C2735"/>
<path d="M384 58.7206H432V117.441H384V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 117.441H432V58.7206L431 58.7206V116.218H384V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 117.441H480V58.7206L479 58.7206V116.218H432V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 117.441H528V58.7206L527 58.7206V116.218H480V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 117.441H576V58.7206L575 58.7206V116.218H528V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 117.441H624V58.7206L623 58.7206V116.218H576V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 117.441H672V58.7206L671 58.7206V116.218H624V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 117.441H720V58.7206L719 58.7206V116.218H672V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 117.441H768V58.7206L767 58.7206V116.218H720V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 117.441H816V58.7206L815 58.7206V116.218H768V117.441Z" fill="#1C2735"/>
<path d="M816 58.7206H864V117.441H816V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 117.441H864V58.7206L863 58.7206V116.218H816V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 117.441H912V58.7206L911 58.7206V116.218H864V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 117.441H960V58.7206L959 58.7206V116.218H912V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 117.441H1008V58.7206L1007 58.7206V116.218H960V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 117.441H1056V58.7206L1055 58.7206V116.218H1008V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 117.441H1104V58.7206L1103 58.7206V116.218H1056V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 117.441H1152V58.7206L1151 58.7206V116.218H1104V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 117.441H1200V58.7206L1199 58.7206V116.218H1152V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 117.441H1248V58.7206L1247 58.7206V116.218H1200V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 117.441H1296V58.7206L1295 58.7206V116.218H1248V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 117.441H1344V58.7206L1343 58.7206V116.218H1296V117.441Z" fill="#1C2735"/>
<path d="M1344 58.7206H1392V117.441H1344V58.7206Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 117.441H1392V58.7206L1391 58.7206V116.218H1344V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 117.441H1440V58.7206L1439 58.7206V116.218H1392V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 168.397H48V117.441H47V167.335H0V168.397Z" fill="#1C2735"/>
<path d="M48 117.441H96V168.397H48V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 168.397H96V117.441H95V167.335H48V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 168.397H144V117.441H143V167.335H96V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 168.397H192V117.441H191V167.335H144V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 168.397H240V117.441H239V167.335H192V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 168.397H288V117.441H287V167.335H240V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 168.397H336V117.441H335V167.335H288V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 168.397H384V117.441H383V167.335H336V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 168.397H432V117.441H431V167.335H384V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 168.397H480V117.441H479V167.335H432V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 168.397H528V117.441H527V167.335H480V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 168.397H576V117.441H575V167.335H528V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 168.397H624V117.441H623V167.335H576V168.397Z" fill="#1C2735"/>
<path d="M624 117.441H672V168.397H624V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 168.397H672V117.441H671V167.335H624V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 168.397H720V117.441H719V167.335H672V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 168.397H768V117.441H767V167.335H720V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 168.397H816V117.441H815V167.335H768V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 168.397H864V117.441H863V167.335H816V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 168.397H912V117.441H911V167.335H864V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 168.397H960V117.441H959V167.335H912V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 168.397H1008V117.441H1007V167.335H960V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 168.397H1056V117.441H1055V167.335H1008V168.397Z" fill="#1C2735"/>
<path d="M1056 117.441H1104V168.397H1056V117.441Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 168.397H1104V117.441H1103V167.335H1056V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 168.397H1152V117.441H1151V167.335H1104V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 168.397H1200V117.441H1199V167.335H1152V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 168.397H1248V117.441H1247V167.335H1200V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 168.397H1296V117.441H1295V167.335H1248V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 168.397H1344V117.441H1343V167.335H1296V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 168.397H1392V117.441H1391V167.335H1344V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 168.397H1440V117.441H1439V167.335H1392V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 227.118H48V168.397H47V225.894H0V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 227.118H96V168.397H95V225.894H48V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 227.118H144V168.397H143V225.894H96V227.118Z" fill="#1C2735"/>
<path d="M144 168.397H192V227.118H144V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 227.118H192V168.397H191V225.894H144V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 227.118H240V168.397H239V225.894H192V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 227.118H288V168.397H287V225.894H240V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 227.118H336V168.397H335V225.894H288V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 227.118H384V168.397H383V225.894H336V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 227.118H432V168.397H431V225.894H384V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 227.118H480V168.397H479V225.894H432V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 227.118H528V168.397H527V225.894H480V227.118Z" fill="#1C2735"/>
<path d="M528 168.397H576V227.118H528V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 227.118H576V168.397H575V225.894H528V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 227.118H624V168.397H623V225.894H576V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 227.118H672V168.397H671V225.894H624V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 227.118H720V168.397H719V225.894H672V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 227.118H768V168.397H767V225.894H720V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 227.118H816V168.397H815V225.894H768V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 227.118H864V168.397H863V225.894H816V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 227.118H912V168.397H911V225.894H864V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 227.118H960V168.397H959V225.894H912V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 227.118H1008V168.397H1007V225.894H960V227.118Z" fill="#1C2735"/>
<path d="M1008 168.397H1056V227.118H1008V168.397Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 227.118H1056V168.397H1055V225.894H1008V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 227.118H1104V168.397H1103V225.894H1056V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 227.118H1152V168.397H1151V225.894H1104V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 227.118H1200V168.397H1199V225.894H1152V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 227.118H1248V168.397H1247V225.894H1200V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 227.118H1296V168.397H1295V225.894H1248V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 227.118H1344V168.397H1343V225.894H1296V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 227.118H1392V168.397H1391V225.894H1344V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 227.118H1440V168.397H1439V225.894H1392V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 285.838H48V227.118H47V284.615H0V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 285.838H96V227.118H95V284.615H48V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 285.838H144V227.118H143V284.615H96V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 285.838H192V227.118H191V284.615H144V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 285.838H240V227.118H239V284.615H192V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 285.838H288V227.118H287V284.615H240V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 285.838H336V227.118H335V284.615H288V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 285.838H384V227.118H383V284.615H336V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 285.838H432V227.118H431V284.615H384V285.838Z" fill="#1C2735"/>
<path d="M432 227.118H480V285.838H432V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 285.838H480V227.118H479V284.615H432V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 285.838H528V227.118H527V284.615H480V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 285.838H576V227.118H575V284.615H528V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 285.838H624V227.118H623V284.615H576V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 285.838H672V227.118H671V284.615H624V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 285.838H720V227.118H719V284.615H672V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 285.838H768V227.118H767V284.615H720V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 285.838H816V227.118H815V284.615H768V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 285.838H864V227.118H863V284.615H816V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 285.838H912V227.118H911V284.615H864V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 285.838H960V227.118H959V284.615H912V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 285.838H1008V227.118H1007V284.615H960V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 285.838H1056V227.118H1055V284.615H1008V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 285.838H1104V227.118H1103V284.615H1056V285.838Z" fill="#1C2735"/>
<path d="M1104 227.118H1152V285.838H1104V227.118Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 285.838H1152V227.118H1151V284.615H1104V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 285.838H1200V227.118H1199V284.615H1152V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 285.838H1248V227.118H1247V284.615H1200V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 285.838H1296V227.118H1295V284.615H1248V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 285.838H1344V227.118H1343V284.615H1296V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 285.838H1392V227.118H1391V284.615H1344V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 285.838H1440V227.118H1439V284.615H1392V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 344.559H48V285.838H47V343.335H0V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 344.559H96V285.838H95V343.335H48V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 344.559H144V285.838H143V343.335H96V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 344.559H192V285.838H191V343.335H144V344.559Z" fill="#1C2735"/>
<path d="M192 285.838H240V344.559H192V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 344.559H240V285.838H239V343.335H192V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 344.559H288V285.838H287V343.335H240V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 344.559H336V285.838H335V343.335H288V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 344.559H384V285.838H383V343.335H336V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 344.559H432V285.838H431V343.335H384V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 344.559H480V285.838H479V343.335H432V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 344.559H528V285.838H527V343.335H480V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 344.559H576V285.838H575V343.335H528V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 344.559H624V285.838H623V343.335H576V344.559Z" fill="#1C2735"/>
<path d="M624 285.838H672V344.559H624V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 344.559H672V285.838H671V343.335H624V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 344.559H720V285.838H719V343.335H672V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 344.559H768V285.838H767V343.335H720V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 344.559H816V285.838H815V343.335H768V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 344.559H864V285.838H863V343.335H816V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 344.559H912V285.838H911V343.335H864V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 344.559H960V285.838H959V343.335H912V344.559Z" fill="#1C2735"/>
<path d="M960 285.838H1008V344.559H960V285.838Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 344.559H1008V285.838H1007V343.335H960V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 344.559H1056V285.838H1055V343.335H1008V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 344.559H1104V285.838H1103V343.335H1056V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 344.559H1152V285.838H1151V343.335H1104V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 344.559H1200V285.838H1199V343.335H1152V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 344.559H1248V285.838H1247V343.335H1200V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 344.559H1296V285.838H1295V343.335H1248V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 344.559H1344V285.838H1343V343.335H1296V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 344.559H1392V285.838H1391V343.335H1344V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 344.559H1440V285.838H1439V343.335H1392V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 403.279H48V344.559H47V402.056H0V403.279Z" fill="#1C2735"/>
<path d="M48 344.559H96V403.279H48V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 403.279H96V344.559H95V402.056H48V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 403.279H144V344.559H143V402.056H96V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 403.279H192V344.559H191V402.056H144V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 403.279H240V344.559H239V402.056H192V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 403.279H288V344.559H287V402.056H240V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 403.279H336V344.559H335V402.056H288V403.279Z" fill="#1C2735"/>
<path d="M336 344.559H384V403.279H336V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 403.279H384V344.559H383V402.056H336V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 403.279H432V344.559H431V402.056H384V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 403.279H480V344.559H479V402.056H432V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 403.279H528V344.559H527V402.056H480V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 403.279H576V344.559H575V402.056H528V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 403.279H624V344.559H623V402.056H576V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 403.279H672V344.559H671V402.056H624V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 403.279H720V344.559H719V402.056H672V403.279Z" fill="#1C2735"/>
<path d="M720 344.559H768V403.279H720V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 403.279H768V344.559H767V402.056H720V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 403.279H816V344.559H815V402.056H768V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 403.279H864V344.559H863V402.056H816V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 403.279H912V344.559H911V402.056H864V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 403.279H960V344.559H959V402.056H912V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 403.279H1008V344.559H1007V402.056H960V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 403.279H1056V344.559H1055V402.056H1008V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 403.279H1104V344.559H1103V402.056H1056V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 403.279H1152V344.559H1151V402.056H1104V403.279Z" fill="#1C2735"/>
<path d="M1152 344.559H1200V403.279H1152V344.559Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 403.279H1200V344.559H1199V402.056H1152V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 403.279H1248V344.559H1247V402.056H1200V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 403.279H1296V344.559H1295V402.056H1248V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 403.279H1344V344.559H1343V402.056H1296V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 403.279H1392V344.559H1391V402.056H1344V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 403.279H1440V344.559H1439V402.056H1392V403.279Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M0 462H48V403.279H47V460.777H0V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M48 462H96V403.279H95V460.777H48V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M96 462H144V403.279H143V460.777H96V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M144 462H192V403.279H191V460.777H144V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M192 462H240V403.279H239V460.777H192V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M240 462H288V403.279H287V460.777H240V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M288 462H336V403.279H335V460.777H288V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M336 462H384V403.279H383V460.777H336V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M384 462H432V403.279H431V460.777H384V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M432 462H480V403.279H479V460.777H432V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M480 462H528V403.279H527V460.777H480V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M528 462H576V403.279H575V460.777H528V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M576 462H624V403.279H623V460.777H576V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M624 462H672V403.279H671V460.777H624V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M672 462H720V403.279H719V460.777H672V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M720 462H768V403.279H767V460.777H720V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M768 462H816V403.279H815V460.777H768V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M816 462H864V403.279H863V460.777H816V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M864 462H912V403.279H911V460.777H864V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M912 462H960V403.279H959V460.777H912V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M960 462H1008V403.279H1007V460.777H960V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1008 462H1056V403.279H1055V460.777H1008V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1056 462H1104V403.279H1103V460.777H1056V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1104 462H1152V403.279H1151V460.777H1104V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1152 462H1200V403.279H1199V460.777H1152V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1200 462H1248V403.279H1247V460.777H1200V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1248 462H1296V403.279H1295V460.777H1248V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1296 462H1344V403.279H1343V460.777H1296V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1344 462H1392V403.279H1391V460.777H1344V462Z" fill="#1C2735"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M1392 462H1440V403.279H1439V460.777H1392V462Z" fill="#1C2735"/>
</g>
<defs>
<clipPath id="clip0_302_24807">
<rect width="1440" height="462" fill="white"/>
</clipPath>
</defs>
</svg>

After

Width:  |  Height:  |  Size: 29 KiB

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 28 KiB

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 5.1 KiB

View File

@ -0,0 +1 @@
<?xml version="1.0" encoding="UTF-8"?><svg id="b" xmlns="http://www.w3.org/2000/svg" width="226.88mm" height="156.73mm" viewBox="0 0 643.13 444.26"><g id="c"><g><path d="M24.39,378.75c3.89-10.9,80.78-223.49,84.44-234.5,3.67-11.01-3.67-15.97-6.88-5.5-3.21,10.46-94.08,249.36-99.13,263.12-5.04,13.77-5.04,23.12,14.69,23.12H289.66v19.27h64.25v-20.92h275.82c9.64,0,18.36-2.2,10.09-23.12-8.26-20.92-93.62-247.16-97.76-259.27-4.13-12.11-11.93-7.15-8.72,2.21s82.61,226.79,85.82,237.26c3.21,10.46,7.34,26.97-29.83,26.97-50.02,0-137.68-41.83-175.78-41.83s-73.87,26.97-92.01,55.59c-18.14-20.37-41.08-55.6-90.19-55.6s-111.52,41.29-175.31,41.29c-23.86,0-40.74-2.62-31.67-28.07Z" style="fill:#9d9d9c;"/><path d="M595.33,339.85c-7.44-28.26-80.51-299.1-80.51-299.1,0,0-6.98,3.29-21.4,3.29-47,0-100.05-44.04-120.99-44.04s-50.48,34.18-50.48,34.18C321.94,34.18,288.66,0,272.84,0,243.52,0,205.37,43.39,148.13,43.39c-12.56,0-20.01-2.63-20.01-2.63,0,0-77.72,290.55-83.3,310.27-5.58,19.72-5.59,32.87,15.36,32.87,50.26,0,141-64.42,169.39-64.42,52.12,0,92.38,82.83,92.38,82.83,0,0,44.43-82.17,91.9-82.17s111.68,65.08,170.78,65.08c18.15,0,18.15-17.09,10.7-45.36Zm-286.19-5.92s-25.13-43.39-74.45-43.39-127.5,63.11-148.91,63.11c-12.1,0-19.08-3.94-13.96-28.26,5.12-24.32,66.55-256.37,66.55-256.37,0,0,8.85,1.32,21.4,1.32,57.24,0,78.18-37.47,111.68-37.47,22.8,0,34.9,18.4,37.69,31.55V333.94Zm249.84,19.72c-21.4,0-97.72-63.11-148.91-63.11s-74.45,43.39-74.45,43.39V64.42c2.79-13.15,14.89-31.55,37.69-31.55,33.51,0,54.45,37.47,111.68,37.47,12.55,0,21.4-1.32,21.4-1.32,0,0,61.43,232.05,66.55,256.37,5.12,24.33-1.86,28.27-13.96,28.27Z" style="fill:#9d9d9c;"/></g></g></svg>

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

@ -0,0 +1 @@
<?xml version="1.0" encoding="UTF-8"?><svg id="b" xmlns="http://www.w3.org/2000/svg" width="226.88mm" height="156.73mm" viewBox="0 0 643.13 444.26"><g id="c"><g><path d="M24.39,378.75c3.89-10.9,80.78-223.49,84.44-234.5,3.67-11.01-3.67-15.97-6.88-5.5-3.21,10.46-94.08,249.36-99.13,263.12-5.04,13.77-5.04,23.12,14.69,23.12H289.66v19.27h64.25v-20.92h275.82c9.64,0,18.36-2.2,10.09-23.12-8.26-20.92-93.62-247.16-97.76-259.27-4.13-12.11-11.93-7.15-8.72,2.21s82.61,226.79,85.82,237.26c3.21,10.46,7.34,26.97-29.83,26.97-50.02,0-137.68-41.83-175.78-41.83s-73.87,26.97-92.01,55.59c-18.14-20.37-41.08-55.6-90.19-55.6s-111.52,41.29-175.31,41.29c-23.86,0-40.74-2.62-31.67-28.07Z" style="fill:#9d9d9c;"/><path d="M595.33,339.85c-7.44-28.26-80.51-299.1-80.51-299.1,0,0-6.98,3.29-21.4,3.29-47,0-100.05-44.04-120.99-44.04s-50.48,34.18-50.48,34.18C321.94,34.18,288.66,0,272.84,0,243.52,0,205.37,43.39,148.13,43.39c-12.56,0-20.01-2.63-20.01-2.63,0,0-77.72,290.55-83.3,310.27-5.58,19.72-5.59,32.87,15.36,32.87,50.26,0,141-64.42,169.39-64.42,52.12,0,92.38,82.83,92.38,82.83,0,0,44.43-82.17,91.9-82.17s111.68,65.08,170.78,65.08c18.15,0,18.15-17.09,10.7-45.36Zm-286.19-5.92s-25.13-43.39-74.45-43.39-127.5,63.11-148.91,63.11c-12.1,0-19.08-3.94-13.96-28.26,5.12-24.32,66.55-256.37,66.55-256.37,0,0,8.85,1.32,21.4,1.32,57.24,0,78.18-37.47,111.68-37.47,22.8,0,34.9,18.4,37.69,31.55V333.94Zm249.84,19.72c-21.4,0-97.72-63.11-148.91-63.11s-74.45,43.39-74.45,43.39V64.42c2.79-13.15,14.89-31.55,37.69-31.55,33.51,0,54.45,37.47,111.68,37.47,12.55,0,21.4-1.32,21.4-1.32,0,0,61.43,232.05,66.55,256.37,5.12,24.33-1.86,28.27-13.96,28.27Z" style="fill:#9d9d9c;"/></g></g></svg>

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

@ -0,0 +1 @@
<?xml version="1.0" encoding="UTF-8"?><svg id="b" xmlns="http://www.w3.org/2000/svg" width="226.88mm" height="156.73mm" viewBox="0 0 643.13 444.26"><g id="c"><g><path d="M24.39,378.75c3.89-10.9,80.78-223.49,84.44-234.5,3.67-11.01-3.67-15.97-6.88-5.5-3.21,10.46-94.08,249.36-99.13,263.12-5.04,13.77-5.04,23.12,14.69,23.12H289.66v19.27h64.25v-20.92h275.82c9.64,0,18.36-2.2,10.09-23.12-8.26-20.92-93.62-247.16-97.76-259.27-4.13-12.11-11.93-7.15-8.72,2.21s82.61,226.79,85.82,237.26c3.21,10.46,7.34,26.97-29.83,26.97-50.02,0-137.68-41.83-175.78-41.83s-73.87,26.97-92.01,55.59c-18.14-20.37-41.08-55.6-90.19-55.6s-111.52,41.29-175.31,41.29c-23.86,0-40.74-2.62-31.67-28.07Z" style="fill:#9d9d9c;"/><path d="M595.33,339.85c-7.44-28.26-80.51-299.1-80.51-299.1,0,0-6.98,3.29-21.4,3.29-47,0-100.05-44.04-120.99-44.04s-50.48,34.18-50.48,34.18C321.94,34.18,288.66,0,272.84,0,243.52,0,205.37,43.39,148.13,43.39c-12.56,0-20.01-2.63-20.01-2.63,0,0-77.72,290.55-83.3,310.27-5.58,19.72-5.59,32.87,15.36,32.87,50.26,0,141-64.42,169.39-64.42,52.12,0,92.38,82.83,92.38,82.83,0,0,44.43-82.17,91.9-82.17s111.68,65.08,170.78,65.08c18.15,0,18.15-17.09,10.7-45.36Zm-286.19-5.92s-25.13-43.39-74.45-43.39-127.5,63.11-148.91,63.11c-12.1,0-19.08-3.94-13.96-28.26,5.12-24.32,66.55-256.37,66.55-256.37,0,0,8.85,1.32,21.4,1.32,57.24,0,78.18-37.47,111.68-37.47,22.8,0,34.9,18.4,37.69,31.55V333.94Zm249.84,19.72c-21.4,0-97.72-63.11-148.91-63.11s-74.45,43.39-74.45,43.39V64.42c2.79-13.15,14.89-31.55,37.69-31.55,33.51,0,54.45,37.47,111.68,37.47,12.55,0,21.4-1.32,21.4-1.32,0,0,61.43,232.05,66.55,256.37,5.12,24.33-1.86,28.27-13.96,28.27Z" style="fill:#9d9d9c;"/></g></g></svg>

After

Width:  |  Height:  |  Size: 1.6 KiB

View File

@ -0,0 +1 @@
Hi from Assets index.html

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,81 @@
const invite_sec_form= document.getElementById('invite-sec-form');
const invite_restart= document.getElementById('invite-restart');
const invite_messages_inpt = document.getElementById('invite-messages-inpt');
const invite_messages_out = document.getElementById('invite-messages-out');
const expire_value = document.getElementById('expire-value');
const expire_range = document.getElementById('expire-range');
const expire_input = document.getElementById('expire-input');
expire_value.innerHTML = expire_range.value;
const clear_messages = () => {
if (invite_messages_inpt.innerHTML!='') { invite_messages_inpt.innerHTML=''; }
if (invite_messages_out.innerHTML!='') { invite_messages_out.innerHTML=''; }
};
const set_email = (elem) => {
clear_messages();
};
const set_expire_input = (elem) => {
expire_value.innerHTML = `${expire_input.value} minutes`
expire_range.value = expire_input.value;
clear_messages();
};
const set_expire = (elem) => {
expire_value.innerHTML = `${expire_range.value} minutes`;
expire_input.value = expire_range.value;
clear_messages();
};
const show_form = (elem) => {
invite_restart.classList.toggle('hidden');
invite_sec_form.classList.toggle('hidden');
};
const invite = async () => {
const email = document.getElementById('email-input').value;
if (email == '') {
invite_messages_inpt.innerHTML='Enter email ';
return;
}
if (!email.includes('@')) {
invite_messages_inpt.innerHTML='Please include a valid email';
return;
}
const elem_roles = document.getElementById('roles-select');
const roles = [...elem_roles.selectedOptions].map(option => option.value).join(',')
if (roles == '') {
invite_messages_inpt.innerHTML='Please include a role';
return;
}
let expire = parseInt(expire_input.value) || 0;
if (expire == 0 ) {
invite_messages_inpt.innerHTML='Please set expiration valid value';
return;
}
expire = (expire * 60);
let send_email = document.getElementById('send-email') ?
document.getElementById('send-email').checked : false;
const data = {
email,
roles,
expire,
send_email,
};
if (target_url != '') {
const response = await fetch(target_url, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
// 'cookie': document.cookie,
}
});
if (response.ok && response.status === 200 ) {
clear_messages();
invite_sec_form.classList.toggle('hidden');
invite_messages_out.innerHTML=await response.text();
invite_restart.classList.toggle('hidden');
} else {
clear_messages();
invite_messages_inpt.innerHTML=await response.text();
}
}
};

View File

@ -0,0 +1,221 @@
const login_messages = document.getElementById('login-messages-inpt');
const otp_messages = document.getElementById('otp-messages-inpt');
const forgot_messages = document.getElementById('forgot-messages');
const target_url = () => {
const url_params = new URL(window.location.toLocaleString()).searchParams;
const target_url = url_params.get('o');
return target_url ? target_url : '/';
};
const sign_up = () => {
if (SIGNIN_URL) {
location.href=SIGNUP_URL;
}
};
const on_forgot_passwd = () => {
if (RESET_URL) {
if (forgot_messages) {
forgot_messages.classList.remove('hidden');
forgot_messages.innerHTML='<p>For <u>password reset</u> you need:</p><p> a <b>valid user, email</b> and a <b>verified TOTP</b> code.</p><p>Otherwise contact with Service Administrators.</p>'
}
const main_form = document.getElementById('main-form');
const otp_form = document.getElementById('otp-form');
const btn_otp_forgot_passwd = document.getElementById('otp-forgot-password');
const otp_login_link = document.getElementById('otp-login-link')
const btn_otp_signin = document.getElementById('otp-signin-button');
const name = document.getElementById('login-name-inpt').value;
if (name == '') {
login_messages.innerHTML='Enter a name';
return;
}
main_form.classList.add('hidden');
if (btn_otp_forgot_passwd) { btn_otp_forgot_passwd.classList.remove('hidden') };
if (otp_login_link) { otp_login_link.classList.remove('hidden') };
if (btn_otp_signin) { btn_otp_signin.classList.add('hidden') };
otp_form.classList.remove('hidden');
}
};
const forgot_passwd = async() => {
const otp_loading = document.getElementById('otp-loading');
const otp_btns = document.getElementById('otp-btns');
const elem_otp_auth = document.getElementById('login-totp-inpt');
const elem_name = document.getElementById('login-name-inpt');
const otp_form = document.getElementById('otp-form');
let name = '';
if (elem_name) { name = elem_name.value; }
let otp_auth = '';
if (elem_otp_auth) {
otp_auth = elem_otp_auth.value;
if (otp_auth == '') {
otp_messages.innerHTML='Please enter a valid TOTP code';
return;
}
const numbers = /^[0-9]+$/;
if(! otp_auth.match(numbers) || otp_auth.length != totp_digits ) {
otp_messages.innerHTML='Please enter a valid TOTP code';
return;
}
}
if (RESET_URL) {
const input_data = { name, value: otp_auth };
const response = await fetch(RESET_URL, {
method: 'POST',
body: JSON.stringify(input_data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
// 'cookie': document.cookie,
}
});
if (otp_loading) { utils.css('show',otp_loading); }
if (otp_btns) { utils.css('hide',otp_btns); }
if (response.ok && response.status === 200 ) {
if (otp_loading) { utils.css('hide',otp_loading); }
const data = await response.text();
if (otp_form) { utils.css('hide',otp_form); }
if (forgot_messages) { forgot_messages.innerHTML=data }
setTimeout(() => { location.href = target_url() }, settings.toast_copy_timeout);
} else {
if (otp_loading) { utils.css('hide',otp_loading); }
if (otp_btns) { utils.css('show',otp_btns); }
if (forgot_messages) { forgot_messages.innerHTML='Errors in password reset'; }
return false;
}
}
};
const set_name = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
if (forgot_messages) { forgot_messages.classList.add('hidden'); }
};
const set_password = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
if (forgot_messages) { forgot_messages.classList.add('hidden'); }
};
const log_in = async () => {
const main_form = document.getElementById('main-form');
const main_btns = document.getElementById('main-btns');
const main_loading = document.getElementById('main-loading');
const otp_form = document.getElementById('otp-form');
const name = document.getElementById('login-name-inpt').value;
if (name == '') {
login_messages.innerHTML='Enter a name';
return;
}
const password = document.getElementById('login-password-inpt').value;
if (password == '') {
login_messages.innerHTML='Please choose a password';
return;
}
const data = {
name,
password,
//timestamp: new Date().toLocaleString(),
};
if (main_btns) { utils.css('hide',main_btns); }
if (main_loading) { utils.css('show',main_loading); }
const response = await fetch(SIGNIN_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
// 'cookie': document.cookie,
}
});
if (main_loading) { utils.css('hide',main_loading); }
if (response.ok && response.status === 200 ) {
const data = await response.text();
if (data.includes('true') && otp_form) {
main_form.classList.add('hidden');
otp_form.classList.remove('hidden');
} else {
location.href=target_url();
}
} else {
login_messages.innerHTML='Signin was not OK';
if (main_btns) { utils.css('show',main_btns); }
return false;
}
};
const set_totp = (elem) => {
if (otp_messages.innerHTML!='') { otp_messages.innerHTML=''; }
};
const otp_in = async () => {
const otp_btns = document.getElementById('otp-btns');
const otp_loading = document.getElementById('otp-loading');
const name = document.getElementById('login-name-inpt').value;
if (name == '') {
otp_messages.innerHTML='Enter a name';
return;
}
const password = document.getElementById('login-password-inpt').value;
if (password == '') {
otp_messages.innerHTML='Please choose a password';
return;
}
const otp_auth = document.getElementById('login-totp-inpt').value;
if (otp_auth == '') {
otp_messages.innerHTML='Please enter a valid TOTP code';
return;
}
const numbers = /^[0-9]+$/;
if(! otp_auth.match(numbers) || otp_auth.length != totp_digits ) {
otp_messages.innerHTML='Please enter a valid TOTP code';
return;
}
const data = {
name,
password,
otp_auth,
};
if (otp_btns) { utils.css('hide',otp_btns);}
if (otp_loading) { utils.css('show',otp_loading); }
const response = await fetch(SIGNIN_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
// 'cookie': document.cookie,
}
});
if (otp_loading) { utils.css('hide',otp_loading); }
if (response.ok && response.status === 200 ) {
location.href=target_url();
} else {
if (otp_btns) { utils.css('show',otp_btns);}
otp_messages.innerHTML='Signin was not OK';
return false;
}
};
const restart = () => {
const main_form = document.getElementById('main-form');
const main_btns = document.getElementById('main-btns');
const otp_form = document.getElementById('otp-form');
const otp_btns = document.getElementById('otp-btns');
if (otp_btns) { utils.css('show',otp_btns);}
if (main_btns) { utils.css('show',main_btns);}
otp_form.classList.add('hidden');
forgot_messages.classList.add('hidden');
forgot_messages.innerHTML='';
main_form.classList.remove('hidden');
login_messages.innerHTML='';
otp_messages.innerHTML='';
};
window.addEventListener('load', () => {
const show_password = document.getElementById('show-password');
const input_password = document.getElementById('login-password-inpt')
const otp_logo_link = document.getElementById('otp-logo-link')
if (show_password) {
show_password.addEventListener('click', (ev) => {
ev.preventDefault();
[...show_password.children].forEach(it => {
it.classList.toggle('hidden');
});
if (input_password.type == 'password') {
input_password.type = 'text';
} else {
input_password.type = 'password';
}
});
}
if (otp_logo_link) { otp_logo_link.href = location.href; }
});

View File

@ -0,0 +1,421 @@
let logs_env = {};
let usrs_env = {};
let log_elem = {};
const load_elem = () => {
log_elem = Object.freeze({
table: document.getElementById('table-log'),
modal: document.getElementById('edit-user-modal'),
dropdown_action: document.getElementById('dropdown-action'),
search_inpt: document.getElementById('table-search-usrs'),
search_clean: document.getElementById('clean-search'),
checkbox_all: document.getElementById('checkbox-all'),
count_select: document.getElementById('count-select'),
alert_danger: document.getElementById('alert-danger'),
danger_message: document.getElementById('danger-message'),
alert_info: document.getElementById('alert-info'),
info_message: document.getElementById('info-message'),
checkbox_all: document.getElementById('checkbox-all'),
alert_timeout: 4000,
user_form: {
name: document.getElementById('usr-name'),
fullname: document.getElementById('usr-fullname'),
email: document.getElementById('usr-email'),
description: document.getElementById('usr-description'),
roles: document.getElementById('usr-roles'),
items: document.getElementById('usr-items'),
status: document.getElementById('usr-status'),
current_status: document.getElementById('current-usr-status'),
isadmin: document.getElementById('usr-isadmin'),
otp_enabled: document.getElementById('usr-otp-enabled'),
otp_verified: document.getElementById('usr-otp-verified'),
otp_defs: document.getElementById('usr-otp-defs'),
},
})
};
const set_user_status = (value,color) => {
if (log_elem.user_form.current_status && log_elem.user_form.status) {
log_elem.user_form.status.value=value;
log_elem.user_form.current_status.innerHTML=
`<div class="inline-flex items-center"><div class="h-2.5 w-2.5 rounded-full bg-${color}-500 mr-2"></div><div>${value}</div></div>`;
}
};
const edit_row = (pos) => {
const id = usrs_env.data && usrs_env.data[pos] ? usrs_env.data[pos] : -1
console.log(id);
if (id > -1) {
edit_usr(`${id}`);
}
};
const close_usr = () => {
if (usrs_env.modal.hide) {
usrs_env.modal.hide();
}
};
const fill_form = () => {
log_elem.user_form.name.value = usrs_env.user.name;
log_elem.user_form.fullname.value = usrs_env.user.fullname;
log_elem.user_form.email.value = usrs_env.user.email;
log_elem.user_form.description.value = usrs_env.user.description;
log_elem.user_form.roles.value = usrs_env.user.roles;
log_elem.user_form.items.value = usrs_env.user.items;
log_elem.user_form.status.value = usrs_env.user.status;
usrs_env.user.isadmin == 1 ?
log_elem.user_form.isadmin.checked=true :
log_elem.user_form.isadmin.checked=false;
usrs_env.user.otp_enabled == 1 ?
log_elem.user_form.otp_enabled.checked = true :
log_elem.user_form.otp_enabled.checked = false;
usrs_env.user.otp_verified == 1 ?
log_elem.user_form.otp_verified.checked = true :
log_elem.user_form.otp_verified.checked = false ;
log_elem.user_form.otp_defs.value = usrs_env.user.otp_defs;
let color = '';
switch (log_elem.user_form.status.value) {
case 'Active':
color='green';
break;
case 'Created':
color='blue';
break;
case 'Pending':
color='orange';
break;
case 'Lock':
color='red';
break;
default:
color='yellow';
}
set_user_status(log_elem.user_form.status.value,color);
};
const collect_form = () => {
usrs_env.user.name = log_elem.user_form.name.value;
usrs_env.user.fullname = log_elem.user_form.fullname.value;
usrs_env.user.email = log_elem.user_form.email.value;
usrs_env.user.description = log_elem.user_form.description.value;
usrs_env.user.roles = log_elem.user_form.roles.value;
usrs_env.user.items = log_elem.user_form.items.value;
usrs_env.user.status = log_elem.user_form.status.value;
log_elem.user_form.isadmin.checked ?
usrs_env.user.isadmin = true :
usrs_env.user.isadmin = false;
log_elem.user_form.otp_enabled.checked ?
usrs_env.user.otp_enabled = true :
usrs_env.user.otp_enabled = false;
log_elem.user_form.otp_verified.checked ?
usrs_env.user.otp_verified = true :
usrs_env.user.otp_verified = false;
usrs_env.user.otp_defs = log_elem.user_form.otp_defs.value;
};
const edit_usr = async (id) => {
if (usrs_env.modal.show) {
const res = await get_user('id',id);
if (res.status && res.data.id == id) {
usrs_env.user = res.data;
fill_form();
usrs_env.modal.show();
}
}
};
const save_all_usr = async (ev) => {
ev.preventDefault();
if (usrs_env.modal.show && usrs_env.user) {
collect_form();
const res = await save_user(usrs_env.user);
// if (res.status) {
usrs_env.modal.hide();
}
};
const on_search = (el) => {
const tbody = log_elem.table.getElementsByTagName('tbody')[0];
let total = 0;
let count_select = 0;
[...tbody.getElementsByTagName('tr')].forEach(tr => {
total +=1;
if (tr.dataset.row && tr.dataset.row.includes(el.value)) {
tr.classList.remove('hidden');
count_select +=1;
} else {
tr.classList.add('hidden');
}
});
if (log_elem.count_select) {
if (count_select == total) {
log_elem.count_select.innerHTML='';
} else {
log_elem.count_select.innerHTML=`${count_select} of`;
}
}
};
const on_search_keydown = (el,event) => {
switch (event.key) {
case 'Delete':
case 'Home':
clean_search();
break;
case 'Enter':
break;
default:
on_search(el);
}
}
const clean_search = () => {
log_elem.search_inpt.value='';
on_search(log_elem.search_inpt);
};
const load_data = () => {
const js_data = document.getElementById('js-data');
if (js_data) { js_data.remove(); }
logs_env.data = [];
if (logs_list) {
try {
logs_env.data = JSON.parse(logs_list)
} catch (error) {
logs_env.data = [];
}
logs_list='';
}
};
const on_check_all = (el,ev) => {
ev.preventDefault();
if (log_elem.checkbox_all) {
const state = log_elem.checkbox_all.checked;
[...log_elem.table.getElementsByClassName('col-check')].forEach(it => {
it.checked = state;
});
}
};
const table_sort = (col) => {
const tbody = log_elem.table.getElementsByTagName('tbody')[0];
const rows = tbody.getElementsByTagName('tr');
const arrow_up = document.getElementById(`${col}-arrow-up`);
const arrow_down = document.getElementById(`${col}-arrow-down`);
const col_rows = [];
[...rows].forEach(it => {
const data=it.dataset.row.split(':');
switch (col) {
case 'when':
col_rows.push({ky: data[0], tr:it})
break;
case 'origin':
col_rows.push({ky: data[1], tr:it})
break;
case 'context':
col_rows.push({ky: data[2], tr:it})
break;
case 'request':
col_rows.push({ky: data[3], tr:it})
break;
case 'roles':
col_rows.push({ky: data[4], tr:it})
break;
case 'agent':
col_rows.push({ky: data[5], tr:it})
break;
}
})
if (col_rows.length == 0) { return; }
const is_asc = !logs_env.cols[col].asc;
logs_env.cols[col].asc = is_asc;
if (is_asc) {
if (arrow_up) { arrow_up.classList.remove('hidden'); }
if (arrow_down) { arrow_down.classList.add('hidden'); }
} else {
if (arrow_up) { arrow_up.classList.add('hidden'); }
if (arrow_down) { arrow_down.classList.remove('hidden'); }
}
col_rows.sort((a,b) => {
if (is_asc) {
if (a.ky > b.ky) { return 1; }
if (a.ky < b.ky) { return -1; }
} else {
if (a.ky > b.ky) { return -1; }
if (a.ky < b.ky) { return 1; }
}
// a must be equal to b
return 0;
});
tbody.innerHTM='';
col_rows.forEach(it => tbody.append(it.tr));
};
const alert_danger = (task,message) => {
if (log_elem.danger_message && log_elem.alert_danger) {
log_elem.danger_message.innerHTML=message;
switch (task) {
case 'show':
log_elem.alert_danger.classList.remove('hidden');
setTimeout(() => {
log_elem.alert_danger.classList.add('hidden');
}, log_elem.alert_timeout);
break;
default:
log_elem.alert_danger.classList.add('hidden');
}
}
};
const alert_info = (task,message) => {
if (log_elem.info_message && log_elem.alert_info) {
log_elem.info_message.innerHTML=message;
switch (task) {
case 'show':
log_elem.alert_info.classList.remove('hidden');
setTimeout(() => {
log_elem.alert_info.classList.add('hidden');
}, log_elem.alert_timeout);
break;
default:
log_elem.alert_info.classList.add('hidden');
}
}
};
const post_fetch = async (source,url = '',source_data) => {
if (url == '') { return }
const response = await fetch(url, {
method: 'POST',
body: JSON.stringify(source_data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
}
});
alert_danger('hide','');
alert_info('hide','');
if (response.ok && response.status === 200 ) {
if (source == 'get_user') {
const name = source_data.name;
const value = source_data.value;
const data = await response.json();
if (data[name] && data[name] == value){
return {status: true, data }
} else {
alert_danger('show',`No data found for ${name}`)
return {status: false, data: {}};
}
} else {
const data = await response.text();
if (data != '') { alert_info('show',data); }
return {status: true, data }
}
} else if ( response.status === 401 ) {
alert_danger('show','Error connection, SESSION expired ? Try Reload !')
return {status: false, data: {}};
} else if ( response.status === 400 ) {
alert_danger('show','Error connection, bad request. No data found !')
return {status: false, data: {}};
} else {
alert_danger('show','Error connection, no data available')
return {status: false, data: {}};
}
};
const get_user = async (name, value) => {
if (GETUSER_URL) {
const data = {
name,
value
};
return await post_fetch('get_user', GETUSER_URL,data);
}
};
const save_user = async (data) => {
if (SAVEUSER_URL) {
return await post_fetch('save_user', SAVEUSER_URL,data);
}
};
const list_act_items = () => {
let act_items=[];
if (log_elem.checkbox_all) {
const state = log_elem.checkbox_all.checked;
[...log_elem.table.getElementsByClassName('col-check')].forEach(it => {
if (it.checked && it.dataset.col) {
if (logs_env && logs_env.data && logs_env.data[it.dataset.col]) {
act_items.push({
id: logs_env.data[it.dataset.col],
src: it,
});
}
}
});
}
return act_items;
};
const log_delete_lines = async (user_id) => {
let status = true;
list_act_items().forEach(async (it) => {
const res = await post_fetch('act_delete', DELETELOG_URL,{
name: user_id, value: it.id
});
if (res.status) { it.src.checked = false;}
status = res.status;
});
return status;
};
const on_log_delete = async (user_id) => {
if (DELETELOG_URL) {
const usrid = user_id == '0' ? '' : user_id;
if (log_elem.dropdown_action) { log_elem.dropdown_action.classList.add('hidden'); }
const status = await log_delete_lines(usrid);
if (log_elem.checkbox_all) { log_elem.checkbox_all.checked=false;}
setTimeout(() => {
if (status) { location.reload(); }
}, log_elem.alert_timeout);
}
};
const clean_log = async (user_id) => {
if (DELETELOG_URL) {
const name = user_id == '0' ? '' : user_id;
if (log_elem.dropdown_action) { log_elem.dropdown_action.classList.add('hidden'); }
const res = await post_fetch('act_delete', DELETELOG_URL,{
name, value: 'ALL'
});
if (log_elem.checkbox_all) { log_elem.checkbox_all.checked=false;}
if (res.status) { location.reload(); }
}
};
const on_log_info = (elem) => {
if (elem && elem.children) {
[...elem.children].forEach((it,idx) => {
if (idx > 0 ) { it.classList.toggle('hidden'); }
});
}
};
window.addEventListener('load', () => {
load_data();
load_elem();
if (log_elem && log_elem.table) {
logs_env.cols = {};
const thead = log_elem.table.getElementsByTagName('thead');
if (thead[0]) {
[...thead[0].getElementsByTagName('th')].forEach(col => {
const th = col.dataset.th;
if (th != 'check') {
logs_env.cols[th]={asc: false};
col.addEventListener('click', (ev) => {
ev.preventDefault();
const src = col;
const th = src.dataset.th;
console.log(`${th}`);
table_sort(th);
});
}
});
}
}
// options with default values
const options = {
placement: 'bottom-right',
backdrop: 'dynamic',
backdropClasses: 'bg-gray-900 bg-opacity-50 dark:bg-opacity-80 fixed inset-0 z-40',
closable: true,
onHide: () => {
console.log('modal is hidden');
},
onShow: () => {
console.log('modal is shown');
},
onToggle: () => {
console.log('modal has been toggled');
}
};
usrs_env.modal = new Modal(log_elem.modal, options);
});

View File

@ -0,0 +1,81 @@
if (typeof settings == 'undefined') { let settings = {}; }
if (typeof site == 'undefined') { let site = {}; }
if (typeof curr_defs == 'undefined') { let curr_defs = {}; }
const define_settings = () => {
settings = Object.freeze({
now: new Date(),
elements: {
themeToggle: {
DarkIcon: document.getElementById('theme-toggle-dark-icon'),
LightIcon: document.getElementById('theme-toggle-light-icon'),
Btn: document.getElementById('theme-toggle'),
},
loading: document.getElementById('loadding-section'),
},
l_store_key: 'defs',
toast_copy_timeout: 2000,
alert_timeout: 3000,
});
curr_defs = {};
site = {
on_color_theme: (color_theme) => {
const elem = document.getElementsByTagName('html')[0];
if (color_theme == 'dark') {
utils.css("hide", settings.elements.themeToggle.DarkIcon);
utils.css("show", settings.elements.themeToggle.LightIcon);
if (!elem.classList.contains('dark')) {
elem.classList.add('dark');
}
} else {
utils.css("show", settings.elements.themeToggle.DarkIcon);
utils.css("hide", settings.elements.themeToggle.LightIcon);
if (elem.classList.contains('dark')) {
elem.classList.remove('dark');
}
}
},
on_loading: (task) => {
if (settings.elements.loading) {
utils.css(task, settings.elements.loading);
}
}
};
}
window.addEventListener('load', () => {
define_settings();
const str_curenv = localStorage.getItem(settings.l_store_key);
if (str_curenv != '') {
try {
data= JSON.parse(str_curenv);
} catch (error) {
console.log(error);
}
if (data) { curr_defs = data};
} else { curr_defs = {}; }
let color_theme = 'light';
if (tpl_defs && tpl_defs.main && tpl_defs.main.color_theme != '') {
color_theme = tpl_defs.main.color_theme;
} else if (curr_defs && curr_defs.theme) {
color_theme = curr_defs.theme;
}
if (settings.elements.themeToggle.Btn) {
site.on_color_theme(color_theme);
settings.elements.themeToggle.Btn.addEventListener('click', (ev) => {
ev.preventDefault();
const elem = document.getElementsByTagName('html')[0];
const next_theme = elem.classList.contains('dark') ? 'light' : 'dark';
if (tpl_defs && tpl_defs.curr_env) {
tpl_defs.curr_env.color_theme = next_theme;
}
if (curr_defs) {
curr_defs.theme=next_theme;
localStorage.setItem(settings.l_store_key, JSON.stringify(curr_defs));
}
site.on_color_theme(next_theme);
if (tpl_defs.main.has_role) { utils.update_session(data); }
});
}
// this is key to hide theme changes before page display
document.body.classList.remove('hidden');
});

View File

@ -0,0 +1,303 @@
const login_messages = document.getElementById('login-messages-inpt');
const otp_section = document.getElementById('otp-section');
const input_password = document.getElementById('login-password-inpt');
const input_repeat_password = document.getElementById('login-repeat-password-inpt');
const password_secure_bar = document.getElementById('password-secure-bar');
const password_secure_val = document.getElementById('password-secure-val');
const password_suggestions = document.getElementById('password-suggestions');
const password_warning = document.getElementById('password-warning');
const password_error = document.getElementById('password-error');
const get_otp_info = () => {
const otp_url = document.getElementById('login-totp-url').value;
if (otp_url && otp_url !== '') {
const otp_message = document.getElementById('totp-message');
const url = new URL(otp_url);
if (url && url.searchParams) {
let algo='';
let digits='';
if (totp_algorithm) {
algo = totp_algorithm;
} else {
algo = url.searchParams.get('algorithm') || '';
}
if (algo !== '') {
algo = `Algorithm: ${algo} | `;
}
if (totp_digits) {
digits = totp_digits || 0;
}
if (digits > 0) {
digits = `| Digits: ${digits}`;
}
const issuer = `Issuer: ${url.searchParams.get('issuer') || ''}`;
otp_message.innerHTML = ` ${algo} ${issuer} ${digits}`;
}
}
}
const check_value = async (name,value,elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
const elem_error = document.getElementById(`${name}-error`);
let res = await check_item(name,value);
if (elem_error) { elem_error.innerHTML = res}
if (res != '') {
elem.focus();
return false;
}
return true;
};
const set_name = async (elem) => {
if (elem.value == '' ) {
login_messages.innerHTML='Please include a valid name';
return false;
}
return await check_value('name',elem.value,elem);
};
const set_email = async (elem) => {
if (elem.value == '' || !elem.value.includes('@')) {
login_messages.innerHTML='Please include a valid email';
return false;
}
return await check_value('email',elem.value,elem);
};
const set_totp = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
};
const make_role_chips = (elem) => {
const arr_items = elem.value == '' ? [] : elem.value.split(',');
const roles_chips = document.getElementById('roles-chips');
if (roles_chips) {
roles_chips.innerHTML='';
if (arr_items.length == 0) { return; }
arr_items.forEach(name => {
if (name.replaceAll(' ','') !== '') {
const chip = document.createElement("span");
chip.id = `badge-${name}`;
const arr_classes = "inline-flex items-center px-2 py-1 mr-2 text-sm font-medium text-blue-800 bg-blue-100 rounded dark:bg-blue-900 dark:text-blue-300".split(" ");
chip.classList.add(...arr_classes);
chip.innerHTML = `${name} <button type="button" class="inline-flex items-center p-0.5 ml-2 text-sm text-blue-400 bg-transparent rounded-sm hover:bg-blue-200 hover:text-blue-900 dark:hover:bg-blue-800 dark:hover:text-blue-300" data-dismiss-target="#badge-${name}" aria-label="Remove">
<svg aria-hidden="true" class="w-3.5 h-3.5" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" clip-rule="evenodd"></path></svg>
<span class="sr-only">Remove role</span></button> `;
roles_chips.append(chip);
document.getElementById(`badge-${name}`).addEventListener('click', (ev) => {
ev.preventDefault();
chip.remove();
elem.value = arr_items.filter(it => it != name).join(',');
});
}
});
}
};
const set_role = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
make_role_chips(elem);
};
const set_items = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
};
const clean_password_messages = () => {
if (password_secure_bar) { password_secure_bar.style.width = '0%'; }
if (password_secure_val) { password_secure_val.innerHTML=0; }
if (password_suggestions) { password_suggestions.innerHTML=''; }
if (password_warning) { password_warning.innerHTML=''; }
if (password_error) { password_error.innerHTML=''; }
};
const set_password = async (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
clean_password_messages();
const feedback = await check_item('password',elem.value);
if (password_secure_bar) { password_secure_bar.style.width = `${parseInt(feedback.score)*10}%`; }
if (password_secure_val) { password_secure_val.innerHTML=feedback.score; }
if (password_suggestions) { password_suggestions.innerHTML=feedback.suggestions; }
if (password_warning) { password_warning.innerHTML=feedback.warning; }
if (PASSWORD_SCORE && feedback.score < PASSWORD_SCORE) {
elem.focus();
if (password_error) {
password_error.innerHTML = `Password under score value: ${PASSWORD_SCORE}`;
}
return false;
}
if (input_password.value !== input_repeat_password.value) {
login_messages.innerHTML = 'Password and repetition not mactch';
elem.focus();
return false;
}
return true;
};
const change_use_totp = (elem) => {
if (otp_section) {
otp_section.classList.toggle('hidden');
}
};
const set_theme = (elem) => {
if (settings && settings.elements) {
const sel_theme = elem.value;
const element = document.getElementsByTagName('html')[0];
const curr_theme = element.classList.contains('dark') ? 'dark': 'light';
if (curr_theme != sel_theme) {
if (sel_theme != 'dark' && element.classList.contains('dark')) {
element.classList.remove('dark');
} else {
element.classList.add('dark');
}
}
}
};
const copy_otp_code = (elem) => {
const content = document.getElementById('login-totp-code').value || '';
const toast_target = document.getElementById('toast-copy-totp-code');
navigator.clipboard.writeText(content).then(
() => {
if (toast_target) {
utils.css('show', toast_target);
setTimeout(() => {
utils.css('hide', toast_target);
}, settings.toast_copy_timeout);
}
},
);
};
const check_item = async (name, value) => {
if (CHECK_URL) {
const data = {
name,
value
};
const response = await fetch(CHECK_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
}
});
login_messages.innerHTML='';
if (response.ok && response.status === 200 ) {
if (name == 'password') {
const text = await response.text();
const arr_text = text.split('|');
return {
score: arr_text[0] || 0,
suggestions: arr_text[1].replace('\n','<br>') || '',
warning: arr_text[2] || '',
};
} else {
return '';
}
} else {
return `Enter a valid value for ${name}`;
}
}
};
const log_in = async () => {
const elem_name = document.getElementById('login-name-inpt');
login_messages.innerHTML='';
if (elem_name.value == '') {
login_messages.innerHTML='Enter a name';
return;
}
if (! await set_name(elem_name)) { return; }
const name = elem_name.value;
const elem_password = document.getElementById('login-password-inpt');
if (elem_password.value == '') {
login_messages.innerHTML='Please include a password';
return;
}
if (!set_password(elem_password)) { return; }
const password = elem_password.value;
const otp_code = document.getElementById('login-totp-code').value || '';
const otp_url = document.getElementById('login-totp-url').value || ''
const otp_auth = document.getElementById('login-totp-inpt').value || '';
if (otp_code != '') {
if (otp_auth == '' && totp_mode == "mandatory") {
login_messages.innerHTML='Please enter a valid TOTP code';
return;
}
if (totp_mode == 'mandatory' || document.getElementById('use-totp').checked) {
const numbers = /^[0-9]+$/;
if(! otp_auth.match(numbers) || otp_auth.length != totp_digits ) {
login_messages.innerHTML='Please enter a valid TOTP code';
return;
}
}
}
const roles = document.getElementById('login-roles-inpt').value;
const fullname= document.getElementById('login-fullname-inpt').value;
const description = document.getElementById('login-description-inpt').value;
const elem_email = document.getElementById('login-email-inpt');
if (! await set_email(elem_email)) { return; }
const email = elem_email.value;
const theme = document.getElementById('theme-select').value;
const items = {'theme': theme };
if (INVITE_KEY && INVITE_KEY != '') {
items.invite_key = INVITE_KEY;
}
if (INVITE_ID && INVITE_ID != '') {
items.invite_id = INVITE_ID;
}
const data = {
id: '0',
name,
fullname,
description,
email,
password,
otp_code,
otp_url,
otp_auth,
roles,
items
//timestamp: new Date().toLocaleString(),
};
const response = await fetch(SIGNIN_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
// 'cookie': document.cookie,
}
});
if (response.ok && response.status === 200 ) {
location.href='/';
} else {
login_messages.innerHTML='Signin response was not OK';
// throw new Error('Signin response was not OK');
return false;
}
};
window.addEventListener('load', () => {
get_otp_info();
const show_password = document.getElementById('show-password');
const show_repeat_password = document.getElementById('show-repeat-password');
const otp_code_text = document.getElementById('opt-code-text')
if (show_password) {
show_password.addEventListener('click', (ev) => {
ev.preventDefault();
[...show_password.children].forEach(it => {
it.classList.toggle('hidden');
});
if (input_password.type == 'password') {
input_password.type = 'text';
} else {
input_password.type = 'password';
}
});
}
if (show_repeat_password) {
show_repeat_password.addEventListener('click', (ev) => {
ev.preventDefault();
[...show_repeat_password.children].forEach(it => {
it.classList.toggle('hidden');
});
if (input_repeat_password.type == 'password') {
input_repeat_password.type = 'text';
} else {
input_repeat_password.type = 'password';
}
});
}
if (otp_code_text) {
otp_code_text.addEventListener('click', (ev) => {
ev.preventDefault();
copy_otp_code(ev);
});
}
});

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,384 @@
const login_messages = document.getElementById('user-messages-inpt');
const otp_section = document.getElementById('otp-section');
const input_password = document.getElementById('user-password-inpt');
const input_repeat_password = document.getElementById('user-repeat-password-inpt');
const password_secure_bar = document.getElementById('password-secure-bar');
const password_secure_val = document.getElementById('password-secure-val');
const password_suggestions = document.getElementById('password-suggestions');
const password_warning = document.getElementById('password-warning');
const password_error = document.getElementById('password-error');
const get_otp_info = () => {
const otp_url = document.getElementById('user-totp-url');
if (otp_url && otp_url.value !== '') {
const otp_message = document.getElementById('totp-message');
const url = new URL(otp_url.value);
if (url && url.searchParams) {
let algo='';
let digits='';
if (totp_algorithm) {
algo = totp_algorithm;
} else {
algo = url.searchParams.get('algorithm') || '';
}
if (algo !== '') {
algo = `Algorithm: ${algo} | `;
}
if (totp_digits) {
digits = totp_digits || 0;
}
if (digits > 0) {
digits = `| Digits: ${digits}`;
}
const issuer = `Issuer: ${url.searchParams.get('issuer') || ''}`;
otp_message.innerHTML = ` ${algo} ${issuer} ${digits}`;
}
}
}
const check_value = async (name,value,elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
const elem_error = document.getElementById(`${name}-error`);
let res = await check_item(name,value);
if (elem_error) { elem_error.innerHTML = res}
if (res != '') {
elem.focus();
return false;
}
return true;
};
const set_name = async (elem) => {
if (elem.value == '' ) {
login_messages.innerHTML='Please include a valid name';
return false;
}
return await check_value('name',elem.value,elem);
};
const set_email = async (elem) => {
if (elem.value == '' || !elem.value.includes('@')) {
login_messages.innerHTML='Please include a valid email';
return false;
}
return await check_value('email',elem.value,elem);
};
const set_totp = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
};
const make_role_chips = (elem) => {
const arr_items = elem.value == '' ? [] : elem.value.split(',');
const roles_chips = document.getElementById('roles-chips');
if (roles_chips) {
roles_chips.innerHTML='';
if (arr_items.length == 0) { return; }
arr_items.forEach(name => {
if (name.replaceAll(' ','') !== '') {
const chip = document.createElement("span");
chip.id = `badge-${name}`;
const arr_classes = "inline-flex items-center px-2 py-1 mr-2 text-sm font-medium text-blue-800 bg-blue-100 rounded dark:bg-blue-900 dark:text-blue-300".split(" ");
chip.classList.add(...arr_classes);
if (elem.disabled) {
chip.innerHTML = `${name}`;
} else {
chip.innerHTML = `${name} <button type="button" class="inline-flex items-center p-0.5 ml-2 text-sm text-blue-400 bg-transparent rounded-sm hover:bg-blue-200 hover:text-blue-900 dark:hover:bg-blue-800 dark:hover:text-blue-300" data-dismiss-target="#badge-${name}" aria-label="Remove">
<svg aria-hidden="true" class="w-3.5 h-3.5" fill="currentColor" viewBox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill-rule="evenodd" d="M4.293 4.293a1 1 0 011.414 0L10 8.586l4.293-4.293a1 1 0 111.414 1.414L11.414 10l4.293 4.293a1 1 0 01-1.414 1.414L10 11.414l-4.293 4.293a1 1 0 01-1.414-1.414L8.586 10 4.293 5.707a1 1 0 010-1.414z" clip-rule="evenodd"></path></svg>
<span class="sr-only">Remove role</span></button> `;
}
roles_chips.append(chip);
if (!elem.disabled) {
document.getElementById(`badge-${name}`).addEventListener('click', (ev) => {
ev.preventDefault();
chip.remove();
elem.value = arr_items.filter(it => it != name).join(',');
});
}
}
});
}
};
const set_role = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
make_role_chips(elem);
};
const set_items = (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
};
const clean_password_messages = () => {
if (password_secure_bar) { password_secure_bar.style.width = '0%'; }
if (password_secure_val) { password_secure_val.innerHTML=0; }
if (password_suggestions) { password_suggestions.innerHTML=''; }
if (password_warning) { password_warning.innerHTML=''; }
if (password_error) { password_error.innerHTML=''; }
};
const set_password = async (elem) => {
if (login_messages.innerHTML!='') { login_messages.innerHTML=''; }
clean_password_messages();
const feedback = await check_item('password',elem.value);
if (password_secure_bar) { password_secure_bar.style.width = `${parseInt(feedback.score)*10}%`; }
if (password_secure_val) { password_secure_val.innerHTML=feedback.score; }
if (password_suggestions) { password_suggestions.innerHTML=feedback.suggestions; }
if (password_warning) { password_warning.innerHTML=feedback.warning; }
if (PASSWORD_SCORE && feedback.score < PASSWORD_SCORE) {
elem.focus();
if (password_error) {
password_error.innerHTML = `Password under score value: ${PASSWORD_SCORE}`;
}
return false;
}
if (input_password.value !== input_repeat_password.value) {
login_messages.innerHTML = 'Password and repetition not mactch';
elem.focus();
return false;
}
return true;
};
const change_use_totp = (elem) => {
if (otp_section) { otp_section.classList.toggle('hidden'); }
if (!elem.checked) {
reset_totp(true);
} else {
const otp_auth = document.getElementById('user-totp-inpt');
const otp_code = document.getElementById('user-totp-code')
if (otp_auth && otp_code && otp_code.value != 0 ) {
otp_auth.parentElement.classList.remove('hidden');
}
}
};
const reset_totp = (hide_parent) => {
const otp_auth = document.getElementById('user-totp-inpt');
const otp_code = document.getElementById('user-totp-code');
if (otp_code) { otp_code.value='' }
const otp_text = document.getElementById('otp-code-text');
if (otp_text) { otp_text.innerHTML='' }
const otp_qr = document.getElementById('otp-code-qr');
if (otp_qr) { otp_qr.innerHTML='' }
const copy_code = document.getElementById('copy-code');
const totp_message = document.getElementById('totp-message');
if (totp_message) { totp_message.innerHTML=''}
const totp_recovery_message = document.getElementById('totp-recovery-message');
if (totp_recovery_message) { totp_recovery_message.classList.remove('hidden')}
if (copy_code) { copy_code.classList.add('hidden') }
if (otp_auth) {
otp_auth.value = '';
if (hide_parent) { otp_auth.parentElement.classList.add('hidden') }
}
};
const on_reset_totp = (elem) => {
if (elem.checked) {
reset_totp(false);
}
};
const set_theme = (elem) => {
if (settings && settings.elements) {
const sel_theme = elem.value;
const element = document.getElementsByTagName('html')[0];
const curr_theme = element.classList.contains('dark') ? 'dark': 'light';
if (curr_theme != sel_theme) {
if (sel_theme != 'dark' && element.classList.contains('dark')) {
element.classList.remove('dark');
} else {
element.classList.add('dark');
}
}
}
};
const copy_otp_code = (elem) => {
const content = document.getElementById('user-totp-code').value || '';
if (content == '') { return }
const toast_target = document.getElementById('toast-copy-totp-code');
navigator.clipboard.writeText(content).then(
() => {
if (toast_target) {
utils.css('show', toast_target);
setTimeout(() => {
utils.css('hide', toast_target);
}, settings.toast_copy_timeout);
}
},
);
};
const check_item = async (name, value) => {
if (CHECK_URL) {
const data = {
name,
value
};
const response = await fetch(CHECK_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
}
});
login_messages.innerHTML='';
if (response.ok && response.status === 200 ) {
if (name == 'password') {
const text = await response.text();
const arr_text = text.split('|');
return {
score: arr_text[0] || 0,
suggestions: arr_text[1].replace('\n','<br>') || '',
warning: arr_text[2] || '',
};
} else {
return '';
}
} else {
return `Enter a valid value for ${name}`;
}
}
};
const log_in = async () => {
login_messages.innerHTML='';
const elem_name = document.getElementById('user-name-inpt');
if (elem_name) {
if (elem_name.value == '') {
login_messages.innerHTML='Enter a name';
return;
}
const data_name = elem_name.dataset.val;
if (data_name != elem_name.value) { if (! await set_name(elem_name)) { return; } }
}
const name = elem_name ? elem_name.value : '';
const elem_password = document.getElementById('user-password-inpt');
if (elem_password) {
if (elem_password.value == '') {
login_messages.innerHTML='Please include a password';
return;
}
if (!set_password(elem_password)) { return; }
}
const password = elem_password ? elem_password.value : '';
let otp_code = '';
let otp_url = '';
let otp_auth = '';
const elem_otp_code = document.getElementById('user-totp-code');
if (elem_otp_code) {
otp_code = elem_otp_code.value;
otp_url = document.getElementById('user-totp-url').value || ''
otp_auth = document.getElementById('user-totp-inpt').value || '';
if (otp_code != '') {
if (otp_auth == '' && totp_mode == "mandatory") {
login_messages.innerHTML='Please enter a valid TOTP code';
return;
}
if (totp_mode == 'mandatory' || document.getElementById('use-totp').checked) {
const numbers = /^[0-9]+$/;
if(! otp_auth.match(numbers) || otp_auth.length != totp_digits ) {
login_messages.innerHTML='Please enter a valid TOTP code';
return;
}
}
}
}
const elem_roles = document.getElementById('user-roles-inpt');
var roles = '';
if (elem_roles) { roles = elem_roles.value; }
const elem_fullname= document.getElementById('user-fullname-inpt');
var fullname = '';
if (elem_fullname) { fullname = elem_fullname.value; }
const elem_description = document.getElementById('user-description-inpt');
var description='';
if (elem_description) { description = elem_description.value; }
const elem_email = document.getElementById('user-email-inpt');
if (elem_email) {
const data_email = elem_email.dataset.val;
if (data_email != elem_email.value) { if (! await set_email(elem_email)) { return; } }
}
const email = elem_email ? elem_email.value : '';
const elem_theme = document.getElementById('theme-select');
var theme='';
var items={};
if (elem_theme){
theme= elem_theme.value;
items = {'theme': theme };
}
if (INVITE_KEY && INVITE_KEY != '') {
items.invite_key = INVITE_KEY;
}
if (INVITE_ID && INVITE_ID != '') {
items.invite_id = INVITE_ID;
}
const data = {
id: id_user ? id_user : '0',
name,
fullname,
description,
email,
password,
otp_code,
otp_url,
otp_auth,
roles,
items,
//timestamp: new Date().toLocaleString(),
};
const response = await fetch(SIGNIN_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
// 'cookie': document.cookie,
}
});
if (response.ok && response.status === 200 ) {
location.href='/';
} else {
login_messages.innerHTML='Signin response was not OK';
// throw new Error('Signin response was not OK');
return false;
}
};
window.addEventListener('load', () => {
get_otp_info();
const show_password = document.getElementById('show-password');
const show_repeat_password = document.getElementById('show-repeat-password');
const otp_code_text = document.getElementById('otp-code-text')
const section_title = document.getElementById('section-title')
const back_btn = document.getElementById('back-btn')
const user_roles = document.getElementById('user-roles-inpt');
if (show_password) {
show_password.addEventListener('click', (ev) => {
ev.preventDefault();
[...show_password.children].forEach(it => {
it.classList.toggle('hidden');
});
if (input_password.type == 'password') {
input_password.type = 'text';
} else {
input_password.type = 'password';
}
});
}
if (show_repeat_password) {
show_repeat_password.addEventListener('click', (ev) => {
ev.preventDefault();
[...show_repeat_password.children].forEach(it => {
it.classList.toggle('hidden');
});
if (input_repeat_password.type == 'password') {
input_repeat_password.type = 'text';
} else {
input_repeat_password.type = 'password';
}
});
}
if (otp_code_text) {
otp_code_text.addEventListener('click', (ev) => {
ev.preventDefault();
copy_otp_code(ev);
});
}
if (back_btn && BACK_URL) {
back_btn.addEventListener('click', (ev) => {
ev.preventDefault();
location.href=BACK_URL;
});
}
if (section_title && TITLE_URL) {
section_title.addEventListener('click', (ev) => {
ev.preventDefault();
location.href=TITLE_URL;
});
}
if (user_roles) {
make_role_chips(user_roles);
}
});

View File

@ -0,0 +1,24 @@
const make_role_chips = (elem) => {
const arr_items = elem.value == '' ? [] : elem.value.split(',');
const roles_chips = document.getElementById('roles-chips');
if (roles_chips) {
roles_chips.innerHTML='';
if (arr_items.length == 0) { return; }
arr_items.forEach(name => {
if (name.replaceAll(' ','') !== '') {
const chip = document.createElement("span");
chip.id = `badge-${name}`;
const arr_classes = "inline-flex items-center px-2 py-1 mr-2 text-sm font-medium text-blue-800 bg-blue-100 rounded dark:bg-blue-900 dark:text-blue-300".split(" ");
chip.classList.add(...arr_classes);
chip.innerHTML = `${name}`;
roles_chips.append(chip);
}
});
}
};
window.addEventListener('load', () => {
const user_roles = document.getElementById('user-roles-inpt');
if (user_roles) {
make_role_chips(user_roles);
}
});

View File

@ -0,0 +1,437 @@
let usrs_env = {};
let usrs_elem = {};
const load_elem = () => {
usrs_elem = Object.freeze({
table: document.getElementById('table-usrs'),
modal: document.getElementById('edit-user-modal'),
dropdown_action: document.getElementById('dropdown-action'),
search_inpt: document.getElementById('table-search-usrs'),
search_clean: document.getElementById('clean-search'),
checkbox_all: document.getElementById('checkbox-all'),
count_select: document.getElementById('count-select'),
alert_danger: document.getElementById('alert-danger'),
danger_message: document.getElementById('danger-message'),
alert_info: document.getElementById('alert-info'),
info_message: document.getElementById('info-message'),
checkbox_all: document.getElementById('checkbox-all'),
alert_timeout: 4000,
form: {
name: document.getElementById('usr-name'),
fullname: document.getElementById('usr-fullname'),
email: document.getElementById('usr-email'),
description: document.getElementById('usr-description'),
roles: document.getElementById('usr-roles'),
items: document.getElementById('usr-items'),
status: document.getElementById('usr-status'),
current_status: document.getElementById('current-usr-status'),
isadmin: document.getElementById('usr-isadmin'),
otp_enabled: document.getElementById('usr-otp-enabled'),
otp_verified: document.getElementById('usr-otp-verified'),
otp_defs: document.getElementById('usr-otp-defs'),
},
})
};
const set_user_status = (value,color) => {
if (usrs_elem.form.current_status && usrs_elem.form.status) {
usrs_elem.form.status.value=value;
usrs_elem.form.current_status.innerHTML=
`<div class="inline-flex items-center"><div class="h-2.5 w-2.5 rounded-full bg-${color}-500 mr-2"></div><div>${value}</div></div>`;
}
};
const edit_row = (pos) => {
const id = usrs_env.data && usrs_env.data[pos] ? usrs_env.data[pos] : -1
console.log(id);
if (id > -1) {
edit_usr(`${id}`);
}
};
const close_usr = () => {
if (usrs_env.modal.hide) {
usrs_env.modal.hide();
}
};
const fill_form = () => {
usrs_elem.form.name.value = usrs_env.user.name;
usrs_elem.form.fullname.value = usrs_env.user.fullname;
usrs_elem.form.email.value = usrs_env.user.email;
usrs_elem.form.description.value = usrs_env.user.description;
usrs_elem.form.roles.value = usrs_env.user.roles;
usrs_elem.form.items.value = usrs_env.user.items;
usrs_elem.form.status.value = usrs_env.user.status;
usrs_env.user.isadmin == 1 ?
usrs_elem.form.isadmin.checked=true :
usrs_elem.form.isadmin.checked=false;
usrs_env.user.otp_enabled == 1 ?
usrs_elem.form.otp_enabled.checked = true :
usrs_elem.form.otp_enabled.checked = false;
usrs_env.user.otp_verified == 1 ?
usrs_elem.form.otp_verified.checked = true :
usrs_elem.form.otp_verified.checked = false ;
usrs_elem.form.otp_defs.value = usrs_env.user.otp_defs;
let color = '';
switch (usrs_elem.form.status.value) {
case 'Active':
color='green';
break;
case 'Created':
color='blue';
break;
case 'Pending':
color='orange';
break;
case 'Lock':
color='red';
break;
default:
color='yellow';
}
set_user_status(usrs_elem.form.status.value,color);
};
const collect_form = () => {
usrs_env.user.name = usrs_elem.form.name.value;
usrs_env.user.fullname = usrs_elem.form.fullname.value;
usrs_env.user.email = usrs_elem.form.email.value;
usrs_env.user.description = usrs_elem.form.description.value;
usrs_env.user.roles = usrs_elem.form.roles.value;
usrs_env.user.items = usrs_elem.form.items.value;
usrs_env.user.status = usrs_elem.form.status.value;
usrs_elem.form.isadmin.checked ?
usrs_env.user.isadmin = true :
usrs_env.user.isadmin = false;
usrs_elem.form.otp_enabled.checked ?
usrs_env.user.otp_enabled = true :
usrs_env.user.otp_enabled = false;
usrs_elem.form.otp_verified.checked ?
usrs_env.user.otp_verified = true :
usrs_env.user.otp_verified = false;
usrs_env.user.otp_defs = usrs_elem.form.otp_defs.value;
};
const edit_usr = async (id) => {
if (usrs_env.modal.show) {
const res = await get_user('id',id);
if (res.status && res.data.id == id) {
usrs_env.user = res.data;
fill_form();
usrs_env.modal.show();
}
}
};
const save_all_usr = async (ev) => {
ev.preventDefault();
if (usrs_env.modal.show && usrs_env.user) {
collect_form();
const res = await save_user(usrs_env.user);
// if (res.status) {
usrs_env.modal.hide();
}
};
const on_search = (el) => {
const tbody = usrs_elem.table.getElementsByTagName('tbody')[0];
let total = 0;
let count_select = 0;
[...tbody.getElementsByTagName('tr')].forEach(tr => {
total +=1;
if (tr.dataset.row && tr.dataset.row.includes(el.value)) {
tr.classList.remove('hidden');
count_select +=1;
} else {
tr.classList.add('hidden');
}
});
if (usrs_elem.count_select) {
if (count_select == total) {
usrs_elem.count_select.innerHTML='';
} else {
usrs_elem.count_select.innerHTML=`${count_select} of`;
}
}
};
const on_search_keydown = (el,event) => {
switch (event.key) {
case 'Delete':
case 'Home':
clean_search();
break;
case 'Enter':
break;
default:
on_search(el);
}
}
const clean_search = () => {
usrs_elem.search_inpt.value='';
on_search(usrs_elem.search_inpt);
};
const load_data = () => {
const js_data = document.getElementById('js-data');
if (js_data) { js_data.remove(); }
usrs_env.data = [];
if (users_list) {
try {
usrs_env.data = JSON.parse(users_list)
} catch (error) {
usrs_env.data = [];
}
users_list='';
}
};
const on_check_all = (el,ev) => {
ev.preventDefault();
if (usrs_elem.checkbox_all) {
const state = usrs_elem.checkbox_all.checked;
[...usrs_elem.table.getElementsByClassName('col-check')].forEach(it => {
it.checked = state;
});
}
};
const table_sort = (col) => {
const tbody = usrs_elem.table.getElementsByTagName('tbody')[0];
const rows = tbody.getElementsByTagName('tr');
const arrow_up = document.getElementById(`${col}-arrow-up`);
const arrow_down = document.getElementById(`${col}-arrow-down`);
const col_rows = [];
[...rows].forEach(it => {
const data=it.dataset.row.split(':');
switch (col) {
case 'name':
col_rows.push({ky: data[0], tr:it})
break;
case 'fullname':
col_rows.push({ky: data[1], tr:it})
break;
case 'email':
col_rows.push({ky: data[2], tr:it})
break;
case 'description':
col_rows.push({ky: data[3], tr:it})
break;
case 'roles':
col_rows.push({ky: data[4], tr:it})
break;
case 'status':
col_rows.push({ky: data[5], tr:it})
break;
case 'isadmin':
col_rows.push({ky: data[6], tr:it})
break;
}
})
if (col_rows.length == 0) { return; }
const is_asc = !usrs_env.cols[col].asc;
usrs_env.cols[col].asc = is_asc;
if (is_asc) {
if (arrow_up) { arrow_up.classList.remove('hidden'); }
if (arrow_down) { arrow_down.classList.add('hidden'); }
} else {
if (arrow_up) { arrow_up.classList.add('hidden'); }
if (arrow_down) { arrow_down.classList.remove('hidden'); }
}
col_rows.sort((a,b) => {
if (is_asc) {
if (a.ky > b.ky) { return 1; }
if (a.ky < b.ky) { return -1; }
} else {
if (a.ky > b.ky) { return -1; }
if (a.ky < b.ky) { return 1; }
}
// a must be equal to b
return 0;
});
tbody.innerHTM='';
col_rows.forEach(it => tbody.append(it.tr));
};
const alert_danger = (task,message) => {
if (usrs_elem.danger_message && usrs_elem.alert_danger) {
usrs_elem.danger_message.innerHTML=message;
switch (task) {
case 'show':
usrs_elem.alert_danger.classList.remove('hidden');
setTimeout(() => {
usrs_elem.alert_danger.classList.add('hidden');
}, usrs_elem.alert_timeout);
break;
default:
usrs_elem.alert_danger.classList.add('hidden');
}
}
};
const alert_info = (task,message) => {
if (usrs_elem.info_message && usrs_elem.alert_info) {
usrs_elem.info_message.innerHTML=message;
switch (task) {
case 'show':
usrs_elem.alert_info.classList.remove('hidden');
setTimeout(() => {
usrs_elem.alert_info.classList.add('hidden');
}, usrs_elem.alert_timeout);
break;
default:
usrs_elem.alert_info.classList.add('hidden');
}
}
};
const post_fetch = async (source,url = '',source_data) => {
if (url == '') { return }
const response = await fetch(url, {
method: 'POST',
body: JSON.stringify(source_data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
}
});
alert_danger('hide','');
alert_info('hide','');
if (response.ok && response.status === 200 ) {
if (source == 'get_user') {
const name = source_data.name;
const value = source_data.value;
const data = await response.json();
if (data[name] && data[name] == value){
return {status: true, data }
} else {
alert_danger('show',`No data found for ${name}`)
return {status: false, data: {}};
}
} else {
const data = await response.text();
if (data != '') { alert_info('show',data); }
return {status: true, data }
}
} else if ( response.status === 401 ) {
alert_danger('show','Error connection, SESSION expired ? Try Reload !')
return {status: false, data: {}};
} else if ( response.status === 400 ) {
alert_danger('show','Error connection, bad request. No data found !')
return {status: false, data: {}};
} else {
alert_danger('show','Error connection, no data available')
return {status: false, data: {}};
}
};
const get_user = async (name, value) => {
if (GETUSER_URL) {
const data = {
name,
value
};
return await post_fetch('get_user', GETUSER_URL,data);
}
};
const save_user = async (data) => {
if (SAVEUSER_URL) {
return await post_fetch('save_user', SAVEUSER_URL,data);
}
};
const list_act_items = () => {
let act_items=[];
if (usrs_elem.checkbox_all) {
const state = usrs_elem.checkbox_all.checked;
[...usrs_elem.table.getElementsByClassName('col-check')].forEach(it => {
if (it.checked && it.dataset.col) {
if (usrs_env && usrs_env.data && usrs_env.data[it.dataset.col]) {
act_items.push({
id: `${usrs_env.data[it.dataset.col]}`,
src: it,
});
}
}
});
}
return act_items;
};
const act_reset_password = async () => {
if (RESETPASSWD_URL) {
if (usrs_elem.dropdown_action) { usrs_elem.dropdown_action.classList.add('hidden'); }
list_act_items().forEach(async (it) => {
const res = await post_fetch('act_reset_password', RESETPASSWD_URL,{
name: "id", value: it.id
});
if (res.status) { it.src.checked = false;}
});
if (usrs_elem.checkbox_all) { usrs_elem.checkbox_all.checked=false;}
}
};
const act_reset_totp = async () => {
if (RESETTOTP_URL) {
if (usrs_elem.dropdown_action) { usrs_elem.dropdown_action.classList.add('hidden'); }
list_act_items().forEach(async (it) => {
const res = await post_fetch('act_reset_totp', RESETTOTP_URL,{
name: "id", value: it.id
});
if (res.status) { it.src.checked = false; }
});
if (usrs_elem.checkbox_all) { usrs_elem.checkbox_all.checked=false;}
}
};
const act_set_isadmin = async () => {
if (SETADMIN_URL) {
if (usrs_elem.dropdown_action) { usrs_elem.dropdown_action.classList.add('hidden'); }
list_act_items().forEach(async (it) => {
const res = await post_fetch('act_set_isadmin', SETADMIN_URL,{
name: "id", value: it.id
});
if (res.status) { it.src.checked = false;}
});
if (usrs_elem.checkbox_all) { usrs_elem.checkbox_all.checked=false;}
}
};
const act_delete = async () => {
if (DELETEUSER_URL) {
if (usrs_elem.dropdown_action) { usrs_elem.dropdown_action.classList.add('hidden'); }
list_act_items().forEach(async (it) => {
const res = await post_fetch('act_delete', DELETEUSER_URL,{
name: "id", value: it.id
});
if (res.status) { it.src.checked = false;}
});
if (usrs_elem.checkbox_all) { usrs_elem.checkbox_all.checked=false;}
}
};
window.addEventListener('load', () => {
// get_otp_info();
load_data();
load_elem();
// if (usrs_elem && usrs_elem.search_clean) {
// usrs_elem.search_clean.addEventListener('click', (ev) => {
// ev.preventDefault();
// usrs_elem.search_inpt.value='';
// });
// }
if (usrs_elem && usrs_elem.table) {
usrs_env.cols = {};
const thead = usrs_elem.table.getElementsByTagName('thead');
if (thead[0]) {
[...thead[0].getElementsByTagName('th')].forEach(col => {
const th = col.dataset.th;
if (th != 'check') {
usrs_env.cols[th]={asc: false};
col.addEventListener('click', (ev) => {
ev.preventDefault();
const src = col;
const th = src.dataset.th;
console.log(`${th}`);
table_sort(th);
});
}
});
}
}
// options with default values
const options = {
placement: 'bottom-right',
backdrop: 'dynamic',
backdropClasses: 'bg-gray-900 bg-opacity-50 dark:bg-opacity-80 fixed inset-0 z-40',
closable: true,
onHide: () => {
console.log('modal is hidden');
},
onShow: () => {
console.log('modal is shown');
},
onToggle: () => {
console.log('modal has been toggled');
}
};
usrs_env.modal = new Modal(usrs_elem.modal, options);
});

View File

@ -0,0 +1,127 @@
let _trace_bin=[];
const utils = {
/* To avoid checking element and global css tasks handling
utils.css('toggle','main-header','',document,'h-10!')
utils.css('get','img','tag:3')
*/
css: (task, css_target, typ='object', root=document, css_name=['hidden'] ) => {
let target = typeof css_target == 'string' ? undefined : css_target;
let is_array = false;
const elem_toggle = (elem, css_class) => {
if (elem.classList) {
if (elem.classList.contains(css_class)) {
elem.classList.remove(css_class);
} else {
elem.classList.add(css_class);
}
}
}
let css_typ = typ;
if (task == 'get' && typ === 'object') { css_typ='id';}
let pos = -1;
if (css_typ.includes(':')) {
const arr_css = typ.split(':');
css_typ = arr_css[0];
pos = arr_css[1];
}
if (typeof target === 'undefined') {
switch(css_typ) {
case 'object':
case 'id':
target = document.getElementById(css_target);
break;
case 'class':
if (root.getElementsByClassName) {
is_array=true;
if (pos != -1) {
const class_targets = root.getElementsByClassName(css_target);
if (class_targets && class_targets[pos]) { target = class_targets[pos]; }
} else {
target = root.getElementsByClassName(css_target);
}
}
break;
case 'tag':
if (root.getElementsByClassName) {
is_array=true;
if (pos != -1) {
const tag_targets = root.getElementsByTagName(css_target);
if (tag_targets && tag_targets[pos]) { target = tag_targets[pos]; }
} else {
target = root.getElementsByTagName(css_target);
}
}
break;
default:
target = document.getElementById(css_target);
}
}
if (target) {
if (typeof target.classList == 'undefined') { is_array = true;}
switch (task) {
case 'get':
return target;
break;
case 'is_add':
case 'is_hide':
if (is_array) {
return target[0].classList && target[0].classList.contains(css_name) ? true : false;
} else {
return target.classList && target.classList.contains(css_name) ? true : false;
}
break;
case 'add':
case 'hide':
if (is_array) {
[...target].forEach(it => { if (it.classList) { it.classList.add(css_name);} });
} else {
if (target.classList) target.classList.add(css_name);
}
return target;
break;
case 'remove':
case 'show':
if (is_array) {
[...target].forEach(it => { if (it.classList) { it.classList.remove(css_name);} } );
} else {
if (target.classList) target.classList.remove(css_name);
}
return target;
break;
case 'is_show':
case 'is_show':
if (is_array) {
return target[0].classList && !target[0].classList.contains(css_name) ? true : false;
} else {
return target.classList && !target.classList.contains(css_name) ? true : false;
}
break;
case 'toggle':
if (is_array) {
[...target].forEach(it => elem_toggle(it, css_name));
} else {
elem_toggle(target, css_name);
}
return target;
default:
break;
}
}
return target;
},
update_session: async (data) => {
if (UPDATE_SESSION_URL) {
const response = await fetch(UPDATE_SESSION_URL, {
method: 'POST',
body: JSON.stringify(data),
headers: {
'Content-type': 'application/json; charset=UTF-8',
}
});
if (!response.ok && response.status != 200 ) {
throw new Error('Update session was not OK');
return false;
}
}
},
};

View File

@ -0,0 +1 @@
console.log("Hello, World!");

Binary file not shown.

View File

Binary file not shown.

View File

@ -0,0 +1 @@
Hi from DIST index.html

View File

@ -0,0 +1 @@
console.log("Hello, World!");

View File

@ -0,0 +1 @@
This file makes sure that Github Pages doesn't process mdBook's output.

View File

@ -0,0 +1 @@
cl

File diff suppressed because one or more lines are too long

File diff suppressed because it is too large Load Diff

After

Width:  |  Height:  |  Size: 434 KiB

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,212 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>Users Admin - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item "><a href="/docserver/index.html"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item "><a href="/docserver/features.html"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item "><a href="/docserver/howto.html"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item "><a href="/docserver/auths.html"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item expanded "><a href="/docserver/adminusers.html" class="active"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/cookie.html"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item "><a href="/docserver/configuration.html"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h1 id="users-admin"><a class="header" href="#users-admin">Users Admin</a></h1>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="policies.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="sessions.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="policies.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="sessions.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

View File

@ -0,0 +1,12 @@
li.part-title { padding-left: 0.5em; }
.navy .chapter li.part-title { background: #353f62; }
.light .chapter li.part-title { background: #9c9cf9; }
.coal .chapter li.part-title { background: #121516; }
.ayu .chapter li.part-title { background: #333334; }
.rust .chapter li.part-title { background: #6b4438; }
.cell24 { width: 2rem; height: 4rem; }
.logo { width: 300px; height: 300px; }
.box { display: flex; align-items: center; justify-content: center; }

View File

@ -0,0 +1,353 @@
@charset "UTF-8";
:root {
--md-admonition-icon--note:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M20.71 7.04c.39-.39.39-1.04 0-1.41l-2.34-2.34c-.37-.39-1.02-.39-1.41 0l-1.84 1.83 3.75 3.75M3 17.25V21h3.75L17.81 9.93l-3.75-3.75L3 17.25z'/></svg>");
--md-admonition-icon--abstract:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M17 9H7V7h10m0 6H7v-2h10m-3 6H7v-2h7M12 3a1 1 0 0 1 1 1 1 1 0 0 1-1 1 1 1 0 0 1-1-1 1 1 0 0 1 1-1m7 0h-4.18C14.4 1.84 13.3 1 12 1c-1.3 0-2.4.84-2.82 2H5a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2V5a2 2 0 0 0-2-2z'/></svg>");
--md-admonition-icon--info:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M13 9h-2V7h2m0 10h-2v-6h2m-1-9A10 10 0 0 0 2 12a10 10 0 0 0 10 10 10 10 0 0 0 10-10A10 10 0 0 0 12 2z'/></svg>");
--md-admonition-icon--tip:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M17.66 11.2c-.23-.3-.51-.56-.77-.82-.67-.6-1.43-1.03-2.07-1.66C13.33 7.26 13 4.85 13.95 3c-.95.23-1.78.75-2.49 1.32-2.59 2.08-3.61 5.75-2.39 8.9.04.1.08.2.08.33 0 .22-.15.42-.35.5-.23.1-.47.04-.66-.12a.58.58 0 0 1-.14-.17c-1.13-1.43-1.31-3.48-.55-5.12C5.78 10 4.87 12.3 5 14.47c.06.5.12 1 .29 1.5.14.6.41 1.2.71 1.73 1.08 1.73 2.95 2.97 4.96 3.22 2.14.27 4.43-.12 6.07-1.6 1.83-1.66 2.47-4.32 1.53-6.6l-.13-.26c-.21-.46-.77-1.26-.77-1.26m-3.16 6.3c-.28.24-.74.5-1.1.6-1.12.4-2.24-.16-2.9-.82 1.19-.28 1.9-1.16 2.11-2.05.17-.8-.15-1.46-.28-2.23-.12-.74-.1-1.37.17-2.06.19.38.39.76.63 1.06.77 1 1.98 1.44 2.24 2.8.04.14.06.28.06.43.03.82-.33 1.72-.93 2.27z'/></svg>");
--md-admonition-icon--success:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='m9 20.42-6.21-6.21 2.83-2.83L9 14.77l9.88-9.89 2.83 2.83L9 20.42z'/></svg>");
--md-admonition-icon--question:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='m15.07 11.25-.9.92C13.45 12.89 13 13.5 13 15h-2v-.5c0-1.11.45-2.11 1.17-2.83l1.24-1.26c.37-.36.59-.86.59-1.41a2 2 0 0 0-2-2 2 2 0 0 0-2 2H8a4 4 0 0 1 4-4 4 4 0 0 1 4 4 3.2 3.2 0 0 1-.93 2.25M13 19h-2v-2h2M12 2A10 10 0 0 0 2 12a10 10 0 0 0 10 10 10 10 0 0 0 10-10c0-5.53-4.5-10-10-10z'/></svg>");
--md-admonition-icon--warning:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M13 14h-2V9h2m0 9h-2v-2h2M1 21h22L12 2 1 21z'/></svg>");
--md-admonition-icon--failure:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M20 6.91 17.09 4 12 9.09 6.91 4 4 6.91 9.09 12 4 17.09 6.91 20 12 14.91 17.09 20 20 17.09 14.91 12 20 6.91z'/></svg>");
--md-admonition-icon--danger:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M11 15H6l7-14v8h5l-7 14v-8z'/></svg>");
--md-admonition-icon--bug:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M14 12h-4v-2h4m0 6h-4v-2h4m6-6h-2.81a5.985 5.985 0 0 0-1.82-1.96L17 4.41 15.59 3l-2.17 2.17a6.002 6.002 0 0 0-2.83 0L8.41 3 7 4.41l1.62 1.63C7.88 6.55 7.26 7.22 6.81 8H4v2h2.09c-.05.33-.09.66-.09 1v1H4v2h2v1c0 .34.04.67.09 1H4v2h2.81c1.04 1.79 2.97 3 5.19 3s4.15-1.21 5.19-3H20v-2h-2.09c.05-.33.09-.66.09-1v-1h2v-2h-2v-1c0-.34-.04-.67-.09-1H20V8z'/></svg>");
--md-admonition-icon--example:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M7 13v-2h14v2H7m0 6v-2h14v2H7M7 7V5h14v2H7M3 8V5H2V4h2v4H3m-1 9v-1h3v4H2v-1h2v-.5H3v-1h1V17H2m2.25-7a.75.75 0 0 1 .75.75c0 .2-.08.39-.21.52L3.12 13H5v1H2v-.92L4 11H2v-1h2.25z'/></svg>");
--md-admonition-icon--quote:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M14 17h3l2-4V7h-6v6h3M6 17h3l2-4V7H5v6h3l-2 4z'/></svg>");
--md-details-icon:
url("data:image/svg+xml;charset=utf-8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'><path d='M8.59 16.58 13.17 12 8.59 7.41 10 6l6 6-6 6-1.41-1.42Z'/></svg>");
}
:is(.admonition) {
display: flow-root;
margin: 1.5625em 0;
padding: 0 1.2rem;
color: var(--fg);
page-break-inside: avoid;
background-color: var(--bg);
border: 0 solid black;
border-inline-start-width: 0.4rem;
border-radius: 0.2rem;
box-shadow: 0 0.2rem 1rem rgba(0, 0, 0, 0.05), 0 0 0.1rem rgba(0, 0, 0, 0.1);
}
@media print {
:is(.admonition) {
box-shadow: none;
}
}
:is(.admonition) > * {
box-sizing: border-box;
}
:is(.admonition) :is(.admonition) {
margin-top: 1em;
margin-bottom: 1em;
}
:is(.admonition) > .tabbed-set:only-child {
margin-top: 0;
}
html :is(.admonition) > :last-child {
margin-bottom: 1.2rem;
}
a.admonition-anchor-link {
display: none;
position: absolute;
left: -1.2rem;
padding-right: 1rem;
}
a.admonition-anchor-link:link, a.admonition-anchor-link:visited {
color: var(--fg);
}
a.admonition-anchor-link:link:hover, a.admonition-anchor-link:visited:hover {
text-decoration: none;
}
a.admonition-anchor-link::before {
content: "§";
}
:is(.admonition-title, summary) {
position: relative;
min-height: 4rem;
margin-block: 0;
margin-inline: -1.6rem -1.2rem;
padding-block: 0.8rem;
padding-inline: 4.4rem 1.2rem;
font-weight: 700;
background-color: rgba(68, 138, 255, 0.1);
display: flex;
}
:is(.admonition-title, summary) p {
margin: 0;
}
html :is(.admonition-title, summary):last-child {
margin-bottom: 0;
}
:is(.admonition-title, summary)::before {
position: absolute;
top: 0.625em;
inset-inline-start: 1.6rem;
width: 2rem;
height: 2rem;
background-color: #448aff;
mask-image: url('data:image/svg+xml;charset=utf-8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"></svg>');
-webkit-mask-image: url('data:image/svg+xml;charset=utf-8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"></svg>');
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-size: contain;
content: "";
}
:is(.admonition-title, summary):hover a.admonition-anchor-link {
display: initial;
}
details.admonition > summary.admonition-title::after {
position: absolute;
top: 0.625em;
inset-inline-end: 1.6rem;
height: 2rem;
width: 2rem;
background-color: currentcolor;
mask-image: var(--md-details-icon);
-webkit-mask-image: var(--md-details-icon);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-size: contain;
content: "";
transform: rotate(0deg);
transition: transform 0.25s;
}
details[open].admonition > summary.admonition-title::after {
transform: rotate(90deg);
}
:is(.admonition):is(.note) {
border-color: #448aff;
}
:is(.note) > :is(.admonition-title, summary) {
background-color: rgba(68, 138, 255, 0.1);
}
:is(.note) > :is(.admonition-title, summary)::before {
background-color: #448aff;
mask-image: var(--md-admonition-icon--note);
-webkit-mask-image: var(--md-admonition-icon--note);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.abstract, .summary, .tldr) {
border-color: #00b0ff;
}
:is(.abstract, .summary, .tldr) > :is(.admonition-title, summary) {
background-color: rgba(0, 176, 255, 0.1);
}
:is(.abstract, .summary, .tldr) > :is(.admonition-title, summary)::before {
background-color: #00b0ff;
mask-image: var(--md-admonition-icon--abstract);
-webkit-mask-image: var(--md-admonition-icon--abstract);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.info, .todo) {
border-color: #00b8d4;
}
:is(.info, .todo) > :is(.admonition-title, summary) {
background-color: rgba(0, 184, 212, 0.1);
}
:is(.info, .todo) > :is(.admonition-title, summary)::before {
background-color: #00b8d4;
mask-image: var(--md-admonition-icon--info);
-webkit-mask-image: var(--md-admonition-icon--info);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.tip, .hint, .important) {
border-color: #00bfa5;
}
:is(.tip, .hint, .important) > :is(.admonition-title, summary) {
background-color: rgba(0, 191, 165, 0.1);
}
:is(.tip, .hint, .important) > :is(.admonition-title, summary)::before {
background-color: #00bfa5;
mask-image: var(--md-admonition-icon--tip);
-webkit-mask-image: var(--md-admonition-icon--tip);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.success, .check, .done) {
border-color: #00c853;
}
:is(.success, .check, .done) > :is(.admonition-title, summary) {
background-color: rgba(0, 200, 83, 0.1);
}
:is(.success, .check, .done) > :is(.admonition-title, summary)::before {
background-color: #00c853;
mask-image: var(--md-admonition-icon--success);
-webkit-mask-image: var(--md-admonition-icon--success);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.question, .help, .faq) {
border-color: #64dd17;
}
:is(.question, .help, .faq) > :is(.admonition-title, summary) {
background-color: rgba(100, 221, 23, 0.1);
}
:is(.question, .help, .faq) > :is(.admonition-title, summary)::before {
background-color: #64dd17;
mask-image: var(--md-admonition-icon--question);
-webkit-mask-image: var(--md-admonition-icon--question);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.warning, .caution, .attention) {
border-color: #ff9100;
}
:is(.warning, .caution, .attention) > :is(.admonition-title, summary) {
background-color: rgba(255, 145, 0, 0.1);
}
:is(.warning, .caution, .attention) > :is(.admonition-title, summary)::before {
background-color: #ff9100;
mask-image: var(--md-admonition-icon--warning);
-webkit-mask-image: var(--md-admonition-icon--warning);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.failure, .fail, .missing) {
border-color: #ff5252;
}
:is(.failure, .fail, .missing) > :is(.admonition-title, summary) {
background-color: rgba(255, 82, 82, 0.1);
}
:is(.failure, .fail, .missing) > :is(.admonition-title, summary)::before {
background-color: #ff5252;
mask-image: var(--md-admonition-icon--failure);
-webkit-mask-image: var(--md-admonition-icon--failure);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.danger, .error) {
border-color: #ff1744;
}
:is(.danger, .error) > :is(.admonition-title, summary) {
background-color: rgba(255, 23, 68, 0.1);
}
:is(.danger, .error) > :is(.admonition-title, summary)::before {
background-color: #ff1744;
mask-image: var(--md-admonition-icon--danger);
-webkit-mask-image: var(--md-admonition-icon--danger);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.bug) {
border-color: #f50057;
}
:is(.bug) > :is(.admonition-title, summary) {
background-color: rgba(245, 0, 87, 0.1);
}
:is(.bug) > :is(.admonition-title, summary)::before {
background-color: #f50057;
mask-image: var(--md-admonition-icon--bug);
-webkit-mask-image: var(--md-admonition-icon--bug);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.example) {
border-color: #7c4dff;
}
:is(.example) > :is(.admonition-title, summary) {
background-color: rgba(124, 77, 255, 0.1);
}
:is(.example) > :is(.admonition-title, summary)::before {
background-color: #7c4dff;
mask-image: var(--md-admonition-icon--example);
-webkit-mask-image: var(--md-admonition-icon--example);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
:is(.admonition):is(.quote, .cite) {
border-color: #9e9e9e;
}
:is(.quote, .cite) > :is(.admonition-title, summary) {
background-color: rgba(158, 158, 158, 0.1);
}
:is(.quote, .cite) > :is(.admonition-title, summary)::before {
background-color: #9e9e9e;
mask-image: var(--md-admonition-icon--quote);
-webkit-mask-image: var(--md-admonition-icon--quote);
mask-repeat: no-repeat;
-webkit-mask-repeat: no-repeat;
mask-size: contain;
-webkit-mask-repeat: no-repeat;
}
.navy :is(.admonition) {
background-color: var(--sidebar-bg);
}
.ayu :is(.admonition), .coal :is(.admonition) {
background-color: var(--theme-hover);
}
.rust :is(.admonition) {
background-color: var(--sidebar-bg);
color: var(--sidebar-fg);
}
.rust .admonition-anchor-link:link, .rust .admonition-anchor-link:visited {
color: var(--sidebar-fg);
}

View File

@ -0,0 +1,212 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>Users auths - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item "><a href="/docserver/index.html"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item "><a href="/docserver/features.html"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item "><a href="/docserver/howto.html"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item expanded "><a href="/docserver/auths.html" class="active"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item "><a href="/docserver/adminusers.html"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/cookie.html"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item "><a href="/docserver/configuration.html"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h1 id="users-authentication-and-authorizations"><a class="header" href="#users-authentication-and-authorizations">Users authentication and authorizations</a></h1>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="roles.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="policies.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="roles.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="policies.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

View File

@ -0,0 +1,78 @@
/*
Based off of the Ayu theme
Original by Dempfi (https://github.com/dempfi/ayu)
*/
.hljs {
display: block;
overflow-x: auto;
background: #191f26;
color: #e6e1cf;
}
.hljs-comment,
.hljs-quote {
color: #5c6773;
font-style: italic;
}
.hljs-variable,
.hljs-template-variable,
.hljs-attribute,
.hljs-attr,
.hljs-regexp,
.hljs-link,
.hljs-selector-id,
.hljs-selector-class {
color: #ff7733;
}
.hljs-number,
.hljs-meta,
.hljs-builtin-name,
.hljs-literal,
.hljs-type,
.hljs-params {
color: #ffee99;
}
.hljs-string,
.hljs-bullet {
color: #b8cc52;
}
.hljs-title,
.hljs-built_in,
.hljs-section {
color: #ffb454;
}
.hljs-keyword,
.hljs-selector-tag,
.hljs-symbol {
color: #ff7733;
}
.hljs-name {
color: #36a3d9;
}
.hljs-tag {
color: #00568d;
}
.hljs-emphasis {
font-style: italic;
}
.hljs-strong {
font-weight: bold;
}
.hljs-addition {
color: #91b362;
}
.hljs-deletion {
color: #d96c75;
}

View File

@ -0,0 +1,688 @@
"use strict";
// Fix back button cache problem
window.onunload = function () { };
// Global variable, shared between modules
function playground_text(playground, hidden = true) {
let code_block = playground.querySelector("code");
if (window.ace && code_block.classList.contains("editable")) {
let editor = window.ace.edit(code_block);
return editor.getValue();
} else if (hidden) {
return code_block.textContent;
} else {
return code_block.innerText;
}
}
(function codeSnippets() {
function fetch_with_timeout(url, options, timeout = 6000) {
return Promise.race([
fetch(url, options),
new Promise((_, reject) => setTimeout(() => reject(new Error('timeout')), timeout))
]);
}
var playgrounds = Array.from(document.querySelectorAll(".playground"));
if (playgrounds.length > 0) {
fetch_with_timeout("https://play.rust-lang.org/meta/crates", {
headers: {
'Content-Type': "application/json",
},
method: 'POST',
mode: 'cors',
})
.then(response => response.json())
.then(response => {
// get list of crates available in the rust playground
let playground_crates = response.crates.map(item => item["id"]);
playgrounds.forEach(block => handle_crate_list_update(block, playground_crates));
});
}
function handle_crate_list_update(playground_block, playground_crates) {
// update the play buttons after receiving the response
update_play_button(playground_block, playground_crates);
// and install on change listener to dynamically update ACE editors
if (window.ace) {
let code_block = playground_block.querySelector("code");
if (code_block.classList.contains("editable")) {
let editor = window.ace.edit(code_block);
editor.addEventListener("change", function (e) {
update_play_button(playground_block, playground_crates);
});
// add Ctrl-Enter command to execute rust code
editor.commands.addCommand({
name: "run",
bindKey: {
win: "Ctrl-Enter",
mac: "Ctrl-Enter"
},
exec: _editor => run_rust_code(playground_block)
});
}
}
}
// updates the visibility of play button based on `no_run` class and
// used crates vs ones available on https://play.rust-lang.org
function update_play_button(pre_block, playground_crates) {
var play_button = pre_block.querySelector(".play-button");
// skip if code is `no_run`
if (pre_block.querySelector('code').classList.contains("no_run")) {
play_button.classList.add("hidden");
return;
}
// get list of `extern crate`'s from snippet
var txt = playground_text(pre_block);
var re = /extern\s+crate\s+([a-zA-Z_0-9]+)\s*;/g;
var snippet_crates = [];
var item;
while (item = re.exec(txt)) {
snippet_crates.push(item[1]);
}
// check if all used crates are available on play.rust-lang.org
var all_available = snippet_crates.every(function (elem) {
return playground_crates.indexOf(elem) > -1;
});
if (all_available) {
play_button.classList.remove("hidden");
} else {
play_button.classList.add("hidden");
}
}
function run_rust_code(code_block) {
var result_block = code_block.querySelector(".result");
if (!result_block) {
result_block = document.createElement('code');
result_block.className = 'result hljs language-bash';
code_block.append(result_block);
}
let text = playground_text(code_block);
let classes = code_block.querySelector('code').classList;
let edition = "2015";
if(classes.contains("edition2018")) {
edition = "2018";
} else if(classes.contains("edition2021")) {
edition = "2021";
}
var params = {
version: "stable",
optimize: "0",
code: text,
edition: edition
};
if (text.indexOf("#![feature") !== -1) {
params.version = "nightly";
}
result_block.innerText = "Running...";
fetch_with_timeout("https://play.rust-lang.org/evaluate.json", {
headers: {
'Content-Type': "application/json",
},
method: 'POST',
mode: 'cors',
body: JSON.stringify(params)
})
.then(response => response.json())
.then(response => {
if (response.result.trim() === '') {
result_block.innerText = "No output";
result_block.classList.add("result-no-output");
} else {
result_block.innerText = response.result;
result_block.classList.remove("result-no-output");
}
})
.catch(error => result_block.innerText = "Playground Communication: " + error.message);
}
// Syntax highlighting Configuration
hljs.configure({
tabReplace: ' ', // 4 spaces
languages: [], // Languages used for auto-detection
});
let code_nodes = Array
.from(document.querySelectorAll('code'))
// Don't highlight `inline code` blocks in headers.
.filter(function (node) {return !node.parentElement.classList.contains("header"); });
if (window.ace) {
// language-rust class needs to be removed for editable
// blocks or highlightjs will capture events
code_nodes
.filter(function (node) {return node.classList.contains("editable"); })
.forEach(function (block) { block.classList.remove('language-rust'); });
code_nodes
.filter(function (node) {return !node.classList.contains("editable"); })
.forEach(function (block) { hljs.highlightBlock(block); });
} else {
code_nodes.forEach(function (block) { hljs.highlightBlock(block); });
}
// Adding the hljs class gives code blocks the color css
// even if highlighting doesn't apply
code_nodes.forEach(function (block) { block.classList.add('hljs'); });
Array.from(document.querySelectorAll("code.language-rust")).forEach(function (block) {
var lines = Array.from(block.querySelectorAll('.boring'));
// If no lines were hidden, return
if (!lines.length) { return; }
block.classList.add("hide-boring");
var buttons = document.createElement('div');
buttons.className = 'buttons';
buttons.innerHTML = "<button class=\"fa fa-eye\" title=\"Show hidden lines\" aria-label=\"Show hidden lines\"></button>";
// add expand button
var pre_block = block.parentNode;
pre_block.insertBefore(buttons, pre_block.firstChild);
pre_block.querySelector('.buttons').addEventListener('click', function (e) {
if (e.target.classList.contains('fa-eye')) {
e.target.classList.remove('fa-eye');
e.target.classList.add('fa-eye-slash');
e.target.title = 'Hide lines';
e.target.setAttribute('aria-label', e.target.title);
block.classList.remove('hide-boring');
} else if (e.target.classList.contains('fa-eye-slash')) {
e.target.classList.remove('fa-eye-slash');
e.target.classList.add('fa-eye');
e.target.title = 'Show hidden lines';
e.target.setAttribute('aria-label', e.target.title);
block.classList.add('hide-boring');
}
});
});
if (window.playground_copyable) {
Array.from(document.querySelectorAll('pre code')).forEach(function (block) {
var pre_block = block.parentNode;
if (!pre_block.classList.contains('playground')) {
var buttons = pre_block.querySelector(".buttons");
if (!buttons) {
buttons = document.createElement('div');
buttons.className = 'buttons';
pre_block.insertBefore(buttons, pre_block.firstChild);
}
var clipButton = document.createElement('button');
clipButton.className = 'fa fa-copy clip-button';
clipButton.title = 'Copy to clipboard';
clipButton.setAttribute('aria-label', clipButton.title);
clipButton.innerHTML = '<i class=\"tooltiptext\"></i>';
buttons.insertBefore(clipButton, buttons.firstChild);
}
});
}
// Process playground code blocks
Array.from(document.querySelectorAll(".playground")).forEach(function (pre_block) {
// Add play button
var buttons = pre_block.querySelector(".buttons");
if (!buttons) {
buttons = document.createElement('div');
buttons.className = 'buttons';
pre_block.insertBefore(buttons, pre_block.firstChild);
}
var runCodeButton = document.createElement('button');
runCodeButton.className = 'fa fa-play play-button';
runCodeButton.hidden = true;
runCodeButton.title = 'Run this code';
runCodeButton.setAttribute('aria-label', runCodeButton.title);
buttons.insertBefore(runCodeButton, buttons.firstChild);
runCodeButton.addEventListener('click', function (e) {
run_rust_code(pre_block);
});
if (window.playground_copyable) {
var copyCodeClipboardButton = document.createElement('button');
copyCodeClipboardButton.className = 'fa fa-copy clip-button';
copyCodeClipboardButton.innerHTML = '<i class="tooltiptext"></i>';
copyCodeClipboardButton.title = 'Copy to clipboard';
copyCodeClipboardButton.setAttribute('aria-label', copyCodeClipboardButton.title);
buttons.insertBefore(copyCodeClipboardButton, buttons.firstChild);
}
let code_block = pre_block.querySelector("code");
if (window.ace && code_block.classList.contains("editable")) {
var undoChangesButton = document.createElement('button');
undoChangesButton.className = 'fa fa-history reset-button';
undoChangesButton.title = 'Undo changes';
undoChangesButton.setAttribute('aria-label', undoChangesButton.title);
buttons.insertBefore(undoChangesButton, buttons.firstChild);
undoChangesButton.addEventListener('click', function () {
let editor = window.ace.edit(code_block);
editor.setValue(editor.originalCode);
editor.clearSelection();
});
}
});
})();
(function themes() {
var html = document.querySelector('html');
var themeToggleButton = document.getElementById('theme-toggle');
var themePopup = document.getElementById('theme-list');
var themeColorMetaTag = document.querySelector('meta[name="theme-color"]');
var stylesheets = {
ayuHighlight: document.querySelector("[href$='ayu-highlight.css']"),
tomorrowNight: document.querySelector("[href$='tomorrow-night.css']"),
highlight: document.querySelector("[href$='highlight.css']"),
};
function showThemes() {
themePopup.style.display = 'block';
themeToggleButton.setAttribute('aria-expanded', true);
themePopup.querySelector("button#" + get_theme()).focus();
}
function updateThemeSelected() {
themePopup.querySelectorAll('.theme-selected').forEach(function (el) {
el.classList.remove('theme-selected');
});
themePopup.querySelector("button#" + get_theme()).classList.add('theme-selected');
}
function hideThemes() {
themePopup.style.display = 'none';
themeToggleButton.setAttribute('aria-expanded', false);
themeToggleButton.focus();
}
function get_theme() {
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch (e) { }
if (theme === null || theme === undefined) {
return default_theme;
} else {
return theme;
}
}
function set_theme(theme, store = true) {
let ace_theme;
if (theme == 'coal' || theme == 'navy') {
stylesheets.ayuHighlight.disabled = true;
stylesheets.tomorrowNight.disabled = false;
stylesheets.highlight.disabled = true;
ace_theme = "ace/theme/tomorrow_night";
} else if (theme == 'ayu') {
stylesheets.ayuHighlight.disabled = false;
stylesheets.tomorrowNight.disabled = true;
stylesheets.highlight.disabled = true;
ace_theme = "ace/theme/tomorrow_night";
} else {
stylesheets.ayuHighlight.disabled = true;
stylesheets.tomorrowNight.disabled = true;
stylesheets.highlight.disabled = false;
ace_theme = "ace/theme/dawn";
}
setTimeout(function () {
themeColorMetaTag.content = getComputedStyle(document.body).backgroundColor;
}, 1);
if (window.ace && window.editors) {
window.editors.forEach(function (editor) {
editor.setTheme(ace_theme);
});
}
var previousTheme = get_theme();
if (store) {
try { localStorage.setItem('mdbook-theme', theme); } catch (e) { }
}
html.classList.remove(previousTheme);
html.classList.add(theme);
updateThemeSelected();
}
// Set theme
var theme = get_theme();
set_theme(theme, false);
themeToggleButton.addEventListener('click', function () {
if (themePopup.style.display === 'block') {
hideThemes();
} else {
showThemes();
}
});
themePopup.addEventListener('click', function (e) {
var theme;
if (e.target.className === "theme") {
theme = e.target.id;
} else if (e.target.parentElement.className === "theme") {
theme = e.target.parentElement.id;
} else {
return;
}
set_theme(theme);
});
themePopup.addEventListener('focusout', function(e) {
// e.relatedTarget is null in Safari and Firefox on macOS (see workaround below)
if (!!e.relatedTarget && !themeToggleButton.contains(e.relatedTarget) && !themePopup.contains(e.relatedTarget)) {
hideThemes();
}
});
// Should not be needed, but it works around an issue on macOS & iOS: https://github.com/rust-lang/mdBook/issues/628
document.addEventListener('click', function(e) {
if (themePopup.style.display === 'block' && !themeToggleButton.contains(e.target) && !themePopup.contains(e.target)) {
hideThemes();
}
});
document.addEventListener('keydown', function (e) {
if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) { return; }
if (!themePopup.contains(e.target)) { return; }
switch (e.key) {
case 'Escape':
e.preventDefault();
hideThemes();
break;
case 'ArrowUp':
e.preventDefault();
var li = document.activeElement.parentElement;
if (li && li.previousElementSibling) {
li.previousElementSibling.querySelector('button').focus();
}
break;
case 'ArrowDown':
e.preventDefault();
var li = document.activeElement.parentElement;
if (li && li.nextElementSibling) {
li.nextElementSibling.querySelector('button').focus();
}
break;
case 'Home':
e.preventDefault();
themePopup.querySelector('li:first-child button').focus();
break;
case 'End':
e.preventDefault();
themePopup.querySelector('li:last-child button').focus();
break;
}
});
})();
(function sidebar() {
var html = document.querySelector("html");
var sidebar = document.getElementById("sidebar");
var sidebarLinks = document.querySelectorAll('#sidebar a');
var sidebarToggleButton = document.getElementById("sidebar-toggle");
var sidebarResizeHandle = document.getElementById("sidebar-resize-handle");
var firstContact = null;
function showSidebar() {
html.classList.remove('sidebar-hidden')
html.classList.add('sidebar-visible');
Array.from(sidebarLinks).forEach(function (link) {
link.setAttribute('tabIndex', 0);
});
sidebarToggleButton.setAttribute('aria-expanded', true);
sidebar.setAttribute('aria-hidden', false);
try { localStorage.setItem('mdbook-sidebar', 'visible'); } catch (e) { }
}
var sidebarAnchorToggles = document.querySelectorAll('#sidebar a.toggle');
function toggleSection(ev) {
ev.currentTarget.parentElement.classList.toggle('expanded');
}
Array.from(sidebarAnchorToggles).forEach(function (el) {
el.addEventListener('click', toggleSection);
});
function hideSidebar() {
html.classList.remove('sidebar-visible')
html.classList.add('sidebar-hidden');
Array.from(sidebarLinks).forEach(function (link) {
link.setAttribute('tabIndex', -1);
});
sidebarToggleButton.setAttribute('aria-expanded', false);
sidebar.setAttribute('aria-hidden', true);
try { localStorage.setItem('mdbook-sidebar', 'hidden'); } catch (e) { }
}
// Toggle sidebar
sidebarToggleButton.addEventListener('click', function sidebarToggle() {
if (html.classList.contains("sidebar-hidden")) {
var current_width = parseInt(
document.documentElement.style.getPropertyValue('--sidebar-width'), 10);
if (current_width < 150) {
document.documentElement.style.setProperty('--sidebar-width', '150px');
}
showSidebar();
} else if (html.classList.contains("sidebar-visible")) {
hideSidebar();
} else {
if (getComputedStyle(sidebar)['transform'] === 'none') {
hideSidebar();
} else {
showSidebar();
}
}
});
sidebarResizeHandle.addEventListener('mousedown', initResize, false);
function initResize(e) {
window.addEventListener('mousemove', resize, false);
window.addEventListener('mouseup', stopResize, false);
html.classList.add('sidebar-resizing');
}
function resize(e) {
var pos = (e.clientX - sidebar.offsetLeft);
if (pos < 20) {
hideSidebar();
} else {
if (html.classList.contains("sidebar-hidden")) {
showSidebar();
}
pos = Math.min(pos, window.innerWidth - 100);
document.documentElement.style.setProperty('--sidebar-width', pos + 'px');
}
}
//on mouseup remove windows functions mousemove & mouseup
function stopResize(e) {
html.classList.remove('sidebar-resizing');
window.removeEventListener('mousemove', resize, false);
window.removeEventListener('mouseup', stopResize, false);
}
document.addEventListener('touchstart', function (e) {
firstContact = {
x: e.touches[0].clientX,
time: Date.now()
};
}, { passive: true });
document.addEventListener('touchmove', function (e) {
if (!firstContact)
return;
var curX = e.touches[0].clientX;
var xDiff = curX - firstContact.x,
tDiff = Date.now() - firstContact.time;
if (tDiff < 250 && Math.abs(xDiff) >= 150) {
if (xDiff >= 0 && firstContact.x < Math.min(document.body.clientWidth * 0.25, 300))
showSidebar();
else if (xDiff < 0 && curX < 300)
hideSidebar();
firstContact = null;
}
}, { passive: true });
// Scroll sidebar to current active section
var activeSection = document.getElementById("sidebar").querySelector(".active");
if (activeSection) {
// https://developer.mozilla.org/en-US/docs/Web/API/Element/scrollIntoView
activeSection.scrollIntoView({ block: 'center' });
}
})();
(function chapterNavigation() {
document.addEventListener('keydown', function (e) {
if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) { return; }
if (window.search && window.search.hasFocus()) { return; }
switch (e.key) {
case 'ArrowRight':
e.preventDefault();
var nextButton = document.querySelector('.nav-chapters.next');
if (nextButton) {
window.location.href = nextButton.href;
}
break;
case 'ArrowLeft':
e.preventDefault();
var previousButton = document.querySelector('.nav-chapters.previous');
if (previousButton) {
window.location.href = previousButton.href;
}
break;
}
});
})();
(function clipboard() {
var clipButtons = document.querySelectorAll('.clip-button');
function hideTooltip(elem) {
elem.firstChild.innerText = "";
elem.className = 'fa fa-copy clip-button';
}
function showTooltip(elem, msg) {
elem.firstChild.innerText = msg;
elem.className = 'fa fa-copy tooltipped';
}
var clipboardSnippets = new ClipboardJS('.clip-button', {
text: function (trigger) {
hideTooltip(trigger);
let playground = trigger.closest("pre");
return playground_text(playground, false);
}
});
Array.from(clipButtons).forEach(function (clipButton) {
clipButton.addEventListener('mouseout', function (e) {
hideTooltip(e.currentTarget);
});
});
clipboardSnippets.on('success', function (e) {
e.clearSelection();
showTooltip(e.trigger, "Copied!");
});
clipboardSnippets.on('error', function (e) {
showTooltip(e.trigger, "Clipboard error!");
});
})();
(function scrollToTop () {
var menuTitle = document.querySelector('.menu-title');
menuTitle.addEventListener('click', function () {
document.scrollingElement.scrollTo({ top: 0, behavior: 'smooth' });
});
})();
(function controllMenu() {
var menu = document.getElementById('menu-bar');
(function controllPosition() {
var scrollTop = document.scrollingElement.scrollTop;
var prevScrollTop = scrollTop;
var minMenuY = -menu.clientHeight - 50;
// When the script loads, the page can be at any scroll (e.g. if you reforesh it).
menu.style.top = scrollTop + 'px';
// Same as parseInt(menu.style.top.slice(0, -2), but faster
var topCache = menu.style.top.slice(0, -2);
menu.classList.remove('sticky');
var stickyCache = false; // Same as menu.classList.contains('sticky'), but faster
document.addEventListener('scroll', function () {
scrollTop = Math.max(document.scrollingElement.scrollTop, 0);
// `null` means that it doesn't need to be updated
var nextSticky = null;
var nextTop = null;
var scrollDown = scrollTop > prevScrollTop;
var menuPosAbsoluteY = topCache - scrollTop;
if (scrollDown) {
nextSticky = false;
if (menuPosAbsoluteY > 0) {
nextTop = prevScrollTop;
}
} else {
if (menuPosAbsoluteY > 0) {
nextSticky = true;
} else if (menuPosAbsoluteY < minMenuY) {
nextTop = prevScrollTop + minMenuY;
}
}
if (nextSticky === true && stickyCache === false) {
menu.classList.add('sticky');
stickyCache = true;
} else if (nextSticky === false && stickyCache === true) {
menu.classList.remove('sticky');
stickyCache = false;
}
if (nextTop !== null) {
menu.style.top = nextTop + 'px';
topCache = nextTop;
}
prevScrollTop = scrollTop;
}, { passive: true });
})();
(function controllBorder() {
menu.classList.remove('bordered');
document.addEventListener('scroll', function () {
if (menu.offsetTop === 0) {
menu.classList.remove('bordered');
} else {
menu.classList.add('bordered');
}
}, { passive: true });
})();
})();

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,430 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>Configurable service - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item "><a href="/docserver/index.html"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item "><a href="/docserver/features.html"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item "><a href="/docserver/howto.html"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item "><a href="/docserver/auths.html"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item "><a href="/docserver/adminusers.html"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/cookie.html"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item expanded "><a href="/docserver/configuration.html" class="active"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h1 id="configurable-service"><a class="header" href="#configurable-service">Configurable service</a></h1>
<p><a href="/docserver/">Doc Server</a> can be adjusted to differents <a href="#modes">modes</a> with <a href="settings.html">Settings</a> in <a href="#config-file">Config file</a>.</p>
<div class="box"><img src="images/docserver.svg" class="logo" /></div>
<h2 id="config-file"><a class="header" href="#config-file">Config file</a></h2>
<p><strong>Config file</strong> path with <a href="settings.html">Settings</a> is passed as argument to <a href="/docserver/">Doc Server</a> executable as:</p>
<pre><code class="language-bash">docserver --config &lt;CONFIG_PATH&gt;
</code></pre>
<p>It should be a <strong><a href="https://en.wikipedia.org/wiki/TOML">TOML</a> file-path with Configuration <a href="settings.html">settings</a> to run <code>WebServer</code></strong>, it is REQUIRED to run <a href="/docserver/">Doc Server</a></p>
<p>It is composed by the following parts:</p>
<ul>
<li>
<p><a href="configuration.html#main">Paths, URLs, Web Server main settings</a> (port, host, protocol, SSL certificates, etc)</p>
</li>
<li>
<p><a href="configuration.html#signup">Signup and Signin</a> settings: <a href="configuration.html#password-score">password</a>, <a href="configuration.html#totp">TOTP</a>, <a href="configuration.html#admin-user-fields">admin fields</a>, etc.</p>
</li>
<li>
<p><a href="configuration.html#mail-service">Mail</a> settings if enabled</p>
</li>
<li>
<p><a href="configuration.html#stores">Stores</a> locations for Users, Auths and Sessions</p>
</li>
<li>
<p><a href="configuration.html#templates">Templates</a>: <strong>tpls</strong> to render requests responses</p>
</li>
<li>
<p><a href="pasetoken.html">Pasetoken</a>: <strong>paseto</strong> for token configurations</p>
</li>
<li>
<p><a href="configuration.html#ui">UI</a> where major user interface are defined:</p>
<ul>
<li>
<p><strong>Assets links</strong> for JS and CSS</p>
</li>
<li>
<p><strong>web_menu_items</strong> as items to show in top menu navigator</p>
</li>
</ul>
</li>
<li>
<p><a href="configuration.html#serv-paths">serv_paths</a> where <a href="static_content.html">static content</a> path are defined as <strong>is_restricted</strong> and with allowed <a href="roles.html">roles</a></p>
</li>
</ul>
<h2 id="main"><a class="header" href="#main">Main</a></h2>
<p>On top of <strong>Configuration file</strong>:</p>
<ul>
<li>
<p><strong>root_path</strong> = “sitehome”, the most important as <u>root path</u>, other settings attibutes ending with <strong>_path</strong> can have relative path to this one or their own absolute or relative paths.</p>
</li>
<li>
<p><strong>name</strong> = “docserver” is a name for service</p>
</li>
<li>
<p><strong>org</strong> = “ORG” an organization name</p>
</li>
<li>
<p><strong>verbose</strong> = 2 some debug info in requests handling is shown with values over 0</p>
</li>
<li>
<p><strong>templates_path</strong> = “templates”, <a href="configuration.html#templates">templates</a> path</p>
</li>
<li>
<p><strong>assets_url</strong> = “/assets” assets path for <a href="configuration.html#templates">templates</a></p>
</li>
<li>
<p><strong>allow_origin</strong> = [“http://localhost:8080”] allow origins for <a href="https://developer.mozilla.org/es/docs/Web/HTTP/Headers/Access-Control-Allow-Origin">access control allow origin</a> webserver directive</p>
</li>
</ul>
<h2 id="https"><a class="header" href="#https">HTTPS</a></h2>
<p><a href="https://en.wikipedia.org/wiki/Transport_Layer_Security">TLS</a> certificates path can be provided in:</p>
<pre><code class="language-toml">cert_file = &quot;cert/fullchain.pem&quot;
key_file = &quot;cert/privkey.pem&quot;
</code></pre>
<p>To use <strong>HTTPS</strong> traffic set <strong>protocol</strong> to <strong>https</strong> rather than <strong>http</strong></p>
<pre><code class="language-toml">protocol = &quot;https&quot;
</code></pre>
<h2 id="signup"><a class="header" href="#signup">Signup</a></h2>
<p>It is about how new users will be created, there are two option:</p>
<ul>
<li>
<p><strong>Open</strong> by using a <strong>signup</strong> form with all details.</p>
</li>
<li>
<p><strong>invitation</strong> by <strong>adminitrators</strong>, new user would receive an <strong>invitation link</strong> (it would include some details like email, roles,etc) connected to a <a href="sesions.html">session</a> token with <strong>expiration time</strong> in seconds, this link can be sent by email if is set in config. By using an <strong>invitation link</strong> new user can fill a <strong>signup</strong> form with all details.</p>
</li>
</ul>
<pre><code class="language-toml">## How to signup by 'invitation' or 'open'
signup_mode = &quot;invitation&quot;
invite_expire = 900 # in seconds
</code></pre>
<h2 id="totp"><a class="header" href="#totp">TOTP</a></h2>
<p><a href="https://en.wikipedia.org/wiki/Time-based_one-time_password">TOPT</a> as <a href="https://en.wikipedia.org/wiki/Multi-factor_authentication">2FA</a> by default is set to <strong>no</strong> but it can also be:</p>
<ul>
<li>
<p><strong>optional</strong> to users settings, so they can activate or deactivate in <strong>signup</strong> and <u>user settings</u> edition.</p>
</li>
<li>
<p><strong>mandatory</strong> means users have to configure in <strong>signup</strong> process.</p>
</li>
</ul>
<p><a href="https://en.wikipedia.org/wiki/Time-based_one-time_password">TOPT</a> can only be reset by <strong>user settings</strong> or <strong>administrators</strong> via <strong>user edition form</strong></p>
<pre><code class="language-toml"># topt Mode: mandatory, optional, no (default)
totp_mode = &quot;optional&quot;
# totp Digits: 6 (default) or 8
totp_digits = 6
# topt Algorithm: sha1 (default), sha256, sha512
totp_algorithm = &quot;sha256&quot;
</code></pre>
<h2 id="password-score"><a class="header" href="#password-score">Password score</a></h2>
<p><a href="/docserver/">Doc Server</a> use <a href="https://www.rust-lang.org">Rust</a> crate <a href="https://github.com/shssoichiro/zxcvbn-rs">zxcvbn</a> for password strength estimator, it can be enforced to a value:</p>
<pre><code class="language-toml">password_score = 3
</code></pre>
<h2 id="admin-user-fields"><a class="header" href="#admin-user-fields">Admin user fields</a></h2>
<p><strong>admin_fields</strong> is a list of user fields or attributes to change only by <strong>administrators</strong>, user can only see their values.</p>
<pre><code class="language-toml">admin_fields = &quot;roles,otp_base32,status&quot;
</code></pre>
<h2 id="mail-service"><a class="header" href="#mail-service">Mail service</a></h2>
<p>It can be set as <strong>true</strong> or <strong>false</strong>, <strong>smtp_auth</strong> is token encrypted that can be generated by <strong>script</strong> <strong>tools/get_mail_token.sh</strong> that runs <a href="/docserver/">Doc Server</a> program using credentials from <a href="pasetoken.html">paseto</a> section (<strong>public_path</strong> and <strong>secret_path</strong>), it can be included not encrypted but is highly not recomended.</p>
<blockquote>
<p>tools/get_mail_token.sh [mail_user] [mail_password]
Provides a token based in srcv pasetoken settings</p>
</blockquote>
<p>As an option enviroment variables MAIL_USER and MAIL_PASSWORD can also be set before to run <strong>tools/get_mail_token.sh</strong></p>
<pre><code class="language-toml">use_mail = true
smtp = &quot;mail.example.com&quot;
smtp_auth = &quot;v4.public...&quot;
mail_from =&quot;admini@example.com&quot;
</code></pre>
<h2 id="stores"><a class="header" href="#stores">Stores</a></h2>
<p><a href="users.html">Users</a> and <a href="sessions.html">Sessions</a> are stored in async <a href="https://github.com/launchbadge/sqlx">sqlx</a> to <a href="sqlite.html">Sqlite</a> or <a href="file_storage.html">File</a> storages, for larger use <a href="https://github.com/launchbadge/sqlx">sqlx</a> as <u>Database Agnostic</u> provides other <a href="https://en.wikipedia.org/wiki/SQL">SQL</a> options</p>
<blockquote>
<p>Prefix <strong>sqlite::</strong> is for <a href="https://www.sqlite.org/index.html">SQLite</a> storage, usually a file ends with <strong>.db</strong> suffix.</p>
</blockquote>
<pre><code class="language-toml">users_store_uri = &quot;sqlite:PATH_TO_users.db&quot;
session_store_uri = &quot;sqlite:PATH_TO_sessions.db&quot;
session_expire = 300
</code></pre>
<blockquote>
<p>Prefix <strong>file:///</strong> if for <strong>FILE</strong> storage, value should be a directory. <strong>session_store_file</strong> value will be final file prefix follow by <a href="https://en.wikipedia.org/wiki/Timestamp">timestamp</a> autogenerated value.</p>
</blockquote>
<pre><code class="language-toml">users_store_uri = &quot;file:///PATH_TO_users_DIRECTORY&quot;
session_store_uri = &quot;file:///PATH_TO_sessions_DIRECTORY&quot;
session_store_file = &quot;session&quot;
session_expire = 300
</code></pre>
<blockquote>
<p><strong>session_expire</strong> is a value in seconds for a session to expire ir is not used, with each request expire session is extended for same value.</p>
</blockquote>
<h2 id="policies"><a class="header" href="#policies">Policies</a></h2>
<p><a href="policies.html">Access policies</a> to be <strong>enfoced</strong> for content access with <a href="https://github.com/casbin/casbin-rs">Casbin</a> require two file path:</p>
<pre><code class="language-toml">authz_model_path = &quot;PATH_TO_model.conf&quot;
authz_policy_path = &quot;PATH_TO_policy.csv&quot;
</code></pre>
<h2 id="templates"><a class="header" href="#templates">Templates</a></h2>
<p><a href="https://tera.netlify.app/">Tera template engine</a> is used to to render response requests. Most of templates are <strong>HTML</strong> and in many cases use to include others from <strong>partials</strong> directory. Pre-render process is in charge of values setting.</p>
<p>Templates suffixes indicate if is <strong>html</strong> or <strong>txt</strong> and ends with <strong>j2</strong> as standard extension for <a href="https://jinja.palletsprojects.com/en/3.1.x/">jinja2</a> templating engine.</p>
<p>Default <strong>templates_path</strong> = “templates” directory layout:</p>
<pre><code class="language-text">├── home.html.j2
├── invite_create.html.j2
├── invite_mail.html.j2
├── invite_mail.txt.j2
├── invite_output.html.j2
├── login.html.j2
├── logout.html.j2
├── pages
│   └── terms-conditions.html.j2
├── partials
│   ├── footer.html.j2
│   ├── form-password.html.j2
│   ├── form-totp.html.j2
│   ├── form-user.html.j2
│   ├── header.html.j2
│   ├── loading.html.j2
│   ├── mini_navbar.html.j2
│   ├── modal_edit_user.html.j2
│   ├── _navbar.html.j2
│   ├── navbar.html.j2
│   └── symbols.html.j2
├── signup.html.j2
├── user_settings.html.j2
└── users.html.j2
</code></pre>
<p>Current templates use <a href="https://unocss.dev/">UnoCSS</a> as CSS engine (compatible with <a href="https://tailwindcss.com/">Tailwindcss</a>) and <a href="https://flowbite.com/docs/getting-started/introduction/">FLowbyte</a> as CSS and JS component library, so they are added from <a href="configuration.html#ui">UI</a> configuration values</p>
<pre><code class="language-toml">[ui]
css_link = &quot;https://cdn.jsdelivr.net/npm/@unocss/reset/tailwind.min.css&quot;
js_link = &quot;https://cdn.jsdelivr.net/npm/@unocss/runtime&quot;
other_css_link = &quot;https://cdnjs.cloudflare.com/ajax/libs/flowbite/1.6.5/flowbite.min.css&quot;
other_js_link=&quot;https://cdnjs.cloudflare.com/ajax/libs/flowbite/1.6.5/flowbite.min.js&quot;
</code></pre>
<p>Other <strong>JS</strong> values are used for basic tasks in page interaction:</p>
<pre><code class="language-toml">[ui]
main_js_link = &quot;/assets/js/main.js&quot;
utils_js_link = &quot;/assets/js/utils.js&quot;
</code></pre>
<p>A <strong>main.css</strong> is used for customized <strong>CSS</strong>, is loaded in <strong>partials/header.html.j2</strong> template as (pay attention to <strong>asserts_url</strong>):</p>
<pre><code class="language-python">/css/main.css
</code></pre>
<p>Values under <strong>[tpls]</strong> section are mapped values for <strong>template filename</strong> used in <a href="/docserver/">Doc Server</a> request handlers to render a response, so basically filenames in <strong>templates_path</strong> value directory.</p>
<h2 id="ui"><a class="header" href="#ui">UI</a></h2>
<p>Contains values for <a href="/configuration.html#templates">templates</a> render for <strong>assets</strong> files: <strong>js</strong>, <strong>css</strong>.</p>
<p>Each <strong>web_menu_items</strong> entry set values (<strong>text</strong>, <strong>url</strong>, <strong>roles</strong>) for each <strong>navigation menu</strong> template used in pages.</p>
<h2 id="serv-paths"><a class="header" href="#serv-paths">Serv Paths</a></h2>
<p>An entry for each <a href="static_content.html">static_content</a> like in example bellow</p>
<p>Example:</p>
<pre><code class="language-toml">[[serve_paths]]
# Server relative Directory Path (sitehome/appdoc)
srv_path = &quot;Server DIRECTORY_PATH&quot;
# Absolute URL path for resquests
url_path = &quot;/appdoc&quot;
# Not found URL to be redirect
not_found = &quot;&quot;
# URL to be redirect if is not authorization found
not_auth = &quot;/login&quot;
# true or false to access control or not to this PATH
is_restricted = true
</code></pre>
<blockquote>
<p>Access <a href="%7B%7D">policies</a> for <strong>serve_path.url_path</strong> should be in <strong>authz_policy_path</strong> file</p>
</blockquote>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="logs.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="settings.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="logs.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="settings.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

View File

@ -0,0 +1,212 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>Cookie set - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item "><a href="/docserver/index.html"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item "><a href="/docserver/features.html"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item "><a href="/docserver/howto.html"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item "><a href="/docserver/auths.html"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item "><a href="/docserver/adminusers.html"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item expanded "><a href="/docserver/cookie.html" class="active"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item "><a href="/docserver/configuration.html"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h1 id="cookie-set"><a class="header" href="#cookie-set">Cookie set</a></h1>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="sessions_file_storage.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="trace.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="sessions_file_storage.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="trace.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

View File

@ -0,0 +1,545 @@
/* CSS for UI elements (a.k.a. chrome) */
@import 'variables.css';
html {
scrollbar-color: var(--scrollbar) var(--bg);
}
#searchresults a,
.content a:link,
a:visited,
a > .hljs {
color: var(--links);
}
/*
body-container is necessary because mobile browsers don't seem to like
overflow-x on the body tag when there is a <meta name="viewport"> tag.
*/
#body-container {
/*
This is used when the sidebar pushes the body content off the side of
the screen on small screens. Without it, dragging on mobile Safari
will want to reposition the viewport in a weird way.
*/
overflow-x: clip;
}
/* Menu Bar */
#menu-bar,
#menu-bar-hover-placeholder {
z-index: 101;
margin: auto calc(0px - var(--page-padding));
}
#menu-bar {
position: relative;
display: flex;
flex-wrap: wrap;
background-color: var(--bg);
border-bottom-color: var(--bg);
border-bottom-width: 1px;
border-bottom-style: solid;
}
#menu-bar.sticky,
.js #menu-bar-hover-placeholder:hover + #menu-bar,
.js #menu-bar:hover,
.js.sidebar-visible #menu-bar {
position: -webkit-sticky;
position: sticky;
top: 0 !important;
}
#menu-bar-hover-placeholder {
position: sticky;
position: -webkit-sticky;
top: 0;
height: var(--menu-bar-height);
}
#menu-bar.bordered {
border-bottom-color: var(--table-border-color);
}
#menu-bar i, #menu-bar .icon-button {
position: relative;
padding: 0 8px;
z-index: 10;
line-height: var(--menu-bar-height);
cursor: pointer;
transition: color 0.5s;
}
@media only screen and (max-width: 420px) {
#menu-bar i, #menu-bar .icon-button {
padding: 0 5px;
}
}
.icon-button {
border: none;
background: none;
padding: 0;
color: inherit;
}
.icon-button i {
margin: 0;
}
.right-buttons {
margin: 0 15px;
}
.right-buttons a {
text-decoration: none;
}
.left-buttons {
display: flex;
margin: 0 5px;
}
.no-js .left-buttons {
display: none;
}
.menu-title {
display: inline-block;
font-weight: 200;
font-size: 2.4rem;
line-height: var(--menu-bar-height);
text-align: center;
margin: 0;
flex: 1;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.js .menu-title {
cursor: pointer;
}
.menu-bar,
.menu-bar:visited,
.nav-chapters,
.nav-chapters:visited,
.mobile-nav-chapters,
.mobile-nav-chapters:visited,
.menu-bar .icon-button,
.menu-bar a i {
color: var(--icons);
}
.menu-bar i:hover,
.menu-bar .icon-button:hover,
.nav-chapters:hover,
.mobile-nav-chapters i:hover {
color: var(--icons-hover);
}
/* Nav Icons */
.nav-chapters {
font-size: 2.5em;
text-align: center;
text-decoration: none;
position: fixed;
top: 0;
bottom: 0;
margin: 0;
max-width: 150px;
min-width: 90px;
display: flex;
justify-content: center;
align-content: center;
flex-direction: column;
transition: color 0.5s, background-color 0.5s;
}
.nav-chapters:hover {
text-decoration: none;
background-color: var(--theme-hover);
transition: background-color 0.15s, color 0.15s;
}
.nav-wrapper {
margin-top: 50px;
display: none;
}
.mobile-nav-chapters {
font-size: 2.5em;
text-align: center;
text-decoration: none;
width: 90px;
border-radius: 5px;
background-color: var(--sidebar-bg);
}
.previous {
float: left;
}
.next {
float: right;
right: var(--page-padding);
}
@media only screen and (max-width: 1080px) {
.nav-wide-wrapper { display: none; }
.nav-wrapper { display: block; }
}
@media only screen and (max-width: 1380px) {
.sidebar-visible .nav-wide-wrapper { display: none; }
.sidebar-visible .nav-wrapper { display: block; }
}
/* Inline code */
:not(pre) > .hljs {
display: inline;
padding: 0.1em 0.3em;
border-radius: 3px;
}
:not(pre):not(a) > .hljs {
color: var(--inline-code-color);
overflow-x: initial;
}
a:hover > .hljs {
text-decoration: underline;
}
pre {
position: relative;
}
pre > .buttons {
position: absolute;
z-index: 100;
right: 0px;
top: 2px;
margin: 0px;
padding: 2px 0px;
color: var(--sidebar-fg);
cursor: pointer;
visibility: hidden;
opacity: 0;
transition: visibility 0.1s linear, opacity 0.1s linear;
}
pre:hover > .buttons {
visibility: visible;
opacity: 1
}
pre > .buttons :hover {
color: var(--sidebar-active);
border-color: var(--icons-hover);
background-color: var(--theme-hover);
}
pre > .buttons i {
margin-left: 8px;
}
pre > .buttons button {
cursor: inherit;
margin: 0px 5px;
padding: 3px 5px;
font-size: 14px;
border-style: solid;
border-width: 1px;
border-radius: 4px;
border-color: var(--icons);
background-color: var(--theme-popup-bg);
transition: 100ms;
transition-property: color,border-color,background-color;
color: var(--icons);
}
@media (pointer: coarse) {
pre > .buttons button {
/* On mobile, make it easier to tap buttons. */
padding: 0.3rem 1rem;
}
}
pre > code {
padding: 1rem;
}
/* FIXME: ACE editors overlap their buttons because ACE does absolute
positioning within the code block which breaks padding. The only solution I
can think of is to move the padding to the outer pre tag (or insert a div
wrapper), but that would require fixing a whole bunch of CSS rules.
*/
.hljs.ace_editor {
padding: 0rem 0rem;
}
pre > .result {
margin-top: 10px;
}
/* Search */
#searchresults a {
text-decoration: none;
}
mark {
border-radius: 2px;
padding: 0 3px 1px 3px;
margin: 0 -3px -1px -3px;
background-color: var(--search-mark-bg);
transition: background-color 300ms linear;
cursor: pointer;
}
mark.fade-out {
background-color: rgba(0,0,0,0) !important;
cursor: auto;
}
.searchbar-outer {
margin-left: auto;
margin-right: auto;
max-width: var(--content-max-width);
}
#searchbar {
width: 100%;
margin: 5px auto 0px auto;
padding: 10px 16px;
transition: box-shadow 300ms ease-in-out;
border: 1px solid var(--searchbar-border-color);
border-radius: 3px;
background-color: var(--searchbar-bg);
color: var(--searchbar-fg);
}
#searchbar:focus,
#searchbar.active {
box-shadow: 0 0 3px var(--searchbar-shadow-color);
}
.searchresults-header {
font-weight: bold;
font-size: 1em;
padding: 18px 0 0 5px;
color: var(--searchresults-header-fg);
}
.searchresults-outer {
margin-left: auto;
margin-right: auto;
max-width: var(--content-max-width);
border-bottom: 1px dashed var(--searchresults-border-color);
}
ul#searchresults {
list-style: none;
padding-left: 20px;
}
ul#searchresults li {
margin: 10px 0px;
padding: 2px;
border-radius: 2px;
}
ul#searchresults li.focus {
background-color: var(--searchresults-li-bg);
}
ul#searchresults span.teaser {
display: block;
clear: both;
margin: 5px 0 0 20px;
font-size: 0.8em;
}
ul#searchresults span.teaser em {
font-weight: bold;
font-style: normal;
}
/* Sidebar */
.sidebar {
position: fixed;
left: 0;
top: 0;
bottom: 0;
width: var(--sidebar-width);
font-size: 0.875em;
box-sizing: border-box;
-webkit-overflow-scrolling: touch;
overscroll-behavior-y: contain;
background-color: var(--sidebar-bg);
color: var(--sidebar-fg);
}
.sidebar-resizing {
-moz-user-select: none;
-webkit-user-select: none;
-ms-user-select: none;
user-select: none;
}
.js:not(.sidebar-resizing) .sidebar {
transition: transform 0.3s; /* Animation: slide away */
}
.sidebar code {
line-height: 2em;
}
.sidebar .sidebar-scrollbox {
overflow-y: auto;
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
padding: 10px 10px;
}
.sidebar .sidebar-resize-handle {
position: absolute;
cursor: col-resize;
width: 0;
right: 0;
top: 0;
bottom: 0;
}
.js .sidebar .sidebar-resize-handle {
cursor: col-resize;
width: 5px;
}
.sidebar-hidden .sidebar {
transform: translateX(calc(0px - var(--sidebar-width)));
}
.sidebar::-webkit-scrollbar {
background: var(--sidebar-bg);
}
.sidebar::-webkit-scrollbar-thumb {
background: var(--scrollbar);
}
.sidebar-visible .page-wrapper {
transform: translateX(var(--sidebar-width));
}
@media only screen and (min-width: 620px) {
.sidebar-visible .page-wrapper {
transform: none;
margin-left: var(--sidebar-width);
}
}
.chapter {
list-style: none outside none;
padding-left: 0;
line-height: 2.2em;
}
.chapter ol {
width: 100%;
}
.chapter li {
display: flex;
color: var(--sidebar-non-existant);
}
.chapter li a {
display: block;
padding: 0;
text-decoration: none;
color: var(--sidebar-fg);
}
.chapter li a:hover {
color: var(--sidebar-active);
}
.chapter li a.active {
color: var(--sidebar-active);
}
.chapter li > a.toggle {
cursor: pointer;
display: block;
margin-left: auto;
padding: 0 10px;
user-select: none;
opacity: 0.68;
}
.chapter li > a.toggle div {
transition: transform 0.5s;
}
/* collapse the section */
.chapter li:not(.expanded) + li > ol {
display: none;
}
.chapter li.chapter-item {
line-height: 1.5em;
margin-top: 0.6em;
}
.chapter li.expanded > a.toggle div {
transform: rotate(90deg);
}
.spacer {
width: 100%;
height: 3px;
margin: 5px 0px;
}
.chapter .spacer {
background-color: var(--sidebar-spacer);
}
@media (-moz-touch-enabled: 1), (pointer: coarse) {
.chapter li a { padding: 5px 0; }
.spacer { margin: 10px 0; }
}
.section {
list-style: none outside none;
padding-left: 20px;
line-height: 1.9em;
}
/* Theme Menu Popup */
.theme-popup {
position: absolute;
left: 10px;
top: var(--menu-bar-height);
z-index: 1000;
border-radius: 4px;
font-size: 0.7em;
color: var(--fg);
background: var(--theme-popup-bg);
border: 1px solid var(--theme-popup-border);
margin: 0;
padding: 0;
list-style: none;
display: none;
/* Don't let the children's background extend past the rounded corners. */
overflow: hidden;
}
.theme-popup .default {
color: var(--icons);
}
.theme-popup .theme {
width: 100%;
border: 0;
margin: 0;
padding: 2px 20px;
line-height: 25px;
white-space: nowrap;
text-align: left;
cursor: pointer;
color: inherit;
background: inherit;
font-size: inherit;
}
.theme-popup .theme:hover {
background-color: var(--theme-hover);
}
.theme-selected::before {
display: inline-block;
content: "✓";
margin-left: -14px;
width: 14px;
}

View File

@ -0,0 +1,204 @@
/* Base styles and content styles */
@import 'variables.css';
:root {
/* Browser default font-size is 16px, this way 1 rem = 10px */
font-size: 62.5%;
}
html {
font-family: "Open Sans", sans-serif;
color: var(--fg);
background-color: var(--bg);
text-size-adjust: none;
-webkit-text-size-adjust: none;
}
body {
margin: 0;
font-size: 1.6rem;
overflow-x: hidden;
}
code {
font-family: var(--mono-font) !important;
font-size: var(--code-font-size);
}
/* make long words/inline code not x overflow */
main {
overflow-wrap: break-word;
}
/* make wide tables scroll if they overflow */
.table-wrapper {
overflow-x: auto;
}
/* Don't change font size in headers. */
h1 code, h2 code, h3 code, h4 code, h5 code, h6 code {
font-size: unset;
}
.left { float: left; }
.right { float: right; }
.boring { opacity: 0.6; }
.hide-boring .boring { display: none; }
.hidden { display: none !important; }
h2, h3 { margin-top: 2.5em; }
h4, h5 { margin-top: 2em; }
.header + .header h3,
.header + .header h4,
.header + .header h5 {
margin-top: 1em;
}
h1:target::before,
h2:target::before,
h3:target::before,
h4:target::before,
h5:target::before,
h6:target::before {
display: inline-block;
content: "»";
margin-left: -30px;
width: 30px;
}
/* This is broken on Safari as of version 14, but is fixed
in Safari Technology Preview 117 which I think will be Safari 14.2.
https://bugs.webkit.org/show_bug.cgi?id=218076
*/
:target {
scroll-margin-top: calc(var(--menu-bar-height) + 0.5em);
}
.page {
outline: 0;
padding: 0 var(--page-padding);
margin-top: calc(0px - var(--menu-bar-height)); /* Compensate for the #menu-bar-hover-placeholder */
}
.page-wrapper {
box-sizing: border-box;
}
.js:not(.sidebar-resizing) .page-wrapper {
transition: margin-left 0.3s ease, transform 0.3s ease; /* Animation: slide away */
}
.content {
overflow-y: auto;
padding: 0 5px 50px 5px;
}
.content main {
margin-left: auto;
margin-right: auto;
max-width: var(--content-max-width);
}
.content p { line-height: 1.45em; }
.content ol { line-height: 1.45em; }
.content ul { line-height: 1.45em; }
.content a { text-decoration: none; }
.content a:hover { text-decoration: underline; }
.content img, .content video { max-width: 100%; }
.content .header:link,
.content .header:visited {
color: var(--fg);
}
.content .header:link,
.content .header:visited:hover {
text-decoration: none;
}
table {
margin: 0 auto;
border-collapse: collapse;
}
table td {
padding: 3px 20px;
border: 1px var(--table-border-color) solid;
}
table thead {
background: var(--table-header-bg);
}
table thead td {
font-weight: 700;
border: none;
}
table thead th {
padding: 3px 20px;
}
table thead tr {
border: 1px var(--table-header-bg) solid;
}
/* Alternate background colors for rows */
table tbody tr:nth-child(2n) {
background: var(--table-alternate-bg);
}
blockquote {
margin: 20px 0;
padding: 0 20px;
color: var(--fg);
background-color: var(--quote-bg);
border-top: .1em solid var(--quote-border);
border-bottom: .1em solid var(--quote-border);
}
kbd {
background-color: var(--table-border-color);
border-radius: 4px;
border: solid 1px var(--theme-popup-border);
box-shadow: inset 0 -1px 0 var(--theme-hover);
display: inline-block;
font-size: var(--code-font-size);
font-family: var(--mono-font);
line-height: 10px;
padding: 4px 5px;
vertical-align: middle;
}
:not(.footnote-definition) + .footnote-definition,
.footnote-definition + :not(.footnote-definition) {
margin-top: 2em;
}
.footnote-definition {
font-size: 0.9em;
margin: 0.5em 0;
}
.footnote-definition p {
display: inline;
}
.tooltiptext {
position: absolute;
visibility: hidden;
color: #fff;
background-color: #333;
transform: translateX(-50%); /* Center by moving tooltip 50% of its width left */
left: -8px; /* Half of the width of the icon */
top: -35px;
font-size: 0.8em;
text-align: center;
border-radius: 6px;
padding: 5px 8px;
margin: 5px;
z-index: 1000;
}
.tooltipped .tooltiptext {
visibility: visible;
}
.chapter li.part-title {
color: var(--sidebar-fg);
margin: 5px 0px;
font-weight: bold;
}
.result-no-output {
font-style: italic;
}

View File

@ -0,0 +1,54 @@
#sidebar,
#menu-bar,
.nav-chapters,
.mobile-nav-chapters {
display: none;
}
#page-wrapper.page-wrapper {
transform: none;
margin-left: 0px;
overflow-y: initial;
}
#content {
max-width: none;
margin: 0;
padding: 0;
}
.page {
overflow-y: initial;
}
code {
background-color: #666666;
border-radius: 5px;
/* Force background to be printed in Chrome */
-webkit-print-color-adjust: exact;
}
pre > .buttons {
z-index: 2;
}
a, a:visited, a:active, a:hover {
color: #4183c4;
text-decoration: none;
}
h1, h2, h3, h4, h5, h6 {
page-break-inside: avoid;
page-break-after: avoid;
}
pre, code {
page-break-inside: avoid;
white-space: pre-wrap;
}
.fa {
display: none !important;
}

View File

@ -0,0 +1,255 @@
/* Globals */
:root {
--sidebar-width: 300px;
--page-padding: 15px;
--content-max-width: 750px;
--menu-bar-height: 50px;
--mono-font: "Source Code Pro", Consolas, "Ubuntu Mono", Menlo, "DejaVu Sans Mono", monospace, monospace;
--code-font-size: 0.875em /* please adjust the ace font size accordingly in editor.js */
}
/* Themes */
.ayu {
--bg: hsl(210, 25%, 8%);
--fg: #c5c5c5;
--sidebar-bg: #14191f;
--sidebar-fg: #c8c9db;
--sidebar-non-existant: #5c6773;
--sidebar-active: #ffb454;
--sidebar-spacer: #2d334f;
--scrollbar: var(--sidebar-fg);
--icons: #737480;
--icons-hover: #b7b9cc;
--links: #0096cf;
--inline-code-color: #ffb454;
--theme-popup-bg: #14191f;
--theme-popup-border: #5c6773;
--theme-hover: #191f26;
--quote-bg: hsl(226, 15%, 17%);
--quote-border: hsl(226, 15%, 22%);
--table-border-color: hsl(210, 25%, 13%);
--table-header-bg: hsl(210, 25%, 28%);
--table-alternate-bg: hsl(210, 25%, 11%);
--searchbar-border-color: #848484;
--searchbar-bg: #424242;
--searchbar-fg: #fff;
--searchbar-shadow-color: #d4c89f;
--searchresults-header-fg: #666;
--searchresults-border-color: #888;
--searchresults-li-bg: #252932;
--search-mark-bg: #e3b171;
}
.coal {
--bg: hsl(200, 7%, 8%);
--fg: #98a3ad;
--sidebar-bg: #292c2f;
--sidebar-fg: #a1adb8;
--sidebar-non-existant: #505254;
--sidebar-active: #3473ad;
--sidebar-spacer: #393939;
--scrollbar: var(--sidebar-fg);
--icons: #43484d;
--icons-hover: #b3c0cc;
--links: #2b79a2;
--inline-code-color: #c5c8c6;
--theme-popup-bg: #141617;
--theme-popup-border: #43484d;
--theme-hover: #1f2124;
--quote-bg: hsl(234, 21%, 18%);
--quote-border: hsl(234, 21%, 23%);
--table-border-color: hsl(200, 7%, 13%);
--table-header-bg: hsl(200, 7%, 28%);
--table-alternate-bg: hsl(200, 7%, 11%);
--searchbar-border-color: #aaa;
--searchbar-bg: #b7b7b7;
--searchbar-fg: #000;
--searchbar-shadow-color: #aaa;
--searchresults-header-fg: #666;
--searchresults-border-color: #98a3ad;
--searchresults-li-bg: #2b2b2f;
--search-mark-bg: #355c7d;
}
.light {
--bg: hsl(0, 0%, 100%);
--fg: hsl(0, 0%, 0%);
--sidebar-bg: #fafafa;
--sidebar-fg: hsl(0, 0%, 0%);
--sidebar-non-existant: #aaaaaa;
--sidebar-active: #1f1fff;
--sidebar-spacer: #f4f4f4;
--scrollbar: #8F8F8F;
--icons: #747474;
--icons-hover: #000000;
--links: #20609f;
--inline-code-color: #301900;
--theme-popup-bg: #fafafa;
--theme-popup-border: #cccccc;
--theme-hover: #e6e6e6;
--quote-bg: hsl(197, 37%, 96%);
--quote-border: hsl(197, 37%, 91%);
--table-border-color: hsl(0, 0%, 95%);
--table-header-bg: hsl(0, 0%, 80%);
--table-alternate-bg: hsl(0, 0%, 97%);
--searchbar-border-color: #aaa;
--searchbar-bg: #fafafa;
--searchbar-fg: #000;
--searchbar-shadow-color: #aaa;
--searchresults-header-fg: #666;
--searchresults-border-color: #888;
--searchresults-li-bg: #e4f2fe;
--search-mark-bg: #a2cff5;
}
.navy {
--bg: hsl(226, 23%, 11%);
--fg: #bcbdd0;
--sidebar-bg: #282d3f;
--sidebar-fg: #c8c9db;
--sidebar-non-existant: #505274;
--sidebar-active: #2b79a2;
--sidebar-spacer: #2d334f;
--scrollbar: var(--sidebar-fg);
--icons: #737480;
--icons-hover: #b7b9cc;
--links: #2b79a2;
--inline-code-color: #c5c8c6;
--theme-popup-bg: #161923;
--theme-popup-border: #737480;
--theme-hover: #282e40;
--quote-bg: hsl(226, 15%, 17%);
--quote-border: hsl(226, 15%, 22%);
--table-border-color: hsl(226, 23%, 16%);
--table-header-bg: hsl(226, 23%, 31%);
--table-alternate-bg: hsl(226, 23%, 14%);
--searchbar-border-color: #aaa;
--searchbar-bg: #aeaec6;
--searchbar-fg: #000;
--searchbar-shadow-color: #aaa;
--searchresults-header-fg: #5f5f71;
--searchresults-border-color: #5c5c68;
--searchresults-li-bg: #242430;
--search-mark-bg: #a2cff5;
}
.rust {
--bg: hsl(60, 9%, 87%);
--fg: #262625;
--sidebar-bg: #3b2e2a;
--sidebar-fg: #c8c9db;
--sidebar-non-existant: #505254;
--sidebar-active: #e69f67;
--sidebar-spacer: #45373a;
--scrollbar: var(--sidebar-fg);
--icons: #737480;
--icons-hover: #262625;
--links: #2b79a2;
--inline-code-color: #6e6b5e;
--theme-popup-bg: #e1e1db;
--theme-popup-border: #b38f6b;
--theme-hover: #99908a;
--quote-bg: hsl(60, 5%, 75%);
--quote-border: hsl(60, 5%, 70%);
--table-border-color: hsl(60, 9%, 82%);
--table-header-bg: #b3a497;
--table-alternate-bg: hsl(60, 9%, 84%);
--searchbar-border-color: #aaa;
--searchbar-bg: #fafafa;
--searchbar-fg: #000;
--searchbar-shadow-color: #aaa;
--searchresults-header-fg: #666;
--searchresults-border-color: #888;
--searchresults-li-bg: #dec2a2;
--search-mark-bg: #e69f67;
}
@media (prefers-color-scheme: dark) {
.light.no-js {
--bg: hsl(200, 7%, 8%);
--fg: #98a3ad;
--sidebar-bg: #292c2f;
--sidebar-fg: #a1adb8;
--sidebar-non-existant: #505254;
--sidebar-active: #3473ad;
--sidebar-spacer: #393939;
--scrollbar: var(--sidebar-fg);
--icons: #43484d;
--icons-hover: #b3c0cc;
--links: #2b79a2;
--inline-code-color: #c5c8c6;
--theme-popup-bg: #141617;
--theme-popup-border: #43484d;
--theme-hover: #1f2124;
--quote-bg: hsl(234, 21%, 18%);
--quote-border: hsl(234, 21%, 23%);
--table-border-color: hsl(200, 7%, 13%);
--table-header-bg: hsl(200, 7%, 28%);
--table-alternate-bg: hsl(200, 7%, 11%);
--searchbar-border-color: #aaa;
--searchbar-bg: #b7b7b7;
--searchbar-fg: #000;
--searchbar-shadow-color: #aaa;
--searchresults-header-fg: #666;
--searchresults-border-color: #98a3ad;
--searchresults-li-bg: #2b2b2f;
--search-mark-bg: #355c7d;
}
}

View File

@ -0,0 +1,29 @@
"use strict";
window.editors = [];
(function(editors) {
if (typeof(ace) === 'undefined' || !ace) {
return;
}
Array.from(document.querySelectorAll('.editable')).forEach(function(editable) {
let display_line_numbers = window.playground_line_numbers || false;
let editor = ace.edit(editable);
editor.setOptions({
highlightActiveLine: false,
showPrintMargin: false,
showLineNumbers: display_line_numbers,
showGutter: display_line_numbers,
maxLines: Infinity,
fontSize: "0.875em" // please adjust the font size of the code in general.css
});
editor.$blockScrolling = Infinity;
editor.getSession().setMode("ace/mode/rust");
editor.originalCode = editor.getValue();
editors.push(editor);
});
})(window.editors);

File diff suppressed because one or more lines are too long

Binary file not shown.

After

Width:  |  Height:  |  Size: 5.5 KiB

View File

@ -0,0 +1,22 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 199.7 184.2">
<style>
@media (prefers-color-scheme: dark) {
svg { fill: white; }
}
</style>
<path d="M189.5,36.8c0.2,2.8,0,5.1-0.6,6.8L153,162c-0.6,2.1-2,3.7-4.2,5c-2.2,1.2-4.4,1.9-6.7,1.9H31.4c-9.6,0-15.3-2.8-17.3-8.4
c-0.8-2.2-0.8-3.9,0.1-5.2c0.9-1.2,2.4-1.8,4.6-1.8H123c7.4,0,12.6-1.4,15.4-4.1s5.7-8.9,8.6-18.4l32.9-108.6
c1.8-5.9,1-11.1-2.2-15.6S169.9,0,164,0H72.7c-1,0-3.1,0.4-6.1,1.1l0.1-0.4C64.5,0.2,62.6,0,61,0.1s-3,0.5-4.3,1.4
c-1.3,0.9-2.4,1.8-3.2,2.8S52,6.5,51.2,8.1c-0.8,1.6-1.4,3-1.9,4.3s-1.1,2.7-1.8,4.2c-0.7,1.5-1.3,2.7-2,3.7c-0.5,0.6-1.2,1.5-2,2.5
s-1.6,2-2.2,2.8s-0.9,1.5-1.1,2.2c-0.2,0.7-0.1,1.8,0.2,3.2c0.3,1.4,0.4,2.4,0.4,3.1c-0.3,3-1.4,6.9-3.3,11.6
c-1.9,4.7-3.6,8.1-5.1,10.1c-0.3,0.4-1.2,1.3-2.6,2.7c-1.4,1.4-2.3,2.6-2.6,3.7c-0.3,0.4-0.3,1.5-0.1,3.4c0.3,1.8,0.4,3.1,0.3,3.8
c-0.3,2.7-1.3,6.3-3,10.8c-1.7,4.5-3.4,8.2-5,11c-0.2,0.5-0.9,1.4-2,2.8c-1.1,1.4-1.8,2.5-2,3.4c-0.2,0.6-0.1,1.8,0.1,3.4
c0.2,1.6,0.2,2.8-0.1,3.6c-0.6,3-1.8,6.7-3.6,11c-1.8,4.3-3.6,7.9-5.4,11c-0.5,0.8-1.1,1.7-2,2.8c-0.8,1.1-1.5,2-2,2.8
s-0.8,1.6-1,2.5c-0.1,0.5,0,1.3,0.4,2.3c0.3,1.1,0.4,1.9,0.4,2.6c-0.1,1.1-0.2,2.6-0.5,4.4c-0.2,1.8-0.4,2.9-0.4,3.2
c-1.8,4.8-1.7,9.9,0.2,15.2c2.2,6.2,6.2,11.5,11.9,15.8c5.7,4.3,11.7,6.4,17.8,6.4h110.7c5.2,0,10.1-1.7,14.7-5.2s7.7-7.8,9.2-12.9
l33-108.6c1.8-5.8,1-10.9-2.2-15.5C194.9,39.7,192.6,38,189.5,36.8z M59.6,122.8L73.8,80c0,0,7,0,10.8,0s28.8-1.7,25.4,17.5
c-3.4,19.2-18.8,25.2-36.8,25.4S59.6,122.8,59.6,122.8z M78.6,116.8c4.7-0.1,18.9-2.9,22.1-17.1S89.2,86.3,89.2,86.3l-8.9,0
l-10.2,30.5C70.2,116.9,74,116.9,78.6,116.8z M75.3,68.7L89,26.2h9.8l0.8,34l23.6-34h9.9l-13.6,42.5h-7.1l12.5-35.4l-24.5,35.4h-6.8
l-0.8-35L82,68.7H75.3z"/>
</svg>
<!-- Original image Copyright Dave Gandy — CC BY 4.0 License -->

After

Width:  |  Height:  |  Size: 1.8 KiB

View File

@ -0,0 +1,238 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>Features - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item "><a href="/docserver/index.html"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item expanded "><a href="/docserver/features.html" class="active"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item "><a href="/docserver/howto.html"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item "><a href="/docserver/auths.html"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item "><a href="/docserver/adminusers.html"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/cookie.html"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item "><a href="/docserver/configuration.html"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h1 id="features"><a class="header" href="#features">Features</a></h1>
<ol>
<li>
<p>Server severals <a href="static_content.html">static contents</a> paths from one web server instance</p>
</li>
<li>
<p><a href="protect_access.html">Protect access content</a> with <a href="auths.html">users authentication and authorizations</a></p>
</li>
<li>
<p>Create keys and <a href="sessions.html">sessions</a> to content access with <a href="pasetoken.html">asymmetric encrypted</a></p>
</li>
<li>
<p>Handle and stored <a href="sessions.html">secure sessions</a> and exchange it via <a href="cookies.html">cookies</a></p>
</li>
<li>
<p>Provides <a href="users.html">users management</a> creation, identification and admin with password and <a href="https://en.wikipedia.org/wiki/Multi-factor_authentication">2FA</a> with optional mail service for notifications</p>
</li>
<li>
<p>Some requests responses are created by a <a href="https://github.com/Keats/tera">template engine</a>, so can be customized to some use cases.</p>
</li>
<li>
<p>Can <a href="trace.html">trace access</a> to <a href="logs.html">logs</a> storages</p>
</li>
<li>
<p><a href="configuration.html">Configurable service</a> with <a href="settings.html">settings</a> via <a href="https://en.wikipedia.org/wiki/TOML">TOML</a> file</p>
</li>
</ol>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="index.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="howto.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="index.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="howto.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

View File

@ -0,0 +1,202 @@
Apache License
Version 2.0, January 2004
http://www.apache.org/licenses/
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
1. Definitions.
"License" shall mean the terms and conditions for use, reproduction,
and distribution as defined by Sections 1 through 9 of this document.
"Licensor" shall mean the copyright owner or entity authorized by
the copyright owner that is granting the License.
"Legal Entity" shall mean the union of the acting entity and all
other entities that control, are controlled by, or are under common
control with that entity. For the purposes of this definition,
"control" means (i) the power, direct or indirect, to cause the
direction or management of such entity, whether by contract or
otherwise, or (ii) ownership of fifty percent (50%) or more of the
outstanding shares, or (iii) beneficial ownership of such entity.
"You" (or "Your") shall mean an individual or Legal Entity
exercising permissions granted by this License.
"Source" form shall mean the preferred form for making modifications,
including but not limited to software source code, documentation
source, and configuration files.
"Object" form shall mean any form resulting from mechanical
transformation or translation of a Source form, including but
not limited to compiled object code, generated documentation,
and conversions to other media types.
"Work" shall mean the work of authorship, whether in Source or
Object form, made available under the License, as indicated by a
copyright notice that is included in or attached to the work
(an example is provided in the Appendix below).
"Derivative Works" shall mean any work, whether in Source or Object
form, that is based on (or derived from) the Work and for which the
editorial revisions, annotations, elaborations, or other modifications
represent, as a whole, an original work of authorship. For the purposes
of this License, Derivative Works shall not include works that remain
separable from, or merely link (or bind by name) to the interfaces of,
the Work and Derivative Works thereof.
"Contribution" shall mean any work of authorship, including
the original version of the Work and any modifications or additions
to that Work or Derivative Works thereof, that is intentionally
submitted to Licensor for inclusion in the Work by the copyright owner
or by an individual or Legal Entity authorized to submit on behalf of
the copyright owner. For the purposes of this definition, "submitted"
means any form of electronic, verbal, or written communication sent
to the Licensor or its representatives, including but not limited to
communication on electronic mailing lists, source code control systems,
and issue tracking systems that are managed by, or on behalf of, the
Licensor for the purpose of discussing and improving the Work, but
excluding communication that is conspicuously marked or otherwise
designated in writing by the copyright owner as "Not a Contribution."
"Contributor" shall mean Licensor and any individual or Legal Entity
on behalf of whom a Contribution has been received by Licensor and
subsequently incorporated within the Work.
2. Grant of Copyright License. Subject to the terms and conditions of
this License, each Contributor hereby grants to You a perpetual,
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
copyright license to reproduce, prepare Derivative Works of,
publicly display, publicly perform, sublicense, and distribute the
Work and such Derivative Works in Source or Object form.
3. Grant of Patent License. Subject to the terms and conditions of
this License, each Contributor hereby grants to You a perpetual,
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
(except as stated in this section) patent license to make, have made,
use, offer to sell, sell, import, and otherwise transfer the Work,
where such license applies only to those patent claims licensable
by such Contributor that are necessarily infringed by their
Contribution(s) alone or by combination of their Contribution(s)
with the Work to which such Contribution(s) was submitted. If You
institute patent litigation against any entity (including a
cross-claim or counterclaim in a lawsuit) alleging that the Work
or a Contribution incorporated within the Work constitutes direct
or contributory patent infringement, then any patent licenses
granted to You under this License for that Work shall terminate
as of the date such litigation is filed.
4. Redistribution. You may reproduce and distribute copies of the
Work or Derivative Works thereof in any medium, with or without
modifications, and in Source or Object form, provided that You
meet the following conditions:
(a) You must give any other recipients of the Work or
Derivative Works a copy of this License; and
(b) You must cause any modified files to carry prominent notices
stating that You changed the files; and
(c) You must retain, in the Source form of any Derivative Works
that You distribute, all copyright, patent, trademark, and
attribution notices from the Source form of the Work,
excluding those notices that do not pertain to any part of
the Derivative Works; and
(d) If the Work includes a "NOTICE" text file as part of its
distribution, then any Derivative Works that You distribute must
include a readable copy of the attribution notices contained
within such NOTICE file, excluding those notices that do not
pertain to any part of the Derivative Works, in at least one
of the following places: within a NOTICE text file distributed
as part of the Derivative Works; within the Source form or
documentation, if provided along with the Derivative Works; or,
within a display generated by the Derivative Works, if and
wherever such third-party notices normally appear. The contents
of the NOTICE file are for informational purposes only and
do not modify the License. You may add Your own attribution
notices within Derivative Works that You distribute, alongside
or as an addendum to the NOTICE text from the Work, provided
that such additional attribution notices cannot be construed
as modifying the License.
You may add Your own copyright statement to Your modifications and
may provide additional or different license terms and conditions
for use, reproduction, or distribution of Your modifications, or
for any such Derivative Works as a whole, provided Your use,
reproduction, and distribution of the Work otherwise complies with
the conditions stated in this License.
5. Submission of Contributions. Unless You explicitly state otherwise,
any Contribution intentionally submitted for inclusion in the Work
by You to the Licensor shall be under the terms and conditions of
this License, without any additional terms or conditions.
Notwithstanding the above, nothing herein shall supersede or modify
the terms of any separate license agreement you may have executed
with Licensor regarding such Contributions.
6. Trademarks. This License does not grant permission to use the trade
names, trademarks, service marks, or product names of the Licensor,
except as required for reasonable and customary use in describing the
origin of the Work and reproducing the content of the NOTICE file.
7. Disclaimer of Warranty. Unless required by applicable law or
agreed to in writing, Licensor provides the Work (and each
Contributor provides its Contributions) on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
implied, including, without limitation, any warranties or conditions
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
PARTICULAR PURPOSE. You are solely responsible for determining the
appropriateness of using or redistributing the Work and assume any
risks associated with Your exercise of permissions under this License.
8. Limitation of Liability. In no event and under no legal theory,
whether in tort (including negligence), contract, or otherwise,
unless required by applicable law (such as deliberate and grossly
negligent acts) or agreed to in writing, shall any Contributor be
liable to You for damages, including any direct, indirect, special,
incidental, or consequential damages of any character arising as a
result of this License or out of the use or inability to use the
Work (including but not limited to damages for loss of goodwill,
work stoppage, computer failure or malfunction, or any and all
other commercial damages or losses), even if such Contributor
has been advised of the possibility of such damages.
9. Accepting Warranty or Additional Liability. While redistributing
the Work or Derivative Works thereof, You may choose to offer,
and charge a fee for, acceptance of support, warranty, indemnity,
or other liability obligations and/or rights consistent with this
License. However, in accepting such obligations, You may act only
on Your own behalf and on Your sole responsibility, not on behalf
of any other Contributor, and only if You agree to indemnify,
defend, and hold each Contributor harmless for any liability
incurred by, or claims asserted against, such Contributor by reason
of your accepting any such warranty or additional liability.
END OF TERMS AND CONDITIONS
APPENDIX: How to apply the Apache License to your work.
To apply the Apache License to your work, attach the following
boilerplate notice, with the fields enclosed by brackets "[]"
replaced with your own identifying information. (Don't include
the brackets!) The text should be enclosed in the appropriate
comment syntax for the file format. We also recommend that a
file or class name and description of purpose be included on the
same "printed page" as the copyright notice for easier
identification within third-party archives.
Copyright [yyyy] [name of copyright owner]
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

View File

@ -0,0 +1,93 @@
Copyright 2010, 2012 Adobe Systems Incorporated (http://www.adobe.com/), with Reserved Font Name 'Source'. All Rights Reserved. Source is a trademark of Adobe Systems Incorporated in the United States and/or other countries.
This Font Software is licensed under the SIL Open Font License, Version 1.1.
This license is copied below, and is also available with a FAQ at:
http://scripts.sil.org/OFL
-----------------------------------------------------------
SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007
-----------------------------------------------------------
PREAMBLE
The goals of the Open Font License (OFL) are to stimulate worldwide
development of collaborative font projects, to support the font creation
efforts of academic and linguistic communities, and to provide a free and
open framework in which fonts may be shared and improved in partnership
with others.
The OFL allows the licensed fonts to be used, studied, modified and
redistributed freely as long as they are not sold by themselves. The
fonts, including any derivative works, can be bundled, embedded,
redistributed and/or sold with any software provided that any reserved
names are not used by derivative works. The fonts and derivatives,
however, cannot be released under any other type of license. The
requirement for fonts to remain under this license does not apply
to any document created using the fonts or their derivatives.
DEFINITIONS
"Font Software" refers to the set of files released by the Copyright
Holder(s) under this license and clearly marked as such. This may
include source files, build scripts and documentation.
"Reserved Font Name" refers to any names specified as such after the
copyright statement(s).
"Original Version" refers to the collection of Font Software components as
distributed by the Copyright Holder(s).
"Modified Version" refers to any derivative made by adding to, deleting,
or substituting -- in part or in whole -- any of the components of the
Original Version, by changing formats or by porting the Font Software to a
new environment.
"Author" refers to any designer, engineer, programmer, technical
writer or other person who contributed to the Font Software.
PERMISSION & CONDITIONS
Permission is hereby granted, free of charge, to any person obtaining
a copy of the Font Software, to use, study, copy, merge, embed, modify,
redistribute, and sell modified and unmodified copies of the Font
Software, subject to the following conditions:
1) Neither the Font Software nor any of its individual components,
in Original or Modified Versions, may be sold by itself.
2) Original or Modified Versions of the Font Software may be bundled,
redistributed and/or sold with any software, provided that each copy
contains the above copyright notice and this license. These can be
included either as stand-alone text files, human-readable headers or
in the appropriate machine-readable metadata fields within text or
binary files as long as those fields can be easily viewed by the user.
3) No Modified Version of the Font Software may use the Reserved Font
Name(s) unless explicit written permission is granted by the corresponding
Copyright Holder. This restriction only applies to the primary font name as
presented to the users.
4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font
Software shall not be used to promote, endorse or advertise any
Modified Version, except to acknowledge the contribution(s) of the
Copyright Holder(s) and the Author(s) or with their explicit written
permission.
5) The Font Software, modified or unmodified, in part or in whole,
must be distributed entirely under this license, and must not be
distributed under any other license. The requirement for fonts to
remain under this license does not apply to any document created
using the Font Software.
TERMINATION
This license becomes null and void if any of the above conditions are
not met.
DISCLAIMER
THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT
OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE
COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL
DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM
OTHER DEALINGS IN THE FONT SOFTWARE.

View File

@ -0,0 +1,100 @@
/* Open Sans is licensed under the Apache License, Version 2.0. See http://www.apache.org/licenses/LICENSE-2.0 */
/* Source Code Pro is under the Open Font License. See https://scripts.sil.org/cms/scripts/page.php?site_id=nrsi&id=OFL */
/* open-sans-300 - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: normal;
font-weight: 300;
src: local('Open Sans Light'), local('OpenSans-Light'),
url('open-sans-v17-all-charsets-300.woff2') format('woff2');
}
/* open-sans-300italic - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: italic;
font-weight: 300;
src: local('Open Sans Light Italic'), local('OpenSans-LightItalic'),
url('open-sans-v17-all-charsets-300italic.woff2') format('woff2');
}
/* open-sans-regular - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: normal;
font-weight: 400;
src: local('Open Sans Regular'), local('OpenSans-Regular'),
url('open-sans-v17-all-charsets-regular.woff2') format('woff2');
}
/* open-sans-italic - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: italic;
font-weight: 400;
src: local('Open Sans Italic'), local('OpenSans-Italic'),
url('open-sans-v17-all-charsets-italic.woff2') format('woff2');
}
/* open-sans-600 - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: normal;
font-weight: 600;
src: local('Open Sans SemiBold'), local('OpenSans-SemiBold'),
url('open-sans-v17-all-charsets-600.woff2') format('woff2');
}
/* open-sans-600italic - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: italic;
font-weight: 600;
src: local('Open Sans SemiBold Italic'), local('OpenSans-SemiBoldItalic'),
url('open-sans-v17-all-charsets-600italic.woff2') format('woff2');
}
/* open-sans-700 - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: normal;
font-weight: 700;
src: local('Open Sans Bold'), local('OpenSans-Bold'),
url('open-sans-v17-all-charsets-700.woff2') format('woff2');
}
/* open-sans-700italic - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: italic;
font-weight: 700;
src: local('Open Sans Bold Italic'), local('OpenSans-BoldItalic'),
url('open-sans-v17-all-charsets-700italic.woff2') format('woff2');
}
/* open-sans-800 - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: normal;
font-weight: 800;
src: local('Open Sans ExtraBold'), local('OpenSans-ExtraBold'),
url('open-sans-v17-all-charsets-800.woff2') format('woff2');
}
/* open-sans-800italic - latin_vietnamese_latin-ext_greek-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Open Sans';
font-style: italic;
font-weight: 800;
src: local('Open Sans ExtraBold Italic'), local('OpenSans-ExtraBoldItalic'),
url('open-sans-v17-all-charsets-800italic.woff2') format('woff2');
}
/* source-code-pro-500 - latin_vietnamese_latin-ext_greek_cyrillic-ext_cyrillic */
@font-face {
font-family: 'Source Code Pro';
font-style: normal;
font-weight: 500;
src: url('source-code-pro-v11-all-charsets-500.woff2') format('woff2');
}

View File

@ -0,0 +1,82 @@
/*
* An increased contrast highlighting scheme loosely based on the
* "Base16 Atelier Dune Light" theme by Bram de Haan
* (http://atelierbram.github.io/syntax-highlighting/atelier-schemes/dune)
* Original Base16 color scheme by Chris Kempson
* (https://github.com/chriskempson/base16)
*/
/* Comment */
.hljs-comment,
.hljs-quote {
color: #575757;
}
/* Red */
.hljs-variable,
.hljs-template-variable,
.hljs-attribute,
.hljs-tag,
.hljs-name,
.hljs-regexp,
.hljs-link,
.hljs-name,
.hljs-selector-id,
.hljs-selector-class {
color: #d70025;
}
/* Orange */
.hljs-number,
.hljs-meta,
.hljs-built_in,
.hljs-builtin-name,
.hljs-literal,
.hljs-type,
.hljs-params {
color: #b21e00;
}
/* Green */
.hljs-string,
.hljs-symbol,
.hljs-bullet {
color: #008200;
}
/* Blue */
.hljs-title,
.hljs-section {
color: #0030f2;
}
/* Purple */
.hljs-keyword,
.hljs-selector-tag {
color: #9d00ec;
}
.hljs {
display: block;
overflow-x: auto;
background: #f6f7f6;
color: #000;
}
.hljs-emphasis {
font-style: italic;
}
.hljs-strong {
font-weight: bold;
}
.hljs-addition {
color: #22863a;
background-color: #f0fff4;
}
.hljs-deletion {
color: #b31d28;
background-color: #ffeef0;
}

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,212 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>How to - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item "><a href="/docserver/index.html"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item "><a href="/docserver/features.html"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item expanded "><a href="/docserver/howto.html" class="active"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item "><a href="/docserver/auths.html"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item "><a href="/docserver/adminusers.html"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/cookie.html"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item "><a href="/docserver/configuration.html"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<h1 id="how-to"><a class="header" href="#how-to">How to</a></h1>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="features.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="quick-start.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="features.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="quick-start.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

Binary file not shown.

After

Width:  |  Height:  |  Size: 710 KiB

View File

@ -0,0 +1,38 @@
<?xml version="1.0" encoding="utf-8"?>
<!-- Generator: Adobe Illustrator 25.4.1, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
<svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
viewBox="0 0 100 100" style="enable-background:new 0 0 100 100;" xml:space="preserve">
<style type="text/css">
.st0{fill:#0061A1;}
</style>
<path d="M63.4,73.1c-0.7,0-1.2,0.5-1.2,1.2v12.2H13.5v-56h15.8c0.7,0,1.2-0.5,1.2-1.2V13.5h31.7v7.3c0,0.7,0.5,1.2,1.2,1.2
s1.2-0.5,1.2-1.2v-8.5c0-0.7-0.5-1.2-1.2-1.2H29.5c-0.4,0-0.8,0.1-1.1,0.4L11.4,28.4C11.2,28.7,11,29,11,29.3v58.5
c0,0.7,0.5,1.2,1.2,1.2h51.2c0.7,0,1.2-0.5,1.2-1.2V74.4C64.6,73.7,64.1,73.1,63.4,73.1z M28.1,15.2v12.9H15.2L28.1,15.2z"/>
<path class="st0" d="M85.3,24.4H51.2c-2,0-3.7,1.6-3.7,3.7v1.4c0,4-2.1,7.7-5.5,9.7c-0.6,0.4-0.7,1.1-0.3,1.7
c0.2,0.3,0.6,0.5,0.9,0.5h42.6c2,0,3.7-1.6,3.7-3.7v-9.7C89,26.1,87.3,24.4,85.3,24.4z M86.5,37.8c0,0.7-0.5,1.2-1.2,1.2H46.2
c2.5-2.6,3.8-6,3.8-9.5v-1.4c0-0.7,0.5-1.2,1.2-1.2h34.1c0.7,0,1.2,0.5,1.2,1.2V37.8z"/>
<path class="st0" d="M63.4,31.7h-7.3c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h7.3c0.7,0,1.2-0.5,1.2-1.2S64.1,31.7,63.4,31.7z"/>
<path class="st0" d="M80.5,31.7H68.3c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h12.2c0.7,0,1.2-0.5,1.2-1.2S81.1,31.7,80.5,31.7z"/>
<path class="st0" d="M85.3,46.3H51.2c-2,0-3.7,1.6-3.7,3.7v8.7c0,4-2.1,7.7-5.5,9.7c-0.6,0.4-0.7,1.1-0.3,1.7
c0.2,0.3,0.6,0.5,0.9,0.5h42.6c2,0,3.7-1.6,3.7-3.7V50C89,48,87.3,46.3,85.3,46.3z M86.5,67.1c0,0.7-0.5,1.2-1.2,1.2H46.2
c2.5-2.6,3.8-6,3.8-9.5V50c0-0.7,0.5-1.2,1.2-1.2h34.1c0.7,0,1.2,0.5,1.2,1.2V67.1z"/>
<path class="st0" d="M56.1,56.1H61c0.7,0,1.2-0.5,1.2-1.2s-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2S55.4,56.1,56.1,56.1z"/>
<path class="st0" d="M80.5,53.7H65.8c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h14.6c0.7,0,1.2-0.5,1.2-1.2S81.1,53.7,80.5,53.7z"/>
<path class="st0" d="M65.8,62.2h-9.7c-0.7,0-1.2,0.5-1.2,1.2c0,0.7,0.5,1.2,1.2,1.2h9.7c0.7,0,1.2-0.5,1.2-1.2
C67.1,62.7,66.5,62.2,65.8,62.2z"/>
<path class="st0" d="M80.5,62.2h-9.7c-0.7,0-1.2,0.5-1.2,1.2c0,0.7,0.5,1.2,1.2,1.2h9.7c0.7,0,1.2-0.5,1.2-1.2
C81.7,62.7,81.1,62.2,80.5,62.2z"/>
<path d="M23.2,35.4c-2.7,0-4.9,2.2-4.9,4.9c0,2.7,2.2,4.9,4.9,4.9s4.9-2.2,4.9-4.9C28.1,37.6,25.9,35.4,23.2,35.4z M23.2,42.7
c-1.3,0-2.4-1.1-2.4-2.4c0-1.3,1.1-2.4,2.4-2.4c1.3,0,2.4,1.1,2.4,2.4C25.6,41.6,24.5,42.7,23.2,42.7z"/>
<path d="M23.2,64.6c-2.7,0-4.9,2.2-4.9,4.9c0,2.7,2.2,4.9,4.9,4.9s4.9-2.2,4.9-4.9C28.1,66.8,25.9,64.6,23.2,64.6z M23.2,71.9
c-1.3,0-2.4-1.1-2.4-2.4s1.1-2.4,2.4-2.4c1.3,0,2.4,1.1,2.4,2.4S24.5,71.9,23.2,71.9z"/>
<path d="M31.7,37.8h4.9c0.7,0,1.2-0.5,1.2-1.2s-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2S31.1,37.8,31.7,37.8z"/>
<path d="M31.7,45.1h4.9c0.7,0,1.2-0.5,1.2-1.2s-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2S31.1,45.1,31.7,45.1z"/>
<path d="M42.7,51.2c0-0.7-0.5-1.2-1.2-1.2h-9.7c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h9.7C42.1,52.4,42.7,51.9,42.7,51.2z"/>
<path d="M31.7,67.1h4.9c0.7,0,1.2-0.5,1.2-1.2c0-0.7-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2C30.5,66.5,31.1,67.1,31.7,67.1z"
/>
<path d="M31.7,74.4h4.9c0.7,0,1.2-0.5,1.2-1.2c0-0.7-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2C30.5,73.8,31.1,74.4,31.7,74.4z"
/>
<path d="M31.7,81.7h24.4c0.7,0,1.2-0.5,1.2-1.2c0-0.7-0.5-1.2-1.2-1.2H31.7c-0.7,0-1.2,0.5-1.2,1.2C30.5,81.1,31.1,81.7,31.7,81.7z"
/>
</svg>

After

Width:  |  Height:  |  Size: 3.3 KiB

File diff suppressed because one or more lines are too long

After

Width:  |  Height:  |  Size: 8.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 816 KiB

View File

@ -0,0 +1,38 @@
<?xml version="1.0" encoding="utf-8"?>
<!-- Generator: Adobe Illustrator 25.4.1, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
<svg version="1.1" id="Capa_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
viewBox="0 0 100 100" style="enable-background:new 0 0 100 100;" xml:space="preserve">
<style type="text/css">
.st0{fill:#0061A1;}
</style>
<path d="M63.4,73.1c-0.7,0-1.2,0.5-1.2,1.2v12.2H13.5v-56h15.8c0.7,0,1.2-0.5,1.2-1.2V13.5h31.7v7.3c0,0.7,0.5,1.2,1.2,1.2
s1.2-0.5,1.2-1.2v-8.5c0-0.7-0.5-1.2-1.2-1.2H29.5c-0.4,0-0.8,0.1-1.1,0.4L11.4,28.4C11.2,28.7,11,29,11,29.3v58.5
c0,0.7,0.5,1.2,1.2,1.2h51.2c0.7,0,1.2-0.5,1.2-1.2V74.4C64.6,73.7,64.1,73.1,63.4,73.1z M28.1,15.2v12.9H15.2L28.1,15.2z"/>
<path class="st0" d="M85.3,24.4H51.2c-2,0-3.7,1.6-3.7,3.7v1.4c0,4-2.1,7.7-5.5,9.7c-0.6,0.4-0.7,1.1-0.3,1.7
c0.2,0.3,0.6,0.5,0.9,0.5h42.6c2,0,3.7-1.6,3.7-3.7v-9.7C89,26.1,87.3,24.4,85.3,24.4z M86.5,37.8c0,0.7-0.5,1.2-1.2,1.2H46.2
c2.5-2.6,3.8-6,3.8-9.5v-1.4c0-0.7,0.5-1.2,1.2-1.2h34.1c0.7,0,1.2,0.5,1.2,1.2V37.8z"/>
<path class="st0" d="M63.4,31.7h-7.3c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h7.3c0.7,0,1.2-0.5,1.2-1.2S64.1,31.7,63.4,31.7z"/>
<path class="st0" d="M80.5,31.7H68.3c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h12.2c0.7,0,1.2-0.5,1.2-1.2S81.1,31.7,80.5,31.7z"/>
<path class="st0" d="M85.3,46.3H51.2c-2,0-3.7,1.6-3.7,3.7v8.7c0,4-2.1,7.7-5.5,9.7c-0.6,0.4-0.7,1.1-0.3,1.7
c0.2,0.3,0.6,0.5,0.9,0.5h42.6c2,0,3.7-1.6,3.7-3.7V50C89,48,87.3,46.3,85.3,46.3z M86.5,67.1c0,0.7-0.5,1.2-1.2,1.2H46.2
c2.5-2.6,3.8-6,3.8-9.5V50c0-0.7,0.5-1.2,1.2-1.2h34.1c0.7,0,1.2,0.5,1.2,1.2V67.1z"/>
<path class="st0" d="M56.1,56.1H61c0.7,0,1.2-0.5,1.2-1.2s-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2S55.4,56.1,56.1,56.1z"/>
<path class="st0" d="M80.5,53.7H65.8c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h14.6c0.7,0,1.2-0.5,1.2-1.2S81.1,53.7,80.5,53.7z"/>
<path class="st0" d="M65.8,62.2h-9.7c-0.7,0-1.2,0.5-1.2,1.2c0,0.7,0.5,1.2,1.2,1.2h9.7c0.7,0,1.2-0.5,1.2-1.2
C67.1,62.7,66.5,62.2,65.8,62.2z"/>
<path class="st0" d="M80.5,62.2h-9.7c-0.7,0-1.2,0.5-1.2,1.2c0,0.7,0.5,1.2,1.2,1.2h9.7c0.7,0,1.2-0.5,1.2-1.2
C81.7,62.7,81.1,62.2,80.5,62.2z"/>
<path d="M23.2,35.4c-2.7,0-4.9,2.2-4.9,4.9c0,2.7,2.2,4.9,4.9,4.9s4.9-2.2,4.9-4.9C28.1,37.6,25.9,35.4,23.2,35.4z M23.2,42.7
c-1.3,0-2.4-1.1-2.4-2.4c0-1.3,1.1-2.4,2.4-2.4c1.3,0,2.4,1.1,2.4,2.4C25.6,41.6,24.5,42.7,23.2,42.7z"/>
<path d="M23.2,64.6c-2.7,0-4.9,2.2-4.9,4.9c0,2.7,2.2,4.9,4.9,4.9s4.9-2.2,4.9-4.9C28.1,66.8,25.9,64.6,23.2,64.6z M23.2,71.9
c-1.3,0-2.4-1.1-2.4-2.4s1.1-2.4,2.4-2.4c1.3,0,2.4,1.1,2.4,2.4S24.5,71.9,23.2,71.9z"/>
<path d="M31.7,37.8h4.9c0.7,0,1.2-0.5,1.2-1.2s-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2S31.1,37.8,31.7,37.8z"/>
<path d="M31.7,45.1h4.9c0.7,0,1.2-0.5,1.2-1.2s-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2S31.1,45.1,31.7,45.1z"/>
<path d="M42.7,51.2c0-0.7-0.5-1.2-1.2-1.2h-9.7c-0.7,0-1.2,0.5-1.2,1.2s0.5,1.2,1.2,1.2h9.7C42.1,52.4,42.7,51.9,42.7,51.2z"/>
<path d="M31.7,67.1h4.9c0.7,0,1.2-0.5,1.2-1.2c0-0.7-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2C30.5,66.5,31.1,67.1,31.7,67.1z"
/>
<path d="M31.7,74.4h4.9c0.7,0,1.2-0.5,1.2-1.2c0-0.7-0.5-1.2-1.2-1.2h-4.9c-0.7,0-1.2,0.5-1.2,1.2C30.5,73.8,31.1,74.4,31.7,74.4z"
/>
<path d="M31.7,81.7h24.4c0.7,0,1.2-0.5,1.2-1.2c0-0.7-0.5-1.2-1.2-1.2H31.7c-0.7,0-1.2,0.5-1.2,1.2C30.5,81.1,31.1,81.7,31.7,81.7z"
/>
</svg>

After

Width:  |  Height:  |  Size: 3.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 728 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 848 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 944 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 852 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 948 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 835 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 367 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 433 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 912 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 1003 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 932 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 847 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 943 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 841 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 938 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 949 KiB

View File

@ -0,0 +1,229 @@
<!DOCTYPE HTML>
<html lang="en" class="sidebar-visible no-js light">
<head>
<!-- Book generated using mdBook -->
<meta charset="UTF-8">
<title>What is this ? - Doc Server</title>
<base href="/docserver/">
<!-- Custom HTML head -->
<meta name="description" content="Doc Server">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="theme-color" content="#ffffff" />
<link rel="icon" href="/docserver/favicon.svg">
<link rel="shortcut icon" href="/docserver/favicon.png">
<link rel="stylesheet" href="/docserver/css/variables.css">
<link rel="stylesheet" href="/docserver/css/general.css">
<link rel="stylesheet" href="/docserver/css/chrome.css">
<link rel="stylesheet" href="/docserver/css/print.css" media="print">
<!-- Fonts -->
<link rel="stylesheet" href="/docserver/FontAwesome/css/font-awesome.css">
<link rel="stylesheet" href="/docserver/fonts/fonts.css">
<!-- Highlight.js Stylesheets -->
<link rel="stylesheet" href="/docserver/highlight.css">
<link rel="stylesheet" href="/docserver/tomorrow-night.css">
<link rel="stylesheet" href="/docserver/ayu-highlight.css">
<!-- Custom theme stylesheets -->
<link rel="stylesheet" href="/docserver/assets/css/custom.css">
<link rel="stylesheet" href="/docserver/assets/css/mdbook-admonish.css">
</head>
<body>
<div id="body-container">
<!-- Provide site root to javascript -->
<script>
var path_to_root = "/docserver/";
var default_theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "navy" : "light";
</script>
<!-- Work around some values being stored in localStorage wrapped in quotes -->
<script>
try {
var theme = localStorage.getItem('mdbook-theme');
var sidebar = localStorage.getItem('mdbook-sidebar');
if (theme.startsWith('"') && theme.endsWith('"')) {
localStorage.setItem('mdbook-theme', theme.slice(1, theme.length - 1));
}
if (sidebar.startsWith('"') && sidebar.endsWith('"')) {
localStorage.setItem('mdbook-sidebar', sidebar.slice(1, sidebar.length - 1));
}
} catch (e) { }
</script>
<!-- Set the theme before any content is loaded, prevents flash -->
<script>
var theme;
try { theme = localStorage.getItem('mdbook-theme'); } catch(e) { }
if (theme === null || theme === undefined) { theme = default_theme; }
var html = document.querySelector('html');
html.classList.remove('no-js')
html.classList.remove('light')
html.classList.add(theme);
html.classList.add('js');
</script>
<!-- Hide / unhide sidebar before it is displayed -->
<script>
var html = document.querySelector('html');
var sidebar = null;
if (document.body.clientWidth >= 1080) {
try { sidebar = localStorage.getItem('mdbook-sidebar'); } catch(e) { }
sidebar = sidebar || 'visible';
} else {
sidebar = 'hidden';
}
html.classList.remove('sidebar-visible');
html.classList.add("sidebar-" + sidebar);
</script>
<nav id="sidebar" class="sidebar" aria-label="Table of contents">
<div class="sidebar-scrollbox">
<ol class="chapter"><li class="chapter-item "><a href="/docserver/introduction.html"><strong aria-hidden="true">1.</strong> Introduction</a></li><li class="chapter-item expanded "><a href="/docserver/index.html" class="active"><strong aria-hidden="true">2.</strong> What is this ?</a></li><li class="chapter-item "><a href="/docserver/features.html"><strong aria-hidden="true">3.</strong> Features</a></li><li class="chapter-item "><a href="/docserver/howto.html"><strong aria-hidden="true">4.</strong> How to</a></li><li class="chapter-item "><a href="/docserver/quick-start.html"><strong aria-hidden="true">5.</strong> Quick Start</a></li><li class="spacer"></li><li class="chapter-item affix "><li class="part-title">Server and Content</li><li class="chapter-item "><a href="/docserver/one_server.html"><strong aria-hidden="true">6.</strong> One web server instance</a></li><li class="chapter-item "><a href="/docserver/server_modes.html"><strong aria-hidden="true">7.</strong> Server modes</a></li><li class="chapter-item "><a href="/docserver/static_content.html"><strong aria-hidden="true">8.</strong> Static contents</a></li><li class="chapter-item "><a href="/docserver/url_path.html"><strong aria-hidden="true">9.</strong> url_path prefix</a></li><li class="chapter-item "><a href="/docserver/protect_access.html"><strong aria-hidden="true">10.</strong> Protect access</a></li><li class="chapter-item affix "><li class="part-title">Users and Policies</li><li class="chapter-item "><a href="/docserver/users.html"><strong aria-hidden="true">11.</strong> Users</a></li><li class="chapter-item "><a href="/docserver/users_sql.html"><strong aria-hidden="true">12.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/users_file_storage.html"><strong aria-hidden="true">13.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/roles.html"><strong aria-hidden="true">14.</strong> Roles</a></li><li class="chapter-item "><a href="/docserver/auths.html"><strong aria-hidden="true">15.</strong> Users auths</a></li><li class="chapter-item "><a href="/docserver/policies.html"><strong aria-hidden="true">16.</strong> Access policies</a></li><li class="chapter-item "><a href="/docserver/adminusers.html"><strong aria-hidden="true">17.</strong> Users Admin</a></li><li class="chapter-item affix "><li class="part-title">Sessions</li><li class="chapter-item "><a href="/docserver/sessions.html"><strong aria-hidden="true">18.</strong> Session data</a></li><li class="chapter-item "><a href="/docserver/pasetoken.html"><strong aria-hidden="true">19.</strong> Pasetoken</a></li><li class="chapter-item "><a href="/docserver/sessions_sql.html"><strong aria-hidden="true">20.</strong> SQL storage</a></li><li class="chapter-item "><a href="/docserver/sessions_file_storage.html"><strong aria-hidden="true">21.</strong> File storage</a></li><li class="chapter-item "><a href="/docserver/cookie.html"><strong aria-hidden="true">22.</strong> Cookie set</a></li><li class="chapter-item affix "><li class="part-title">Logs and Trace</li><li class="chapter-item "><a href="/docserver/trace.html"><strong aria-hidden="true">23.</strong> Traced</a></li><li class="chapter-item "><a href="/docserver/logs.html"><strong aria-hidden="true">24.</strong> Sessions logs</a></li><li class="chapter-item affix "><li class="part-title">Configuration</li><li class="chapter-item "><a href="/docserver/configuration.html"><strong aria-hidden="true">25.</strong> Configurable service</a></li><li class="chapter-item "><a href="/docserver/settings.html"><strong aria-hidden="true">26.</strong> Config settings</a></li><li class="spacer"></li></ol>
</div>
<div id="sidebar-resize-handle" class="sidebar-resize-handle"></div>
</nav>
<div id="page-wrapper" class="page-wrapper">
<div class="page">
<div id="menu-bar-hover-placeholder"></div>
<div id="menu-bar" class="menu-bar sticky bordered">
<div class="left-buttons">
<button id="sidebar-toggle" class="icon-button" type="button" title="Toggle Table of Contents" aria-label="Toggle Table of Contents" aria-controls="sidebar">
<i class="fa fa-bars"></i>
</button>
<button id="sidebar-toggle" onclick="location.href='/'" class="icon-button" type="button" title="Home" aria-label="Home" aria-controls="">
<i class="fa fa-home"></i>
</button>
<button id="theme-toggle" class="icon-button" type="button" title="Change theme" aria-label="Change theme" aria-haspopup="true" aria-expanded="false" aria-controls="theme-list">
<i class="fa fa-paint-brush"></i>
</button>
<ul id="theme-list" class="theme-popup" aria-label="Themes" role="menu">
<li role="none"><button role="menuitem" class="theme" id="light">Light</button></li>
<li role="none"><button role="menuitem" class="theme" id="rust">Rust</button></li>
<li role="none"><button role="menuitem" class="theme" id="coal">Coal</button></li>
<li role="none"><button role="menuitem" class="theme" id="navy">Navy</button></li>
<li role="none"><button role="menuitem" class="theme" id="ayu">Ayu</button></li>
</ul>
<button id="search-toggle" class="icon-button" type="button" title="Search. (Shortkey: s)" aria-label="Toggle Searchbar" aria-expanded="false" aria-keyshortcuts="S" aria-controls="searchbar">
<i class="fa fa-search"></i>
</button>
</div>
<h1 class="menu-title">Doc Server</h1>
<div class="right-buttons">
<a href="/docserver/print.html" title="Print this book" aria-label="Print this book">
<i id="print-button" class="fa fa-print"></i>
</a>
<a href="/logout" title="Sing out" aria-label="Sign out">
<i id="logout-button" class="fa fa-sign-out"></i>
</a>
</div>
</div>
<div id="search-wrapper" class="hidden">
<form id="searchbar-outer" class="searchbar-outer">
<input type="search" id="searchbar" name="searchbar" placeholder="Search this book ..." aria-controls="searchresults-outer" aria-describedby="searchresults-header">
</form>
<div id="searchresults-outer" class="searchresults-outer hidden">
<div id="searchresults-header" class="searchresults-header"></div>
<ul id="searchresults">
</ul>
</div>
</div>
<!-- Apply ARIA attributes after the sidebar and the sidebar toggle button are added to the DOM -->
<script>
document.getElementById('sidebar-toggle').setAttribute('aria-expanded', sidebar === 'visible');
document.getElementById('sidebar').setAttribute('aria-hidden', sidebar !== 'visible');
Array.from(document.querySelectorAll('#sidebar a')).forEach(function(link) {
link.setAttribute('tabIndex', sidebar === 'visible' ? 0 : -1);
});
</script>
<div id="content" class="content">
<main>
<p><a href="/docserver/">Doc Server</a> is a <strong>Web Server</strong> in <a href="https://www.rust-lang.org">Rust</a> for <a href="/docserver/static_content.html">static content</a>
with <strong>Control access</strong> via <a href="/docserver/auths.html">authorizations</a>, fully customizable with <a href="/docserver/settings.html">settings</a>.</p>
<p><svg style="display:none"><symbol id="symbl-check" fill="none" stroke="currentColor" stroke-width="1.5" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg"> <path stroke-linecap="round" stroke-linejoin="round" d="M4.5 12.75l6 6 9-13.5"></path></symbol></svg></p>
<div class="box"><img src="images/docserver.svg" class="logo" /></div>
<h2 id="one-server"><a class="header" href="#one-server">One Server</a></h2>
<p>Most of documentation or <a href="static_content.html">static content</a> is generated to be served from a WebServer using <strong>absoulte</strong>, <strong>relative</strong> <a href="https://www.w3.org/TR/WD-html40-970917/htmlweb.html">URL</a> or a <a href="https://developer.mozilla.org/en-US/docs/Web/HTML/Element/base">document base URL</a>, this is a big issue to handle several at once from one unique server under <a href="https://en.wikipedia.org/wiki/URL">url paths</a> if are not set properly. In many cases there is no need for a dedicated web server for each documentation or <a href="static_content.html">static content</a> that needs to be published or served. Some documentation generators are more or less configurables to use some <a href="https://www.w3.org/TR/WD-html40-970917/htmlweb.html">URL</a> <strong>relative path</strong>, this should include pages and assets (images, css, js, etc.).</p>
<h2 id="protect-access-content"><a class="header" href="#protect-access-content">Protect access content</a></h2>
<p>In many case some <a href="protect_access.html">protect access content</a> mode is mandatory, from private infos to restricted documentation or development docs, whether it is for internal use or new versions not delivered yet.</p>
<p><a href="https://en.wikipedia.org/wiki/Basic_access_authentication">Basic authentication</a> or via one <a href="https://en.wikipedia.org/wiki/JSON_Web_Token">token</a> could be too simple and flat to tailor specific contexts, so a full real <a href="sessions.html">sessions</a> handling should implement <a href="users.html">users</a> (identity entities), <a href="roles.html">roles</a> (groups), <a href="policies.html">policies</a> (permissions) and <a href="pasetoken.html">secure tokens</a> (as data container).</p>
<h2 id="secure-sessions"><a class="header" href="#secure-sessions">Secure sessions</a></h2>
<p><a href="sessions.html">Sessions management</a> to content access with <u>secure storage</u> (some kind of persistence has to be provided among requests) and exchange with clients upon requests. Their data will allow <a href="https://en.wikipedia.org/wiki/Web_server">webserver</a> to allow or deny an <a href="https://www.w3.org/TR/WD-html40-970917/htmlweb.html">URL</a> access request in a <a href="https://en.wikipedia.org/wiki/Middleware">middleware</a> stage.</p>
<h2 id="users"><a class="header" href="#users">Users</a></h2>
<p>At least a simple <a href="users.html">users management</a> creation, identification and admin to link <a href="sessions.html">sessions</a> to an identified entity. Signin access should use secure passwords and <a href="https://en.wikipedia.org/wiki/Multi-factor_authentication">2FA</a> with optional mail service for notifications. Signup modes should be also provided for close or open contexts.</p>
<p>Some requests responses are created by a <a href="https://github.com/Keats/tera">template engine</a>, so can be customized to some use cases.</p>
<h2 id="logging"><a class="header" href="#logging">Logging</a></h2>
<p>Some kind of mechanism for <a href="logs.html">logs</a> storages and <a href="trace.html">trace access</a> to know what is happening and how is it used.</p>
<h2 id="configuration-settings"><a class="header" href="#configuration-settings">Configuration settings</a></h2>
<p><a href="configuration.html">Service configuration</a> with <a href="settings.html">settings</a> via <a href="https://en.wikipedia.org/wiki/TOML">TOML</a> file will help to customize server and services to context and needs.</p>
</main>
<nav class="nav-wrapper" aria-label="Page navigation">
<!-- Mobile navigation buttons -->
<a rel="prev" href="introduction.html" class="mobile-nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="features.html" class="mobile-nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
<div style="clear: both"></div>
</nav>
</div>
</div>
<nav class="nav-wide-wrapper" aria-label="Page navigation">
<a rel="prev" href="introduction.html" class="nav-chapters previous" title="Previous chapter" aria-label="Previous chapter" aria-keyshortcuts="Left">
<i class="fa fa-angle-left"></i>
</a>
<a rel="next" href="features.html" class="nav-chapters next" title="Next chapter" aria-label="Next chapter" aria-keyshortcuts="Right">
<i class="fa fa-angle-right"></i>
</a>
</nav>
</div>
<script>
window.playground_copyable = true;
</script>
<script src="/docserver/ace.js"></script>
<script src="/docserver/editor.js"></script>
<script src="/docserver/mode-rust.js"></script>
<script src="/docserver/theme-dawn.js"></script>
<script src="/docserver/theme-tomorrow_night.js"></script>
<script src="/docserver/elasticlunr.min.js"></script>
<script src="/docserver/mark.min.js"></script>
<script src="/docserver/searcher.js"></script>
<script src="/docserver/clipboard.min.js"></script>
<script src="/docserver/highlight.js"></script>
<script src="/docserver/book.js"></script>
<!-- Custom JS scripts -->
</div>
</body>
</html>

Some files were not shown because too many files have changed in this diff Show More