diff --git a/.drone.yml b/.drone.yml index 7954751..2585169 100644 --- a/.drone.yml +++ b/.drone.yml @@ -1,17 +1,14 @@ --- kind: pipeline -type: exec -name: just a test +type: docker +name: default + +workspace: + base: /go + path: src/dn42regsrv steps: - - name: environment + - name: build + image: golang commands: - - pwd - - env - - df -h - - ls -la - - ls -la ../ - - id - - ps -ef - - + - go get diff --git a/API.md b/API.md new file mode 100644 index 0000000..6e4fc8c --- /dev/null +++ b/API.md @@ -0,0 +1,500 @@ +# dn42regsrv API Description + +## Registry API + +The general form of the registry query API is: + +``` +GET /api/registry/{type}/{object}/{key}/{attribute}?raw +``` + +* Prefixing with a '*' performs a case insensitive, substring match +* A '*' on its own means match everything +* Otherwise an exact, case sensitive match is performed + +By default, results are returned as JSON objects, and the registry data is decorated +with markdown style links depending on relations defined in the DN42 schema. For object +results, a 'Backlinks' section is also added providing an array of registry objects that +reference this one. + +If the 'raw' parameter is provided, attributes are returned un-decorated exactly +as contained in the registry. + +Some examples will help clarify: + +* Return a JSON object, with keys for each registry type and values containing a count +of the number of registry objects for each type + +``` +wget -O - -q http://localhost:8042/api/registry/ | jq +{ + "as-block": 8, + "as-set": 34, + "aut-num": 1486, + "domain": 451, + "inet6num": 746, + "inetnum": 1276, + "key-cert": 7, + "mntner": 1379, + "organisation": 275, + "person": 1388, + "registry": 4, + "role": 14, + "route": 892, + "route-set": 2, + "route6": 596, + "schema": 18, + "tinc-key": 25, + "tinc-keyset": 3 +} + +``` + +* Return a list of all objects in the role type + +``` +wget -O - -q http://localhost:8042/api/registry/role | jq +{ + "role": [ + "ORG-NETRAVNEN-DN42", + "PACKETPUSHERS-DN42", + "CCCKC-DN42", + "FLHB-ABUSE-DN42", + "NIXNODES-DN42", + "ORG-SHACK-ABUSE-DN42", + "ORG-SHACK-TECH-DN42", + "ORG-YANE-DN42", + "SOURIS-DN42", + "CCCHB-ABUSE-DN42", + "MAGLAB-DN42", + "NL-ZUID-DN42", + "ORG-SHACK-ADMIN-DN42", + "ALENAN-DN42" + ] +} +``` + +* Returns a list of all objects in types that match 'route' + +``` +wget -O - -q http://localhost:8042/api/registry/*route | jq +{ + "route": [ + "172.20.28.0_27", + "172.23.220.0_24", + "172.23.82.0_25", + "10.149.0.0_16", + +... + + "172.20.128.0_27", + "172.22.127.32_27" + ], + "route-set": [ + "RS-DN42", + "RS-DN42-NATIVE" + ], + "route6": [ + "fd42:df42::_48", + "fd5c:0f0f:39fc::_48", + +... + + "fd16:c638:3d7c::_48", + "fd23::_48" + ] +} +``` + +* Returns the mntner/BURBLE-MNT object (in decorated format) + +``` +wget -O - -q http://localhost:8042/api/registry/mntner/BURBLE-MNT | jq +{ + "mntner/BURBLE-MNT": { + "Attributes": [ + [ + "mntner", + "BURBLE-MNT" + ], + [ + "descr", + "burble.dn42 https://dn42.burble.com/" + ], + [ + "admin-c", + "[BURBLE-DN42](person/BURBLE-DN42)" + ], + [ + "tech-c", + "[BURBLE-DN42](person/BURBLE-DN42)" + ], + [ + "auth", + "pgp-fingerprint 1C08F282095CCDA432AECC657B9FE8780CFB6593" + ], + [ + "mnt-by", + "[BURBLE-MNT](mntner/BURBLE-MNT)" + ], + [ + "source", + "[DN42](registry/DN42)" + ] + ], + "Backlinks": [ + "aut-num/AS4242422602", + "aut-num/AS4242422601", + "mntner/BURBLE-MNT", + "route/172.20.129.160_27", + "as-set/AS4242422601:AS-DOWNSTREAM", + "as-set/AS4242422601:AS-TRANSIT", + "person/BURBLE-DN42", + "inet6num/fd42:4242:2601::_48", + "domain/burble.dn42", + "domain/collector.dn42", + "route6/fd42:4242:2601::_48", + "inetnum/172.20.129.160_27" + ] + } +} +``` + +* Returns error 404, exact searches are case sensitive + +``` +wget -O - -q http://localhost:8042/api/registry/mntner/burble-mnt | jq +``` + +* Returns domain names matching 'burble' in raw format + +``` +wget -O - -q http://localhost:8042/api/registry/domain/*burble?raw | jq +{ + "domain/burble.dn42": [ + [ + "domain", + "burble.dn42" + ], + [ + "descr", + "burble.dn42 https://dn42.burble.com/" + ], + [ + "admin-c", + "BURBLE-DN42" + ], + [ + "tech-c", + "BURBLE-DN42" + ], + [ + "mnt-by", + "BURBLE-MNT" + ], + [ + "nserver", + "ns1.burble.dn42 172.20.129.161" + ], + [ + "nserver", + "ns1.burble.dn42 fd42:4242:2601:ac53::1" + ], + [ + "ds-rdata", + "61857 13 2 bd35e3efe3325d2029fb652e01604a48b677cc2f44226eeabee54b456c67680c" + ], + [ + "source", + "DN42" + ] + ] +} +``` + +* Returns all objects matching 172.20.0 + +``` +wget -O - -q http://localhost:8042/api/registry/*/*172.20.0 | jq +{ + "inetnum/172.20.0.0_14": { + "Attributes": [ + [ + "inetnum", + "172.20.0.0 - 172.23.255.255" + ], + [ + "cidr", + "172.20.0.0/14" + ], + +... and so on +``` + +* Returns the nic-hdl attribute for all person objects + +``` +wget -O - -q http://localhost:8042/api/registry/person/*/nic-hdl | jq +{ + "person/0RIGO-DN42": { + "nic-hdl": [ + "0RIGO-DN42" + ] + }, + "person/0XDRAGON-DN42": { + "nic-hdl": [ + "0XDRAGON-DN42" + ] + }, + "person/1714-DN42": { + "nic-hdl": [ + "1714-DN42" + ] + }, + +... and so on +``` + +* return raw contact (-c) attributes in aut-num objects that contain 'burble' + +``` +wget -O - -q http://localhost:8042/api/registry/aut-num/*/*-c/*burble?raw | jq +{ + "aut-num/AS4242422601": { + "admin-c": [ + "BURBLE-DN42" + ], + "tech-c": [ + "BURBLE-DN42" + ] + }, + "aut-num/AS4242422602": { + "admin-c": [ + "BURBLE-DN42" + ], + "tech-c": [ + "BURBLE-DN42" + ] + } +} +``` + +A special query exists to return metadata about the registry + +``` +GET /api/registry/.meta +``` + +Example Output (JSON format): +``` +wget -O - -q http://localhost:8042/api/dns/.meta | jq +``` + +``` +{ + "Commit": "fa89d022d0c2a48bcfbee405e2f3685f3b9cf063" +} +``` + +## Route Origin Authorisation (ROA) API + +Route Origin Authorisation (ROA) data can be obtained from the server in +JSON and bird formats. + +### JSON format output + +``` +GET /api/roa/json +``` + +Provides IPv4 and IPv6 ROAs in JSON format, suitable for use with +[gortr](https://github.com/cloudflare/gortr). + +Example Output: +``` +wget -O - -q http://localhost:8042/api/roa/json | jq +``` + +``` +{ + "metadata": { + "counts": 1564, + "generated": 1550402199, + "valid": 1550445399 + }, + "roas": [ + { + "prefix": "172.23.128.0/26", + "maxLength": 29, + "asn": "AS4242422747" + }, + { + "prefix": "172.22.129.192/26", + "maxLength": 29, + "asn": "AS4242423976" + }, + { + "prefix": "10.110.0.0/16", + "maxLength": 24, + "asn": "AS65110" + }, + +... and so on +``` + +### Bird format output + +``` +GET /api/roa/bird/{bird version}/{IP family} +``` + +Provides ROA data suitable for including in to bird. + +{bird version} must be either 1 or 2 + +{IP family} can be 4, 6 or 46 to provide both IPv4 and IPv6 results + + +Example Output: +``` +wget -O - -q http://localhost:8042/api/roa/bird/1/4 +``` + +``` +# +# dn42regsrv ROA Generator +# Last Updated: 2019-02-17 11:16:39.668799525 +0000 GMT m=+0.279049704 +# Commit: 3cbc349bf770493c016888ff785227ded2a7d866 +# +roa 172.23.128.0/26 max 29 as 4242422747; +roa 172.22.129.192/26 max 29 as 4242423976; +roa 10.110.0.0/16 max 24 as 65110; +roa 172.20.164.0/26 max 29 as 4242423023; +roa 172.20.135.200/29 max 29 as 4242420448; +roa 10.65.0.0/20 max 24 as 4242420420; +roa 172.20.149.136/29 max 29 as 4242420234; +roa 10.160.0.0/13 max 24 as 65079; +roa 10.169.0.0/16 max 24 as 65534; + +... and so on +``` + +``` +wget -O - -q http://localhost:8042/api/roa/bird/2/6 +``` + +``` +# +# dn42regsrv ROA Generator +# Last Updated: 2019-02-17 11:16:39.668799525 +0000 GMT m=+0.279049704 +# Commit: 3cbc349bf770493c016888ff785227ded2a7d866 +# +route fdc3:10cd:ae9d::/48 max 64 as 4242420789; +route fd41:9805:7b69:4000::/51 max 64 as 4242420846; +route fd41:9805:7b69:4000::/51 max 64 as 4242420845; +route fd41:9805:7b69:4000::/51 max 64 as 4242420847; +route fddf:ebfd:a801:2331::/64 max 64 as 65530; +route fd42:1a2b:de57::/48 max 64 as 4242422454; +route fd42:7879:7879::/48 max 64 as 4242421787; + +... and so on +``` + +### filter{,6}.txt + +``` +GET /api/roa/filter/{IP family} +``` + +Provides the contents of filter.txt and filter6.txt in json format. + +{IP family} can be 4, 6 or 46 to provide both IPv4 and IPv6 results + + +Example Output: +``` +wget -O - -q http://localhost:8042/api/roa/filter/6 | jq +``` + +``` +[ + { + "nr": 1001, + "action": "permit", + "prefix": "fd00::/8", + "minlen": 44, + "maxlen": 64 + }, + { + "nr": 9999, + "action": "deny", + "prefix": "::/0", + "minlen": 0, + "maxlen": 128 + } +] +``` + + +## DNS Root Zone API + +The DNS API provides a list of resource records that can be used to create a root zone for DN42 +related domains. By polling the API, DNS servers are able to keep their root zone delegations and +DNSSEC records up to date. + + +``` +GET /api/dns/root-zone?format={[json|bind]} +``` + +Format may either 'json' or 'bind' to provide resource records in either format. The default +output format is JSON. + +Example Output (JSON format): +``` +wget -O - -q http://localhost:8042/api/dns/root-zone?format=json | jq +``` + +``` +{ + "Records": [ + { + "Name": "dn42", + "Type": "NS", + "Content": "b.delegation-servers.dn42.", + "Comment": "DN42 Authoritative Zone" + }, + { + "Name": "dn42", + "Type": "NS", + "Content": "j.delegation-servers.dn42.", + "Comment": "DN42 Authoritative Zone" + }, + +... and so on +``` + +Example Output (BIND format): +``` +wget -O - -q http://localhost:8042/api/dns/root-zone?format=bind +``` + +``` +;; DN42 Root Zone Records +;; Commit Reference: 2cc95d9101268ce82239dee1f947e4a8273524a9 +;; Generated: 2019-03-08 19:40:51.264803795 +0000 GMT m=+0.197704585 +dn42 IN NS b.delegation-servers.dn42. ; DN42 Authoritative Zone +dn42 IN NS j.delegation-servers.dn42. ; DN42 Authoritative Zone +dn42 IN NS y.delegation-servers.dn42. ; DN42 Authoritative Zone +dn42 IN DS 64441 10 2 6dadda00f5986bd26fe4f162669742cf7eba07d212b525acac9840ee06cb2799 ; DN42 Authoritative Zone +dn42 IN DS 56676 10 2 4b559c949eb796f5502f05bd5bb2143672e7ef935286db552955f291bb81093e ; DN42 Authoritative Zone +d.f.ip6.arpa IN NS b.delegation-servers.dn42. ; DN42 Authoritative Zone +d.f.ip6.arpa IN NS j.delegation-servers.dn42. ; DN42 Authoritative Zone +d.f.ip6.arpa IN NS y.delegation-servers.dn42. ; DN42 Authoritative Zone +d.f.ip6.arpa IN DS 64441 10 2 9057500a3b6e09bf45a60ed8891f2e649c6812d5d149c45a3c560fa0a619 +5c49 ; DN42 Authoritative Zone +d.f.ip6.arpa IN DS 56676 10 2 d93cfd941025aaa445283d33e27157bb9a2df0a9c1389fdf5e36a377fc31 +4736 ; DN42 Authoritative Zone + +... and so on +``` diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..27b3b1f --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2019 Simon Marsh + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md index 2ac5841..c2626d3 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,89 @@ -# pipeline-test +# dn42regsrv + +A REST API for the DN42 registry, written in Go, to provide a bridge between +interactive applications and registry data. + +A public instance of the API and explorer web app can be accessed via: + +* [https://explorer.burble.com/](https://explorer.burble.com/) (Internet link) +* [http://explorer.collector.dn42/](http://explorer.collector.dn42/) (DN42 Link) + +## Features + +* REST API for querying DN42 registry objects +* Able to decorate objects with relationship information based on SCHEMA type definitions +* Includes a simple webserver for delivering static files which can be used to deliver + basic web applications utilising the API (such as the included DN42 Registry Explorer) +* Automatic pull from the DN42 git repository to keep the registry up to date +* Includes a responsive web app for exploring the registry +* API endpoints for ROA data in JSON, and bird formats +* API endpoint to support the creation of DNS root zone records + +## Building + +#### Using locally installed go + +Requires [git](https://git-scm.com/) and [go](https://golang.org) +``` +go get -insecure git.dn42.us/burble/dn42regsrv +``` + +#### Without installing go + +Using container runtime to build with the golang container: +``` +docker run -v ${PWD}:/go/bin golang go get -insecure git.dn42.us/burble/dn42regsrv +``` + +Or use the *contrib/build.sh* script after cloning the repo. + +## Running + +#### As a service + +Use --help to view configurable options +``` +${GOPATH}/bin/dn42regsrv --help +``` + +The server requires access to a clone of the DN42 registry and for +the git executable to be accessible. +If you want to use the auto pull feature then the registry must +also be writable by the server. + +``` +cd ${GOPTH}/src/git.dn42.us/burble/dn42regsrv +git clone http://git.dn42.us/dn42/registry.git +${GOPATH}/dn42regsrv +``` + +A sample service file is included for running the server under systemd + +#### Within a container + +A container build script (*contrib/buildah.sh*) is included in the +contrib directory. The script uses [buildah](https://buildah.io/). + +See the *contrib/entrypoint.sh* script for environment variables that can +be set when running the container. + +## Using + +By default the server will be listening on port 8042. +See the [API.md](API.md) file for a detailed description of the API. + + +## Support + +Please feel free to raise issues or create pull requests for the project git repository. + +## #ToDo + +### Server + +- Add WHOIS interface + +### DN42 Registry Explorer Web App + +- Allow for attribute searches -Test pipeline \ No newline at end of file diff --git a/StaticRoot/anchorme.min.js b/StaticRoot/anchorme.min.js new file mode 100644 index 0000000..82e74ee --- /dev/null +++ b/StaticRoot/anchorme.min.js @@ -0,0 +1 @@ +!function(e,a){"object"==typeof exports&&"undefined"!=typeof module?module.exports=a():"function"==typeof define&&define.amd?define(a):e.anchorme=a()}(this,function(){"use strict";function e(e,a){return a={exports:{}},e(a,a.exports),a.exports}var a=e(function(e,a){function n(e){return e||(e={attributes:[],ips:!0,emails:!0,urls:!0,files:!0,truncate:1/0,defaultProtocol:"http://",list:!1}),"object"!=typeof e.attributes&&(e.attributes=[]),"boolean"!=typeof e.ips&&(e.ips=!0),"boolean"!=typeof e.emails&&(e.emails=!0),"boolean"!=typeof e.urls&&(e.urls=!0),"boolean"!=typeof e.files&&(e.files=!0),"boolean"!=typeof e.list&&(e.list=!1),"string"!=typeof e.defaultProtocol&&"function"!=typeof e.defaultProtocol&&(e.defaultProtocol="http://"),"number"==typeof e.truncate||"object"==typeof e.truncate&&null!==e.truncate||(e.truncate=1/0),e}function t(e){return!isNaN(Number(e))&&!(Number(e)>65535)}Object.defineProperty(a,"__esModule",{value:!0}),a.defaultOptions=n,a.isPort=t}),n=e(function(e,a){Object.defineProperty(a,"__esModule",{value:!0}),a.tlds=["com","org","net","uk","gov","edu","io","cc","co","aaa","aarp","abarth","abb","abbott","abbvie","abc","able","abogado","abudhabi","ac","academy","accenture","accountant","accountants","aco","active","actor","ad","adac","ads","adult","ae","aeg","aero","aetna","af","afamilycompany","afl","africa","ag","agakhan","agency","ai","aig","aigo","airbus","airforce","airtel","akdn","al","alfaromeo","alibaba","alipay","allfinanz","allstate","ally","alsace","alstom","am","americanexpress","americanfamily","amex","amfam","amica","amsterdam","analytics","android","anquan","anz","ao","aol","apartments","app","apple","aq","aquarelle","ar","aramco","archi","army","arpa","art","arte","as","asda","asia","associates","at","athleta","attorney","au","auction","audi","audible","audio","auspost","author","auto","autos","avianca","aw","aws","ax","axa","az","azure","ba","baby","baidu","banamex","bananarepublic","band","bank","bar","barcelona","barclaycard","barclays","barefoot","bargains","baseball","basketball","bauhaus","bayern","bb","bbc","bbt","bbva","bcg","bcn","bd","be","beats","beauty","beer","bentley","berlin","best","bestbuy","bet","bf","bg","bh","bharti","bi","bible","bid","bike","bing","bingo","bio","biz","bj","black","blackfriday","blanco","blockbuster","blog","bloomberg","blue","bm","bms","bmw","bn","bnl","bnpparibas","bo","boats","boehringer","bofa","bom","bond","boo","book","booking","boots","bosch","bostik","boston","bot","boutique","box","br","bradesco","bridgestone","broadway","broker","brother","brussels","bs","bt","budapest","bugatti","build","builders","business","buy","buzz","bv","bw","by","bz","bzh","ca","cab","cafe","cal","call","calvinklein","cam","camera","camp","cancerresearch","canon","capetown","capital","capitalone","car","caravan","cards","care","career","careers","cars","cartier","casa","case","caseih","cash","casino","cat","catering","catholic","cba","cbn","cbre","cbs","cd","ceb","center","ceo","cern","cf","cfa","cfd","cg","ch","chanel","channel","chase","chat","cheap","chintai","chloe","christmas","chrome","chrysler","church","ci","cipriani","circle","cisco","citadel","citi","citic","city","cityeats","ck","cl","claims","cleaning","click","clinic","clinique","clothing","cloud","club","clubmed","cm","cn","coach","codes","coffee","college","cologne","comcast","commbank","community","company","compare","computer","comsec","condos","construction","consulting","contact","contractors","cooking","cookingchannel","cool","coop","corsica","country","coupon","coupons","courses","cr","credit","creditcard","creditunion","cricket","crown","crs","cruise","cruises","csc","cu","cuisinella","cv","cw","cx","cy","cymru","cyou","cz","dabur","dad","dance","data","date","dating","datsun","day","dclk","dds","de","deal","dealer","deals","degree","delivery","dell","deloitte","delta","democrat","dental","dentist","desi","design","dev","dhl","diamonds","diet","digital","direct","directory","discount","discover","dish","diy","dj","dk","dm","dnp","do","docs","doctor","dodge","dog","doha","domains","dot","download","drive","dtv","dubai","duck","dunlop","duns","dupont","durban","dvag","dvr","dz","earth","eat","ec","eco","edeka","education","ee","eg","email","emerck","energy","engineer","engineering","enterprises","epost","epson","equipment","er","ericsson","erni","es","esq","estate","esurance","et","eu","eurovision","eus","events","everbank","exchange","expert","exposed","express","extraspace","fage","fail","fairwinds","faith","family","fan","fans","farm","farmers","fashion","fast","fedex","feedback","ferrari","ferrero","fi","fiat","fidelity","fido","film","final","finance","financial","fire","firestone","firmdale","fish","fishing","fit","fitness","fj","fk","flickr","flights","flir","florist","flowers","fly","fm","fo","foo","food","foodnetwork","football","ford","forex","forsale","forum","foundation","fox","fr","free","fresenius","frl","frogans","frontdoor","frontier","ftr","fujitsu","fujixerox","fun","fund","furniture","futbol","fyi","ga","gal","gallery","gallo","gallup","game","games","gap","garden","gb","gbiz","gd","gdn","ge","gea","gent","genting","george","gf","gg","ggee","gh","gi","gift","gifts","gives","giving","gl","glade","glass","gle","global","globo","gm","gmail","gmbh","gmo","gmx","gn","godaddy","gold","goldpoint","golf","goo","goodhands","goodyear","goog","google","gop","got","gp","gq","gr","grainger","graphics","gratis","green","gripe","group","gs","gt","gu","guardian","gucci","guge","guide","guitars","guru","gw","gy","hair","hamburg","hangout","haus","hbo","hdfc","hdfcbank","health","healthcare","help","helsinki","here","hermes","hgtv","hiphop","hisamitsu","hitachi","hiv","hk","hkt","hm","hn","hockey","holdings","holiday","homedepot","homegoods","homes","homesense","honda","honeywell","horse","hospital","host","hosting","hot","hoteles","hotmail","house","how","hr","hsbc","ht","htc","hu","hughes","hyatt","hyundai","ibm","icbc","ice","icu","id","ie","ieee","ifm","ikano","il","im","imamat","imdb","immo","immobilien","in","industries","infiniti","info","ing","ink","institute","insurance","insure","int","intel","international","intuit","investments","ipiranga","iq","ir","irish","is","iselect","ismaili","ist","istanbul","it","itau","itv","iveco","iwc","jaguar","java","jcb","jcp","je","jeep","jetzt","jewelry","jio","jlc","jll","jm","jmp","jnj","jo","jobs","joburg","jot","joy","jp","jpmorgan","jprs","juegos","juniper","kaufen","kddi","ke","kerryhotels","kerrylogistics","kerryproperties","kfh","kg","kh","ki","kia","kim","kinder","kindle","kitchen","kiwi","km","kn","koeln","komatsu","kosher","kp","kpmg","kpn","kr","krd","kred","kuokgroup","kw","ky","kyoto","kz","la","lacaixa","ladbrokes","lamborghini","lamer","lancaster","lancia","lancome","land","landrover","lanxess","lasalle","lat","latino","latrobe","law","lawyer","lb","lc","lds","lease","leclerc","lefrak","legal","lego","lexus","lgbt","li","liaison","lidl","life","lifeinsurance","lifestyle","lighting","like","lilly","limited","limo","lincoln","linde","link","lipsy","live","living","lixil","lk","loan","loans","locker","locus","loft","lol","london","lotte","lotto","love","lpl","lplfinancial","lr","ls","lt","ltd","ltda","lu","lundbeck","lupin","luxe","luxury","lv","ly","ma","macys","madrid","maif","maison","makeup","man","management","mango","market","marketing","markets","marriott","marshalls","maserati","mattel","mba","mc","mcd","mcdonalds","mckinsey","md","me","med","media","meet","melbourne","meme","memorial","men","menu","meo","metlife","mg","mh","miami","microsoft","mil","mini","mint","mit","mitsubishi","mk","ml","mlb","mls","mm","mma","mn","mo","mobi","mobile","mobily","moda","moe","moi","mom","monash","money","monster","montblanc","mopar","mormon","mortgage","moscow","moto","motorcycles","mov","movie","movistar","mp","mq","mr","ms","msd","mt","mtn","mtpc","mtr","mu","museum","mutual","mv","mw","mx","my","mz","na","nab","nadex","nagoya","name","nationwide","natura","navy","nba","nc","ne","nec","netbank","netflix","network","neustar","new","newholland","news","next","nextdirect","nexus","nf","nfl","ng","ngo","nhk","ni","nico","nike","nikon","ninja","nissan","nissay","nl","no","nokia","northwesternmutual","norton","now","nowruz","nowtv","np","nr","nra","nrw","ntt","nu","nyc","nz","obi","observer","off","office","okinawa","olayan","olayangroup","oldnavy","ollo","om","omega","one","ong","onl","online","onyourside","ooo","open","oracle","orange","organic","orientexpress","origins","osaka","otsuka","ott","ovh","pa","page","pamperedchef","panasonic","panerai","paris","pars","partners","parts","party","passagens","pay","pccw","pe","pet","pf","pfizer","pg","ph","pharmacy","philips","phone","photo","photography","photos","physio","piaget","pics","pictet","pictures","pid","pin","ping","pink","pioneer","pizza","pk","pl","place","play","playstation","plumbing","plus","pm","pn","pnc","pohl","poker","politie","porn","post","pr","pramerica","praxi","press","prime","pro","prod","productions","prof","progressive","promo","properties","property","protection","pru","prudential","ps","pt","pub","pw","pwc","py","qa","qpon","quebec","quest","qvc","racing","radio","raid","re","read","realestate","realtor","realty","recipes","red","redstone","redumbrella","rehab","reise","reisen","reit","reliance","ren","rent","rentals","repair","report","republican","rest","restaurant","review","reviews","rexroth","rich","richardli","ricoh","rightathome","ril","rio","rip","rmit","ro","rocher","rocks","rodeo","rogers","room","rs","rsvp","ru","ruhr","run","rw","rwe","ryukyu","sa","saarland","safe","safety","sakura","sale","salon","samsclub","samsung","sandvik","sandvikcoromant","sanofi","sap","sapo","sarl","sas","save","saxo","sb","sbi","sbs","sc","sca","scb","schaeffler","schmidt","scholarships","school","schule","schwarz","science","scjohnson","scor","scot","sd","se","seat","secure","security","seek","select","sener","services","ses","seven","sew","sex","sexy","sfr","sg","sh","shangrila","sharp","shaw","shell","shia","shiksha","shoes","shop","shopping","shouji","show","showtime","shriram","si","silk","sina","singles","site","sj","sk","ski","skin","sky","skype","sl","sling","sm","smart","smile","sn","sncf","so","soccer","social","softbank","software","sohu","solar","solutions","song","sony","soy","space","spiegel","spot","spreadbetting","sr","srl","srt","st","stada","staples","star","starhub","statebank","statefarm","statoil","stc","stcgroup","stockholm","storage","store","stream","studio","study","style","su","sucks","supplies","supply","support","surf","surgery","suzuki","sv","swatch","swiftcover","swiss","sx","sy","sydney","symantec","systems","sz","tab","taipei","talk","taobao","target","tatamotors","tatar","tattoo","tax","taxi","tc","tci","td","tdk","team","tech","technology","tel","telecity","telefonica","temasek","tennis","teva","tf","tg","th","thd","theater","theatre","tiaa","tickets","tienda","tiffany","tips","tires","tirol","tj","tjmaxx","tjx","tk","tkmaxx","tl","tm","tmall","tn","to","today","tokyo","tools","top","toray","toshiba","total","tours","town","toyota","toys","tr","trade","trading","training","travel","travelchannel","travelers","travelersinsurance","trust","trv","tt","tube","tui","tunes","tushu","tv","tvs","tw","tz","ua","ubank","ubs","uconnect","ug","unicom","university","uno","uol","ups","us","uy","uz","va","vacations","vana","vanguard","vc","ve","vegas","ventures","verisign","versicherung","vet","vg","vi","viajes","video","vig","viking","villas","vin","vip","virgin","visa","vision","vista","vistaprint","viva","vivo","vlaanderen","vn","vodka","volkswagen","volvo","vote","voting","voto","voyage","vu","vuelos","wales","walmart","walter","wang","wanggou","warman","watch","watches","weather","weatherchannel","webcam","weber","website","wed","wedding","weibo","weir","wf","whoswho","wien","wiki","williamhill","win","windows","wine","winners","wme","wolterskluwer","woodside","work","works","world","wow","ws","wtc","wtf","xbox","xerox","xfinity","xihuan","xin","xn--11b4c3d","xn--1ck2e1b","xn--1qqw23a","xn--30rr7y","xn--3bst00m","xn--3ds443g","xn--3e0b707e","xn--3oq18vl8pn36a","xn--3pxu8k","xn--42c2d9a","xn--45brj9c","xn--45q11c","xn--4gbrim","xn--54b7fta0cc","xn--55qw42g","xn--55qx5d","xn--5su34j936bgsg","xn--5tzm5g","xn--6frz82g","xn--6qq986b3xl","xn--80adxhks","xn--80ao21a","xn--80aqecdr1a","xn--80asehdb","xn--80aswg","xn--8y0a063a","xn--90a3ac","xn--90ae","xn--90ais","xn--9dbq2a","xn--9et52u","xn--9krt00a","xn--b4w605ferd","xn--bck1b9a5dre4c","xn--c1avg","xn--c2br7g","xn--cck2b3b","xn--cg4bki","xn--clchc0ea0b2g2a9gcd","xn--czr694b","xn--czrs0t","xn--czru2d","xn--d1acj3b","xn--d1alf","xn--e1a4c","xn--eckvdtc9d","xn--efvy88h","xn--estv75g","xn--fct429k","xn--fhbei","xn--fiq228c5hs","xn--fiq64b","xn--fiqs8s","xn--fiqz9s","xn--fjq720a","xn--flw351e","xn--fpcrj9c3d","xn--fzc2c9e2c","xn--fzys8d69uvgm","xn--g2xx48c","xn--gckr3f0f","xn--gecrj9c","xn--gk3at1e","xn--h2brj9c","xn--hxt814e","xn--i1b6b1a6a2e","xn--imr513n","xn--io0a7i","xn--j1aef","xn--j1amh","xn--j6w193g","xn--jlq61u9w7b","xn--jvr189m","xn--kcrx77d1x4a","xn--kprw13d","xn--kpry57d","xn--kpu716f","xn--kput3i","xn--l1acc","xn--lgbbat1ad8j","xn--mgb9awbf","xn--mgba3a3ejt","xn--mgba3a4f16a","xn--mgba7c0bbn0a","xn--mgbaam7a8h","xn--mgbab2bd","xn--mgbai9azgqp6j","xn--mgbayh7gpa","xn--mgbb9fbpob","xn--mgbbh1a71e","xn--mgbc0a9azcg","xn--mgbca7dzdo","xn--mgberp4a5d4ar","xn--mgbi4ecexp","xn--mgbpl2fh","xn--mgbt3dhd","xn--mgbtx2b","xn--mgbx4cd0ab","xn--mix891f","xn--mk1bu44c","xn--mxtq1m","xn--ngbc5azd","xn--ngbe9e0a","xn--node","xn--nqv7f","xn--nqv7fs00ema","xn--nyqy26a","xn--o3cw4h","xn--ogbpf8fl","xn--p1acf","xn--p1ai","xn--pbt977c","xn--pgbs0dh","xn--pssy2u","xn--q9jyb4c","xn--qcka1pmc","xn--qxam","xn--rhqv96g","xn--rovu88b","xn--s9brj9c","xn--ses554g","xn--t60b56a","xn--tckwe","xn--tiq49xqyj","xn--unup4y","xn--vermgensberater-ctb","xn--vermgensberatung-pwb","xn--vhquv","xn--vuq861b","xn--w4r85el8fhu5dnra","xn--w4rs40l","xn--wgbh1c","xn--wgbl6a","xn--xhq521b","xn--xkc2al3hye2a","xn--xkc2dl3a5ee0h","xn--y9a3aq","xn--yfro4i67o","xn--ygbi2ammx","xn--zfr164b","xperia","xxx","xyz","yachts","yahoo","yamaxun","yandex","ye","yodobashi","yoga","yokohama","you","youtube","yt","yun","za","zappos","zara","zero","zip","zippo","zm","zone","zuerich","zw"],a.htmlAttrs=["src=","data=","href=","cite=","formaction=","icon=","manifest=","poster=","codebase=","background=","profile=","usemap="]}),t=e(function(e,a){function t(e){var a=e.match(o);if(null===a)return!1;for(var t=r.length-1;t>=0;t--)if(r[t].test(e))return!1;var i=a[2];return!!i&&-1!==n.tlds.indexOf(i)}Object.defineProperty(a,"__esModule",{value:!0});var o=/^[a-z0-9!#$%&'*+\-\/=?^_`{|}~.]+@([a-z0-9%\-]+\.){1,}([a-z0-9\-]+)?$/i,r=[/^[!#$%&'*+\-\/=?^_`{|}~.]/,/[.]{2,}[a-z0-9!#$%&'*+\-\/=?^_`{|}~.]+@/i,/\.@/];a.default=t}),o=e(function(e,n){function t(e){if(!o.test(e))return!1;var n=e.split("."),t=Number(n[0]);if(isNaN(t)||t>255||t<0)return!1;var r=Number(n[1]);if(isNaN(r)||r>255||r<0)return!1;var i=Number(n[2]);if(isNaN(i)||i>255||i<0)return!1;var s=Number((n[3].match(/^\d+/)||[])[0]);if(isNaN(s)||s>255||s<0)return!1;var c=(n[3].match(/(^\d+)(:)(\d+)/)||[])[3];return!(c&&!a.isPort(c))}Object.defineProperty(n,"__esModule",{value:!0});var o=/^(\d{1,3}\.){3}\d{1,3}(:\d{1,5})?(\/([a-z0-9\-._~:\/\?#\[\]@!$&'\(\)\*\+,;=%]+)?)?$/i;n.default=t}),r=e(function(e,t){function o(e){var t=e.match(r);return null!==t&&("string"==typeof t[3]&&(-1!==n.tlds.indexOf(t[3].toLowerCase())&&!(t[5]&&!a.isPort(t[5]))))}Object.defineProperty(t,"__esModule",{value:!0});var r=/^(https?:\/\/|ftps?:\/\/)?([a-z0-9%\-]+\.){1,}([a-z0-9\-]+)?(:(\d{1,5}))?(\/([a-z0-9\-._~:\/\?#\[\]@!$&'\(\)\*\+,;=%]+)?)?$/i;t.default=o}),i=e(function(e,a){function n(e,a,t){return e.forEach(function(o,r){!(o.indexOf(".")>-1)||e[r-1]===a&&e[r+1]===t||e[r+1]!==a&&e[r+1]!==t||(e[r]=e[r]+e[r+1],"string"==typeof e[r+2]&&(e[r]=e[r]+e[r+2]),"string"==typeof e[r+3]&&(e[r]=e[r]+e[r+3]),"string"==typeof e[r+4]&&(e[r]=e[r]+e[r+4]),e.splice(r+1,4),n(e,a,t))}),e}function t(e){return e=n(e,"(",")"),e=n(e,"[","]"),e=n(e,'"','"'),e=n(e,"'","'")}Object.defineProperty(a,"__esModule",{value:!0}),a.fixSeparators=n,a.default=t}),s=e(function(e,a){function n(e){var a=e.replace(/([\s\(\)\[\]<>"'])/g,"\0$1\0").replace(/([?;:,.!]+)(?=(\0|$|\s))/g,"\0$1\0").split("\0");return i.default(a)}function t(e){return e.join("")}Object.defineProperty(a,"__esModule",{value:!0}),a.separate=n,a.deSeparate=t}),c=e(function(e,a){function n(e){return e=e.toLowerCase(),0===e.indexOf("http://")?"http://":0===e.indexOf("https://")?"https://":0===e.indexOf("ftp://")?"ftp://":0===e.indexOf("ftps://")?"ftps://":0===e.indexOf("file:///")?"file:///":0===e.indexOf("mailto:")&&"mailto:"}Object.defineProperty(a,"__esModule",{value:!0}),a.default=n}),l=e(function(e,a){function i(e,a){return e.map(function(i,s){var l=encodeURI(i);if(l.indexOf(".")<1&&!c.default(l))return i;var u=null,d=c.default(l)||"";return d&&(l=l.substr(d.length)),a.files&&"file:///"===d&&l.split(/\/|\\/).length-1&&(u={reason:"file",protocol:d,raw:i,encoded:l}),!u&&a.urls&&r.default(l)&&(u={reason:"url",protocol:d||("function"==typeof a.defaultProtocol?a.defaultProtocol(i):a.defaultProtocol),raw:i,encoded:l}),!u&&a.emails&&t.default(l)&&(u={reason:"email",protocol:"mailto:",raw:i,encoded:l}),!u&&a.ips&&o.default(l)&&(u={reason:"ip",protocol:d||("function"==typeof a.defaultProtocol?a.defaultProtocol(i):a.defaultProtocol),raw:i,encoded:l}),u&&("'"!==e[s-1]&&'"'!==e[s-1]||!~n.htmlAttrs.indexOf(e[s-2]))?u:i})}Object.defineProperty(a,"__esModule",{value:!0}),a.default=i}),u=e(function(e,a){function n(e,a){var n=o.separate(e),r=l.default(n,a);if(a.exclude)for(var i=0;ia.truncate&&(t=t.substring(0,a.truncate)+"..."),"object"==typeof a.truncate&&t.length>a.truncate[0]+a.truncate[1]&&(t=t.substr(0,a.truncate[0])+"..."+t.substr(t.length-a.truncate[1])),void 0===a.attributes&&(a.attributes=[]),'"+t+""}Object.defineProperty(a,"__esModule",{value:!0});var o=s;a.default=n}),d=e(function(e,n){Object.defineProperty(n,"__esModule",{value:!0});var i=function(e,n){return n=a.defaultOptions(n),u.default(e,n)};i.validate={ip:o.default,url:function(e){var a=c.default(e)||"";return e=e.substr(a.length),e=encodeURI(e),r.default(e)},email:t.default},n.default=i});return function(e){return e&&e.__esModule?e.default:e}(d)}); \ No newline at end of file diff --git a/StaticRoot/bootstrap.min.css b/StaticRoot/bootstrap.min.css new file mode 100644 index 0000000..416b0d7 --- /dev/null +++ b/StaticRoot/bootstrap.min.css @@ -0,0 +1,12 @@ +/*! + * Bootswatch v4.2.1 + * Homepage: https://bootswatch.com + * Copyright 2012-2018 Thomas Park + * Licensed under MIT + * Based on Bootstrap +*//*! + * Bootstrap v4.2.1 (https://getbootstrap.com/) + * Copyright 2011-2018 The Bootstrap Authors + * Copyright 2011-2018 Twitter, Inc. + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) + */@import url("https://fonts.googleapis.com/css?family=Source+Sans+Pro");:root{--blue: #B58900;--indigo: #6610f2;--purple: #6f42c1;--pink: #e83e8c;--red: #D33682;--orange: #fd7e14;--yellow: #CB4B16;--green: #2AA198;--teal: #20c997;--cyan: #268BD2;--white: #fff;--gray: #839496;--gray-dark: #073642;--primary: #B58900;--secondary: #839496;--success: #2AA198;--info: #268BD2;--warning: #CB4B16;--danger: #D33682;--light: #FDF6E3;--dark: #073642;--breakpoint-xs: 0;--breakpoint-sm: 576px;--breakpoint-md: 768px;--breakpoint-lg: 992px;--breakpoint-xl: 1200px;--font-family-sans-serif: "Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";--font-family-monospace: SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace}*,*::before,*::after{-webkit-box-sizing:border-box;box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:rgba(0,0,0,0)}article,aside,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;font-family:"Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";font-size:1rem;font-weight:400;line-height:1.5;color:#839496;text-align:left;background-color:#002B36}[tabindex="-1"]:focus{outline:0 !important}hr{-webkit-box-sizing:content-box;box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:0.5rem}p{margin-top:0;margin-bottom:1rem}abbr[title],abbr[data-original-title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;border-bottom:0;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul,dl{margin-top:0;margin-bottom:1rem}ol ol,ul ul,ol ul,ul ol{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#2AA198;text-decoration:none;background-color:transparent}a:hover{color:#2AA198;text-decoration:underline}a:not([href]):not([tabindex]){color:inherit;text-decoration:none}a:not([href]):not([tabindex]):hover,a:not([href]):not([tabindex]):focus{color:inherit;text-decoration:none}a:not([href]):not([tabindex]):focus{outline:0}pre,code,kbd,samp{font-family:SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace;font-size:1em}pre{margin-top:0;margin-bottom:1rem;overflow:auto}figure{margin:0 0 1rem}img{vertical-align:middle;border-style:none}svg{overflow:hidden;vertical-align:middle}table{border-collapse:collapse}caption{padding-top:0.75rem;padding-bottom:0.75rem;color:#839496;text-align:left;caption-side:bottom}th{text-align:inherit}label{display:inline-block;margin-bottom:0.5rem}button{border-radius:0}button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}input,button,select,optgroup,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}button,[type="button"],[type="reset"],[type="submit"]{-webkit-appearance:button}button::-moz-focus-inner,[type="button"]::-moz-focus-inner,[type="reset"]::-moz-focus-inner,[type="submit"]::-moz-focus-inner{padding:0;border-style:none}input[type="radio"],input[type="checkbox"]{-webkit-box-sizing:border-box;box-sizing:border-box;padding:0}input[type="date"],input[type="time"],input[type="datetime-local"],input[type="month"]{-webkit-appearance:listbox}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;max-width:100%;padding:0;margin-bottom:.5rem;font-size:1.5rem;line-height:inherit;color:inherit;white-space:normal}progress{vertical-align:baseline}[type="number"]::-webkit-inner-spin-button,[type="number"]::-webkit-outer-spin-button{height:auto}[type="search"]{outline-offset:-2px;-webkit-appearance:none}[type="search"]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item;cursor:pointer}template{display:none}[hidden]{display:none !important}h1,h2,h3,h4,h5,h6,.h1,.h2,.h3,.h4,.h5,.h6{margin-bottom:0.5rem;font-family:inherit;font-weight:500;line-height:1.2;color:inherit}h1,.h1{font-size:2.5rem}h2,.h2{font-size:2rem}h3,.h3{font-size:1.75rem}h4,.h4{font-size:1.5rem}h5,.h5{font-size:1.25rem}h6,.h6{font-size:1rem}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:6rem;font-weight:300;line-height:1.2}.display-2{font-size:5.5rem;font-weight:300;line-height:1.2}.display-3{font-size:4.5rem;font-weight:300;line-height:1.2}.display-4{font-size:3.5rem;font-weight:300;line-height:1.2}hr{margin-top:1rem;margin-bottom:1rem;border:0;border-top:1px solid rgba(0,0,0,0.1)}small,.small{font-size:80%;font-weight:400}mark,.mark{padding:0.2em;background-color:#fcf8e3}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:0.5rem}.initialism{font-size:90%;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote-footer{display:block;font-size:80%;color:#839496}.blockquote-footer::before{content:"\2014\00A0"}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:0.25rem;background-color:#002B36;border:1px solid #dee2e6;border-radius:0.25rem;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:0.5rem;line-height:1}.figure-caption{font-size:90%;color:#839496}code{font-size:87.5%;color:#e83e8c;word-break:break-word}a>code{color:inherit}kbd{padding:0.2rem 0.4rem;font-size:87.5%;color:#fff;background-color:#002B36;border-radius:0.2rem}kbd kbd{padding:0;font-size:100%;font-weight:700}pre{display:block;font-size:87.5%;color:inherit}pre code{font-size:inherit;color:inherit;word-break:normal}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width: 576px){.container{max-width:540px}}@media (min-width: 768px){.container{max-width:720px}}@media (min-width: 992px){.container{max-width:960px}}@media (min-width: 1200px){.container{max-width:1140px}}.container-fluid{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-15px;margin-left:-15px}.no-gutters{margin-right:0;margin-left:0}.no-gutters>.col,.no-gutters>[class*="col-"]{padding-right:0;padding-left:0}.col-1,.col-2,.col-3,.col-4,.col-5,.col-6,.col-7,.col-8,.col-9,.col-10,.col-11,.col-12,.col,.col-auto,.col-sm-1,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm,.col-sm-auto,.col-md-1,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-10,.col-md-11,.col-md-12,.col-md,.col-md-auto,.col-lg-1,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg,.col-lg-auto,.col-xl-1,.col-xl-2,.col-xl-3,.col-xl-4,.col-xl-5,.col-xl-6,.col-xl-7,.col-xl-8,.col-xl-9,.col-xl-10,.col-xl-11,.col-xl-12,.col-xl,.col-xl-auto{position:relative;width:100%;padding-right:15px;padding-left:15px}.col{-ms-flex-preferred-size:0;flex-basis:0;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-auto{-webkit-box-flex:0;-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-1{-webkit-box-flex:0;-ms-flex:0 0 8.3333333333%;flex:0 0 8.3333333333%;max-width:8.3333333333%}.col-2{-webkit-box-flex:0;-ms-flex:0 0 16.6666666667%;flex:0 0 16.6666666667%;max-width:16.6666666667%}.col-3{-webkit-box-flex:0;-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-4{-webkit-box-flex:0;-ms-flex:0 0 33.3333333333%;flex:0 0 33.3333333333%;max-width:33.3333333333%}.col-5{-webkit-box-flex:0;-ms-flex:0 0 41.6666666667%;flex:0 0 41.6666666667%;max-width:41.6666666667%}.col-6{-webkit-box-flex:0;-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-7{-webkit-box-flex:0;-ms-flex:0 0 58.3333333333%;flex:0 0 58.3333333333%;max-width:58.3333333333%}.col-8{-webkit-box-flex:0;-ms-flex:0 0 66.6666666667%;flex:0 0 66.6666666667%;max-width:66.6666666667%}.col-9{-webkit-box-flex:0;-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-10{-webkit-box-flex:0;-ms-flex:0 0 83.3333333333%;flex:0 0 83.3333333333%;max-width:83.3333333333%}.col-11{-webkit-box-flex:0;-ms-flex:0 0 91.6666666667%;flex:0 0 91.6666666667%;max-width:91.6666666667%}.col-12{-webkit-box-flex:0;-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-first{-webkit-box-ordinal-group:0;-ms-flex-order:-1;order:-1}.order-last{-webkit-box-ordinal-group:14;-ms-flex-order:13;order:13}.order-0{-webkit-box-ordinal-group:1;-ms-flex-order:0;order:0}.order-1{-webkit-box-ordinal-group:2;-ms-flex-order:1;order:1}.order-2{-webkit-box-ordinal-group:3;-ms-flex-order:2;order:2}.order-3{-webkit-box-ordinal-group:4;-ms-flex-order:3;order:3}.order-4{-webkit-box-ordinal-group:5;-ms-flex-order:4;order:4}.order-5{-webkit-box-ordinal-group:6;-ms-flex-order:5;order:5}.order-6{-webkit-box-ordinal-group:7;-ms-flex-order:6;order:6}.order-7{-webkit-box-ordinal-group:8;-ms-flex-order:7;order:7}.order-8{-webkit-box-ordinal-group:9;-ms-flex-order:8;order:8}.order-9{-webkit-box-ordinal-group:10;-ms-flex-order:9;order:9}.order-10{-webkit-box-ordinal-group:11;-ms-flex-order:10;order:10}.order-11{-webkit-box-ordinal-group:12;-ms-flex-order:11;order:11}.order-12{-webkit-box-ordinal-group:13;-ms-flex-order:12;order:12}.offset-1{margin-left:8.3333333333%}.offset-2{margin-left:16.6666666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.3333333333%}.offset-5{margin-left:41.6666666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.3333333333%}.offset-8{margin-left:66.6666666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.3333333333%}.offset-11{margin-left:91.6666666667%}@media (min-width: 576px){.col-sm{-ms-flex-preferred-size:0;flex-basis:0;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-sm-auto{-webkit-box-flex:0;-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-sm-1{-webkit-box-flex:0;-ms-flex:0 0 8.3333333333%;flex:0 0 8.3333333333%;max-width:8.3333333333%}.col-sm-2{-webkit-box-flex:0;-ms-flex:0 0 16.6666666667%;flex:0 0 16.6666666667%;max-width:16.6666666667%}.col-sm-3{-webkit-box-flex:0;-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-sm-4{-webkit-box-flex:0;-ms-flex:0 0 33.3333333333%;flex:0 0 33.3333333333%;max-width:33.3333333333%}.col-sm-5{-webkit-box-flex:0;-ms-flex:0 0 41.6666666667%;flex:0 0 41.6666666667%;max-width:41.6666666667%}.col-sm-6{-webkit-box-flex:0;-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-sm-7{-webkit-box-flex:0;-ms-flex:0 0 58.3333333333%;flex:0 0 58.3333333333%;max-width:58.3333333333%}.col-sm-8{-webkit-box-flex:0;-ms-flex:0 0 66.6666666667%;flex:0 0 66.6666666667%;max-width:66.6666666667%}.col-sm-9{-webkit-box-flex:0;-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-sm-10{-webkit-box-flex:0;-ms-flex:0 0 83.3333333333%;flex:0 0 83.3333333333%;max-width:83.3333333333%}.col-sm-11{-webkit-box-flex:0;-ms-flex:0 0 91.6666666667%;flex:0 0 91.6666666667%;max-width:91.6666666667%}.col-sm-12{-webkit-box-flex:0;-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-sm-first{-webkit-box-ordinal-group:0;-ms-flex-order:-1;order:-1}.order-sm-last{-webkit-box-ordinal-group:14;-ms-flex-order:13;order:13}.order-sm-0{-webkit-box-ordinal-group:1;-ms-flex-order:0;order:0}.order-sm-1{-webkit-box-ordinal-group:2;-ms-flex-order:1;order:1}.order-sm-2{-webkit-box-ordinal-group:3;-ms-flex-order:2;order:2}.order-sm-3{-webkit-box-ordinal-group:4;-ms-flex-order:3;order:3}.order-sm-4{-webkit-box-ordinal-group:5;-ms-flex-order:4;order:4}.order-sm-5{-webkit-box-ordinal-group:6;-ms-flex-order:5;order:5}.order-sm-6{-webkit-box-ordinal-group:7;-ms-flex-order:6;order:6}.order-sm-7{-webkit-box-ordinal-group:8;-ms-flex-order:7;order:7}.order-sm-8{-webkit-box-ordinal-group:9;-ms-flex-order:8;order:8}.order-sm-9{-webkit-box-ordinal-group:10;-ms-flex-order:9;order:9}.order-sm-10{-webkit-box-ordinal-group:11;-ms-flex-order:10;order:10}.order-sm-11{-webkit-box-ordinal-group:12;-ms-flex-order:11;order:11}.order-sm-12{-webkit-box-ordinal-group:13;-ms-flex-order:12;order:12}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.3333333333%}.offset-sm-2{margin-left:16.6666666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.3333333333%}.offset-sm-5{margin-left:41.6666666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.3333333333%}.offset-sm-8{margin-left:66.6666666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.3333333333%}.offset-sm-11{margin-left:91.6666666667%}}@media (min-width: 768px){.col-md{-ms-flex-preferred-size:0;flex-basis:0;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-md-auto{-webkit-box-flex:0;-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-md-1{-webkit-box-flex:0;-ms-flex:0 0 8.3333333333%;flex:0 0 8.3333333333%;max-width:8.3333333333%}.col-md-2{-webkit-box-flex:0;-ms-flex:0 0 16.6666666667%;flex:0 0 16.6666666667%;max-width:16.6666666667%}.col-md-3{-webkit-box-flex:0;-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-md-4{-webkit-box-flex:0;-ms-flex:0 0 33.3333333333%;flex:0 0 33.3333333333%;max-width:33.3333333333%}.col-md-5{-webkit-box-flex:0;-ms-flex:0 0 41.6666666667%;flex:0 0 41.6666666667%;max-width:41.6666666667%}.col-md-6{-webkit-box-flex:0;-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-md-7{-webkit-box-flex:0;-ms-flex:0 0 58.3333333333%;flex:0 0 58.3333333333%;max-width:58.3333333333%}.col-md-8{-webkit-box-flex:0;-ms-flex:0 0 66.6666666667%;flex:0 0 66.6666666667%;max-width:66.6666666667%}.col-md-9{-webkit-box-flex:0;-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-md-10{-webkit-box-flex:0;-ms-flex:0 0 83.3333333333%;flex:0 0 83.3333333333%;max-width:83.3333333333%}.col-md-11{-webkit-box-flex:0;-ms-flex:0 0 91.6666666667%;flex:0 0 91.6666666667%;max-width:91.6666666667%}.col-md-12{-webkit-box-flex:0;-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-md-first{-webkit-box-ordinal-group:0;-ms-flex-order:-1;order:-1}.order-md-last{-webkit-box-ordinal-group:14;-ms-flex-order:13;order:13}.order-md-0{-webkit-box-ordinal-group:1;-ms-flex-order:0;order:0}.order-md-1{-webkit-box-ordinal-group:2;-ms-flex-order:1;order:1}.order-md-2{-webkit-box-ordinal-group:3;-ms-flex-order:2;order:2}.order-md-3{-webkit-box-ordinal-group:4;-ms-flex-order:3;order:3}.order-md-4{-webkit-box-ordinal-group:5;-ms-flex-order:4;order:4}.order-md-5{-webkit-box-ordinal-group:6;-ms-flex-order:5;order:5}.order-md-6{-webkit-box-ordinal-group:7;-ms-flex-order:6;order:6}.order-md-7{-webkit-box-ordinal-group:8;-ms-flex-order:7;order:7}.order-md-8{-webkit-box-ordinal-group:9;-ms-flex-order:8;order:8}.order-md-9{-webkit-box-ordinal-group:10;-ms-flex-order:9;order:9}.order-md-10{-webkit-box-ordinal-group:11;-ms-flex-order:10;order:10}.order-md-11{-webkit-box-ordinal-group:12;-ms-flex-order:11;order:11}.order-md-12{-webkit-box-ordinal-group:13;-ms-flex-order:12;order:12}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.3333333333%}.offset-md-2{margin-left:16.6666666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.3333333333%}.offset-md-5{margin-left:41.6666666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.3333333333%}.offset-md-8{margin-left:66.6666666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.3333333333%}.offset-md-11{margin-left:91.6666666667%}}@media (min-width: 992px){.col-lg{-ms-flex-preferred-size:0;flex-basis:0;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-lg-auto{-webkit-box-flex:0;-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-lg-1{-webkit-box-flex:0;-ms-flex:0 0 8.3333333333%;flex:0 0 8.3333333333%;max-width:8.3333333333%}.col-lg-2{-webkit-box-flex:0;-ms-flex:0 0 16.6666666667%;flex:0 0 16.6666666667%;max-width:16.6666666667%}.col-lg-3{-webkit-box-flex:0;-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-lg-4{-webkit-box-flex:0;-ms-flex:0 0 33.3333333333%;flex:0 0 33.3333333333%;max-width:33.3333333333%}.col-lg-5{-webkit-box-flex:0;-ms-flex:0 0 41.6666666667%;flex:0 0 41.6666666667%;max-width:41.6666666667%}.col-lg-6{-webkit-box-flex:0;-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-lg-7{-webkit-box-flex:0;-ms-flex:0 0 58.3333333333%;flex:0 0 58.3333333333%;max-width:58.3333333333%}.col-lg-8{-webkit-box-flex:0;-ms-flex:0 0 66.6666666667%;flex:0 0 66.6666666667%;max-width:66.6666666667%}.col-lg-9{-webkit-box-flex:0;-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-lg-10{-webkit-box-flex:0;-ms-flex:0 0 83.3333333333%;flex:0 0 83.3333333333%;max-width:83.3333333333%}.col-lg-11{-webkit-box-flex:0;-ms-flex:0 0 91.6666666667%;flex:0 0 91.6666666667%;max-width:91.6666666667%}.col-lg-12{-webkit-box-flex:0;-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-lg-first{-webkit-box-ordinal-group:0;-ms-flex-order:-1;order:-1}.order-lg-last{-webkit-box-ordinal-group:14;-ms-flex-order:13;order:13}.order-lg-0{-webkit-box-ordinal-group:1;-ms-flex-order:0;order:0}.order-lg-1{-webkit-box-ordinal-group:2;-ms-flex-order:1;order:1}.order-lg-2{-webkit-box-ordinal-group:3;-ms-flex-order:2;order:2}.order-lg-3{-webkit-box-ordinal-group:4;-ms-flex-order:3;order:3}.order-lg-4{-webkit-box-ordinal-group:5;-ms-flex-order:4;order:4}.order-lg-5{-webkit-box-ordinal-group:6;-ms-flex-order:5;order:5}.order-lg-6{-webkit-box-ordinal-group:7;-ms-flex-order:6;order:6}.order-lg-7{-webkit-box-ordinal-group:8;-ms-flex-order:7;order:7}.order-lg-8{-webkit-box-ordinal-group:9;-ms-flex-order:8;order:8}.order-lg-9{-webkit-box-ordinal-group:10;-ms-flex-order:9;order:9}.order-lg-10{-webkit-box-ordinal-group:11;-ms-flex-order:10;order:10}.order-lg-11{-webkit-box-ordinal-group:12;-ms-flex-order:11;order:11}.order-lg-12{-webkit-box-ordinal-group:13;-ms-flex-order:12;order:12}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.3333333333%}.offset-lg-2{margin-left:16.6666666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.3333333333%}.offset-lg-5{margin-left:41.6666666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.3333333333%}.offset-lg-8{margin-left:66.6666666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.3333333333%}.offset-lg-11{margin-left:91.6666666667%}}@media (min-width: 1200px){.col-xl{-ms-flex-preferred-size:0;flex-basis:0;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;max-width:100%}.col-xl-auto{-webkit-box-flex:0;-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-xl-1{-webkit-box-flex:0;-ms-flex:0 0 8.3333333333%;flex:0 0 8.3333333333%;max-width:8.3333333333%}.col-xl-2{-webkit-box-flex:0;-ms-flex:0 0 16.6666666667%;flex:0 0 16.6666666667%;max-width:16.6666666667%}.col-xl-3{-webkit-box-flex:0;-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-xl-4{-webkit-box-flex:0;-ms-flex:0 0 33.3333333333%;flex:0 0 33.3333333333%;max-width:33.3333333333%}.col-xl-5{-webkit-box-flex:0;-ms-flex:0 0 41.6666666667%;flex:0 0 41.6666666667%;max-width:41.6666666667%}.col-xl-6{-webkit-box-flex:0;-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-xl-7{-webkit-box-flex:0;-ms-flex:0 0 58.3333333333%;flex:0 0 58.3333333333%;max-width:58.3333333333%}.col-xl-8{-webkit-box-flex:0;-ms-flex:0 0 66.6666666667%;flex:0 0 66.6666666667%;max-width:66.6666666667%}.col-xl-9{-webkit-box-flex:0;-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-xl-10{-webkit-box-flex:0;-ms-flex:0 0 83.3333333333%;flex:0 0 83.3333333333%;max-width:83.3333333333%}.col-xl-11{-webkit-box-flex:0;-ms-flex:0 0 91.6666666667%;flex:0 0 91.6666666667%;max-width:91.6666666667%}.col-xl-12{-webkit-box-flex:0;-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-xl-first{-webkit-box-ordinal-group:0;-ms-flex-order:-1;order:-1}.order-xl-last{-webkit-box-ordinal-group:14;-ms-flex-order:13;order:13}.order-xl-0{-webkit-box-ordinal-group:1;-ms-flex-order:0;order:0}.order-xl-1{-webkit-box-ordinal-group:2;-ms-flex-order:1;order:1}.order-xl-2{-webkit-box-ordinal-group:3;-ms-flex-order:2;order:2}.order-xl-3{-webkit-box-ordinal-group:4;-ms-flex-order:3;order:3}.order-xl-4{-webkit-box-ordinal-group:5;-ms-flex-order:4;order:4}.order-xl-5{-webkit-box-ordinal-group:6;-ms-flex-order:5;order:5}.order-xl-6{-webkit-box-ordinal-group:7;-ms-flex-order:6;order:6}.order-xl-7{-webkit-box-ordinal-group:8;-ms-flex-order:7;order:7}.order-xl-8{-webkit-box-ordinal-group:9;-ms-flex-order:8;order:8}.order-xl-9{-webkit-box-ordinal-group:10;-ms-flex-order:9;order:9}.order-xl-10{-webkit-box-ordinal-group:11;-ms-flex-order:10;order:10}.order-xl-11{-webkit-box-ordinal-group:12;-ms-flex-order:11;order:11}.order-xl-12{-webkit-box-ordinal-group:13;-ms-flex-order:12;order:12}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.3333333333%}.offset-xl-2{margin-left:16.6666666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.3333333333%}.offset-xl-5{margin-left:41.6666666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.3333333333%}.offset-xl-8{margin-left:66.6666666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.3333333333%}.offset-xl-11{margin-left:91.6666666667%}}.table{width:100%;margin-bottom:1rem;background-color:transparent}.table th,.table td{padding:0.75rem;vertical-align:top;border-top:1px solid #073642}.table thead th{vertical-align:bottom;border-bottom:2px solid #073642}.table tbody+tbody{border-top:2px solid #073642}.table .table{background-color:#002B36}.table-sm th,.table-sm td{padding:0.3rem}.table-bordered{border:1px solid #073642}.table-bordered th,.table-bordered td{border:1px solid #073642}.table-bordered thead th,.table-bordered thead td{border-bottom-width:2px}.table-borderless th,.table-borderless td,.table-borderless thead th,.table-borderless tbody+tbody{border:0}.table-striped tbody tr:nth-of-type(odd){background-color:rgba(255,255,255,0.05)}.table-hover tbody tr:hover{background-color:rgba(255,255,255,0.075)}.table-primary,.table-primary>th,.table-primary>td{background-color:#eadeb8}.table-primary th,.table-primary td,.table-primary thead th,.table-primary tbody+tbody{border-color:#d9c27a}.table-hover .table-primary:hover{background-color:#e4d5a4}.table-hover .table-primary:hover>td,.table-hover .table-primary:hover>th{background-color:#e4d5a4}.table-secondary,.table-secondary>th,.table-secondary>td{background-color:#dce1e2}.table-secondary th,.table-secondary td,.table-secondary thead th,.table-secondary tbody+tbody{border-color:#bfc7c8}.table-hover .table-secondary:hover{background-color:#ced5d6}.table-hover .table-secondary:hover>td,.table-hover .table-secondary:hover>th{background-color:#ced5d6}.table-success,.table-success>th,.table-success>td{background-color:#c3e5e2}.table-success th,.table-success td,.table-success thead th,.table-success tbody+tbody{border-color:#90cec9}.table-hover .table-success:hover{background-color:#b1ddd9}.table-hover .table-success:hover>td,.table-hover .table-success:hover>th{background-color:#b1ddd9}.table-info,.table-info>th,.table-info>td{background-color:#c2dff2}.table-info th,.table-info td,.table-info thead th,.table-info tbody+tbody{border-color:#8ec3e8}.table-hover .table-info:hover{background-color:#add4ee}.table-hover .table-info:hover>td,.table-hover .table-info:hover>th{background-color:#add4ee}.table-warning,.table-warning>th,.table-warning>td{background-color:#f0cdbe}.table-warning th,.table-warning td,.table-warning thead th,.table-warning tbody+tbody{border-color:#e4a186}.table-hover .table-warning:hover{background-color:#ebbda9}.table-hover .table-warning:hover>td,.table-hover .table-warning:hover>th{background-color:#ebbda9}.table-danger,.table-danger>th,.table-danger>td{background-color:#f3c7dc}.table-danger th,.table-danger td,.table-danger thead th,.table-danger tbody+tbody{border-color:#e896be}.table-hover .table-danger:hover{background-color:#efb2cf}.table-hover .table-danger:hover>td,.table-hover .table-danger:hover>th{background-color:#efb2cf}.table-light,.table-light>th,.table-light>td{background-color:#fefcf7}.table-light th,.table-light td,.table-light thead th,.table-light tbody+tbody{border-color:#fefaf0}.table-hover .table-light:hover{background-color:#fbf4e0}.table-hover .table-light:hover>td,.table-hover .table-light:hover>th{background-color:#fbf4e0}.table-dark,.table-dark>th,.table-dark>td{background-color:#bac7ca}.table-dark th,.table-dark td,.table-dark thead th,.table-dark tbody+tbody{border-color:#7e969d}.table-hover .table-dark:hover{background-color:#acbbbf}.table-hover .table-dark:hover>td,.table-hover .table-dark:hover>th{background-color:#acbbbf}.table-active,.table-active>th,.table-active>td{background-color:rgba(255,255,255,0.075)}.table-hover .table-active:hover{background-color:rgba(242,242,242,0.075)}.table-hover .table-active:hover>td,.table-hover .table-active:hover>th{background-color:rgba(242,242,242,0.075)}.table .thead-dark th{color:#002B36;background-color:#adb5bd;border-color:#a5adb6}.table .thead-light th{color:#495057;background-color:#EEE8D5;border-color:#073642}.table-dark{color:#002B36;background-color:#adb5bd}.table-dark th,.table-dark td,.table-dark thead th{border-color:#a5adb6}.table-dark.table-bordered{border:0}.table-dark.table-striped tbody tr:nth-of-type(odd){background-color:rgba(255,255,255,0.05)}.table-dark.table-hover tbody tr:hover{background-color:rgba(255,255,255,0.075)}@media (max-width: 575.98px){.table-responsive-sm{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-sm>.table-bordered{border:0}}@media (max-width: 767.98px){.table-responsive-md{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-md>.table-bordered{border:0}}@media (max-width: 991.98px){.table-responsive-lg{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-lg>.table-bordered{border:0}}@media (max-width: 1199.98px){.table-responsive-xl{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive-xl>.table-bordered{border:0}}.table-responsive{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch;-ms-overflow-style:-ms-autohiding-scrollbar}.table-responsive>.table-bordered{border:0}.form-control{display:block;width:100%;height:calc(2.25rem + 2px);padding:0.375rem 0.75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;background-color:#A9BDBD;background-clip:padding-box;border:1px solid rgba(0,0,0,0.15);border-radius:0.25rem;-webkit-transition:border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out}@media screen and (prefers-reduced-motion: reduce){.form-control{-webkit-transition:none;transition:none}}.form-control::-ms-expand{background-color:transparent;border:0}.form-control:focus{color:#495057;background-color:#A9BDBD;border-color:#1394b5;outline:0;-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25)}.form-control::-webkit-input-placeholder{color:#657B83;opacity:1}.form-control::-ms-input-placeholder{color:#657B83;opacity:1}.form-control::placeholder{color:#657B83;opacity:1}.form-control:disabled,.form-control[readonly]{background-color:#657B83;opacity:1}select.form-control:focus::-ms-value{color:#495057;background-color:#A9BDBD}.form-control-file,.form-control-range{display:block;width:100%}.col-form-label{padding-top:calc(0.375rem + 1px);padding-bottom:calc(0.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(0.5rem + 1px);padding-bottom:calc(0.5rem + 1px);font-size:1.25rem;line-height:1.5}.col-form-label-sm{padding-top:calc(0.25rem + 1px);padding-bottom:calc(0.25rem + 1px);font-size:0.875rem;line-height:1.5}.form-control-plaintext{display:block;width:100%;padding-top:0.375rem;padding-bottom:0.375rem;margin-bottom:0;line-height:1.5;color:#839496;background-color:transparent;border:solid transparent;border-width:1px 0}.form-control-plaintext.form-control-sm,.form-control-plaintext.form-control-lg{padding-right:0;padding-left:0}.form-control-sm{height:calc(1.8125rem + 2px);padding:0.25rem 0.5rem;font-size:0.875rem;line-height:1.5;border-radius:0.2rem}.form-control-lg{height:calc(2.875rem + 2px);padding:0.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:0.3rem}select.form-control[size],select.form-control[multiple]{height:auto}textarea.form-control{height:auto}.form-group{margin-bottom:1rem}.form-text{display:block;margin-top:0.25rem}.form-row{display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-5px;margin-left:-5px}.form-row>.col,.form-row>[class*="col-"]{padding-right:5px;padding-left:5px}.form-check{position:relative;display:block;padding-left:1.25rem}.form-check-input{position:absolute;margin-top:0.3rem;margin-left:-1.25rem}.form-check-input:disabled ~ .form-check-label{color:#839496}.form-check-label{margin-bottom:0}.form-check-inline{display:-webkit-inline-box;display:-ms-inline-flexbox;display:inline-flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;padding-left:0;margin-right:0.75rem}.form-check-inline .form-check-input{position:static;margin-top:0;margin-right:0.3125rem;margin-left:0}.valid-feedback{display:none;width:100%;margin-top:0.25rem;font-size:80%;color:#2AA198}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:0.25rem 0.5rem;margin-top:.1rem;font-size:0.875rem;line-height:1.5;color:#fff;background-color:rgba(42,161,152,0.9);border-radius:0.25rem}.was-validated .form-control:valid,.form-control.is-valid{border-color:#2AA198;padding-right:2.25rem;background-repeat:no-repeat;background-position:center right calc(2.25rem / 4);background-size:calc(2.25rem / 2) calc(2.25rem / 2);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%232AA198' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e")}.was-validated .form-control:valid:focus,.form-control.is-valid:focus{border-color:#2AA198;-webkit-box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25);box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25)}.was-validated .form-control:valid ~ .valid-feedback,.was-validated .form-control:valid ~ .valid-tooltip,.form-control.is-valid ~ .valid-feedback,.form-control.is-valid ~ .valid-tooltip{display:block}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:2.25rem;background-position:top calc(2.25rem / 4) right calc(2.25rem / 4)}.was-validated .custom-select:valid,.custom-select.is-valid{border-color:#2AA198;padding-right:3.4375rem;background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3e%3cpath fill='%23073642' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right 0.75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%232AA198' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e") no-repeat center right 1.75rem/1.125rem 1.125rem}.was-validated .custom-select:valid:focus,.custom-select.is-valid:focus{border-color:#2AA198;-webkit-box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25);box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25)}.was-validated .custom-select:valid ~ .valid-feedback,.was-validated .custom-select:valid ~ .valid-tooltip,.custom-select.is-valid ~ .valid-feedback,.custom-select.is-valid ~ .valid-tooltip{display:block}.was-validated .form-control-file:valid ~ .valid-feedback,.was-validated .form-control-file:valid ~ .valid-tooltip,.form-control-file.is-valid ~ .valid-feedback,.form-control-file.is-valid ~ .valid-tooltip{display:block}.was-validated .form-check-input:valid ~ .form-check-label,.form-check-input.is-valid ~ .form-check-label{color:#2AA198}.was-validated .form-check-input:valid ~ .valid-feedback,.was-validated .form-check-input:valid ~ .valid-tooltip,.form-check-input.is-valid ~ .valid-feedback,.form-check-input.is-valid ~ .valid-tooltip{display:block}.was-validated .custom-control-input:valid ~ .custom-control-label,.custom-control-input.is-valid ~ .custom-control-label{color:#2AA198}.was-validated .custom-control-input:valid ~ .custom-control-label::before,.custom-control-input.is-valid ~ .custom-control-label::before{border-color:#2AA198}.was-validated .custom-control-input:valid ~ .valid-feedback,.was-validated .custom-control-input:valid ~ .valid-tooltip,.custom-control-input.is-valid ~ .valid-feedback,.custom-control-input.is-valid ~ .valid-tooltip{display:block}.was-validated .custom-control-input:valid:checked ~ .custom-control-label::before,.custom-control-input.is-valid:checked ~ .custom-control-label::before{border-color:#35c9be;background:#35c9be -webkit-gradient(linear, left top, left bottom, from(#2db2aa), to(#35c9be)) repeat-x;background:#35c9be linear-gradient(180deg, #2db2aa, #35c9be) repeat-x}.was-validated .custom-control-input:valid:focus ~ .custom-control-label::before,.custom-control-input.is-valid:focus ~ .custom-control-label::before{-webkit-box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25);box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25)}.was-validated .custom-control-input:valid:focus:not(:checked) ~ .custom-control-label::before,.custom-control-input.is-valid:focus:not(:checked) ~ .custom-control-label::before{border-color:#2AA198}.was-validated .custom-file-input:valid ~ .custom-file-label,.custom-file-input.is-valid ~ .custom-file-label{border-color:#2AA198}.was-validated .custom-file-input:valid ~ .valid-feedback,.was-validated .custom-file-input:valid ~ .valid-tooltip,.custom-file-input.is-valid ~ .valid-feedback,.custom-file-input.is-valid ~ .valid-tooltip{display:block}.was-validated .custom-file-input:valid:focus ~ .custom-file-label,.custom-file-input.is-valid:focus ~ .custom-file-label{border-color:#2AA198;-webkit-box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25);box-shadow:0 0 0 0.2rem rgba(42,161,152,0.25)}.invalid-feedback{display:none;width:100%;margin-top:0.25rem;font-size:80%;color:#D33682}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:0.25rem 0.5rem;margin-top:.1rem;font-size:0.875rem;line-height:1.5;color:#fff;background-color:rgba(211,54,130,0.9);border-radius:0.25rem}.was-validated .form-control:invalid,.form-control.is-invalid{border-color:#D33682;padding-right:2.25rem;background-repeat:no-repeat;background-position:center right calc(2.25rem / 4);background-size:calc(2.25rem / 2) calc(2.25rem / 2);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23D33682' viewBox='-2 -2 7 7'%3e%3cpath stroke='%23d9534f' d='M0 0l3 3m0-3L0 3'/%3e%3ccircle r='.5'/%3e%3ccircle cx='3' r='.5'/%3e%3ccircle cy='3' r='.5'/%3e%3ccircle cx='3' cy='3' r='.5'/%3e%3c/svg%3E")}.was-validated .form-control:invalid:focus,.form-control.is-invalid:focus{border-color:#D33682;-webkit-box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25);box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25)}.was-validated .form-control:invalid ~ .invalid-feedback,.was-validated .form-control:invalid ~ .invalid-tooltip,.form-control.is-invalid ~ .invalid-feedback,.form-control.is-invalid ~ .invalid-tooltip{display:block}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:2.25rem;background-position:top calc(2.25rem / 4) right calc(2.25rem / 4)}.was-validated .custom-select:invalid,.custom-select.is-invalid{border-color:#D33682;padding-right:3.4375rem;background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3e%3cpath fill='%23073642' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right 0.75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23D33682' viewBox='-2 -2 7 7'%3e%3cpath stroke='%23d9534f' d='M0 0l3 3m0-3L0 3'/%3e%3ccircle r='.5'/%3e%3ccircle cx='3' r='.5'/%3e%3ccircle cy='3' r='.5'/%3e%3ccircle cx='3' cy='3' r='.5'/%3e%3c/svg%3E") no-repeat center right 1.75rem/1.125rem 1.125rem}.was-validated .custom-select:invalid:focus,.custom-select.is-invalid:focus{border-color:#D33682;-webkit-box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25);box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25)}.was-validated .custom-select:invalid ~ .invalid-feedback,.was-validated .custom-select:invalid ~ .invalid-tooltip,.custom-select.is-invalid ~ .invalid-feedback,.custom-select.is-invalid ~ .invalid-tooltip{display:block}.was-validated .form-control-file:invalid ~ .invalid-feedback,.was-validated .form-control-file:invalid ~ .invalid-tooltip,.form-control-file.is-invalid ~ .invalid-feedback,.form-control-file.is-invalid ~ .invalid-tooltip{display:block}.was-validated .form-check-input:invalid ~ .form-check-label,.form-check-input.is-invalid ~ .form-check-label{color:#D33682}.was-validated .form-check-input:invalid ~ .invalid-feedback,.was-validated .form-check-input:invalid ~ .invalid-tooltip,.form-check-input.is-invalid ~ .invalid-feedback,.form-check-input.is-invalid ~ .invalid-tooltip{display:block}.was-validated .custom-control-input:invalid ~ .custom-control-label,.custom-control-input.is-invalid ~ .custom-control-label{color:#D33682}.was-validated .custom-control-input:invalid ~ .custom-control-label::before,.custom-control-input.is-invalid ~ .custom-control-label::before{border-color:#D33682}.was-validated .custom-control-input:invalid ~ .invalid-feedback,.was-validated .custom-control-input:invalid ~ .invalid-tooltip,.custom-control-input.is-invalid ~ .invalid-feedback,.custom-control-input.is-invalid ~ .invalid-tooltip{display:block}.was-validated .custom-control-input:invalid:checked ~ .custom-control-label::before,.custom-control-input.is-invalid:checked ~ .custom-control-label::before{border-color:#dc609c;background:#dc609c -webkit-gradient(linear, left top, left bottom, from(#bb588d), to(#dc609c)) repeat-x;background:#dc609c linear-gradient(180deg, #bb588d, #dc609c) repeat-x}.was-validated .custom-control-input:invalid:focus ~ .custom-control-label::before,.custom-control-input.is-invalid:focus ~ .custom-control-label::before{-webkit-box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25);box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25)}.was-validated .custom-control-input:invalid:focus:not(:checked) ~ .custom-control-label::before,.custom-control-input.is-invalid:focus:not(:checked) ~ .custom-control-label::before{border-color:#D33682}.was-validated .custom-file-input:invalid ~ .custom-file-label,.custom-file-input.is-invalid ~ .custom-file-label{border-color:#D33682}.was-validated .custom-file-input:invalid ~ .invalid-feedback,.was-validated .custom-file-input:invalid ~ .invalid-tooltip,.custom-file-input.is-invalid ~ .invalid-feedback,.custom-file-input.is-invalid ~ .invalid-tooltip{display:block}.was-validated .custom-file-input:invalid:focus ~ .custom-file-label,.custom-file-input.is-invalid:focus ~ .custom-file-label{border-color:#D33682;-webkit-box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25);box-shadow:0 0 0 0.2rem rgba(211,54,130,0.25)}.form-inline{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row wrap;flex-flow:row wrap;-webkit-box-align:center;-ms-flex-align:center;align-items:center}.form-inline .form-check{width:100%}@media (min-width: 576px){.form-inline label{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center;margin-bottom:0}.form-inline .form-group{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-flex:0;-ms-flex:0 0 auto;flex:0 0 auto;-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row wrap;flex-flow:row wrap;-webkit-box-align:center;-ms-flex-align:center;align-items:center;margin-bottom:0}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-plaintext{display:inline-block}.form-inline .input-group,.form-inline .custom-select{width:auto}.form-inline .form-check{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center;width:auto;padding-left:0}.form-inline .form-check-input{position:relative;margin-top:0;margin-right:0.25rem;margin-left:0}.form-inline .custom-control{-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center}.form-inline .custom-control-label{margin-bottom:0}}.btn{display:inline-block;font-weight:400;color:#839496;text-align:center;vertical-align:middle;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:transparent;border:1px solid transparent;padding:0.375rem 0.75rem;font-size:1rem;line-height:1.5;border-radius:0.25rem;-webkit-transition:color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:color 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out}@media screen and (prefers-reduced-motion: reduce){.btn{-webkit-transition:none;transition:none}}.btn:hover{color:#839496;text-decoration:none}.btn:focus,.btn.focus{outline:0;-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25)}.btn.disabled,.btn:disabled{opacity:0.65}.btn:not(:disabled):not(.disabled){cursor:pointer}a.btn.disabled,fieldset:disabled a.btn{pointer-events:none}.btn-primary{color:#fff;background:#B58900 -webkit-gradient(linear, left top, left bottom, from(#9a7b08), to(#B58900)) repeat-x;background:#B58900 linear-gradient(180deg, #9a7b08, #B58900) repeat-x;border-color:#B58900}.btn-primary:hover{color:#fff;background:#8f6c00 -webkit-gradient(linear, left top, left bottom, from(#796208), to(#8f6c00)) repeat-x;background:#8f6c00 linear-gradient(180deg, #796208, #8f6c00) repeat-x;border-color:#826200}.btn-primary:focus,.btn-primary.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(192,155,38,0.5);box-shadow:0 0 0 0.2rem rgba(192,155,38,0.5)}.btn-primary.disabled,.btn-primary:disabled{color:#fff;background-color:#B58900;border-color:#B58900;background-image:none}.btn-primary:not(:disabled):not(.disabled):active,.btn-primary:not(:disabled):not(.disabled).active,.show>.btn-primary.dropdown-toggle{color:#fff;background-color:#826200;background-image:none;border-color:#755900}.btn-primary:not(:disabled):not(.disabled):active:focus,.btn-primary:not(:disabled):not(.disabled).active:focus,.show>.btn-primary.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(192,155,38,0.5);box-shadow:0 0 0 0.2rem rgba(192,155,38,0.5)}.btn-secondary{color:#fff;background:#839496 -webkit-gradient(linear, left top, left bottom, from(#6f8488), to(#839496)) repeat-x;background:#839496 linear-gradient(180deg, #6f8488, #839496) repeat-x;border-color:#839496}.btn-secondary:hover{color:#fff;background:#6f8183 -webkit-gradient(linear, left top, left bottom, from(#5f7478), to(#6f8183)) repeat-x;background:#6f8183 linear-gradient(180deg, #5f7478, #6f8183) repeat-x;border-color:#697b7d}.btn-secondary:focus,.btn-secondary.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(150,164,166,0.5);box-shadow:0 0 0 0.2rem rgba(150,164,166,0.5)}.btn-secondary.disabled,.btn-secondary:disabled{color:#fff;background-color:#839496;border-color:#839496;background-image:none}.btn-secondary:not(:disabled):not(.disabled):active,.btn-secondary:not(:disabled):not(.disabled).active,.show>.btn-secondary.dropdown-toggle{color:#fff;background-color:#697b7d;background-image:none;border-color:#647476}.btn-secondary:not(:disabled):not(.disabled):active:focus,.btn-secondary:not(:disabled):not(.disabled).active:focus,.show>.btn-secondary.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(150,164,166,0.5);box-shadow:0 0 0 0.2rem rgba(150,164,166,0.5)}.btn-success{color:#fff;background:#2AA198 -webkit-gradient(linear, left top, left bottom, from(#248f89), to(#2AA198)) repeat-x;background:#2AA198 linear-gradient(180deg, #248f89, #2AA198) repeat-x;border-color:#2AA198}.btn-success:hover{color:#fff;background:#22837b -webkit-gradient(linear, left top, left bottom, from(#1d7671), to(#22837b)) repeat-x;background:#22837b linear-gradient(180deg, #1d7671, #22837b) repeat-x;border-color:#1f7972}.btn-success:focus,.btn-success.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(74,175,167,0.5);box-shadow:0 0 0 0.2rem rgba(74,175,167,0.5)}.btn-success.disabled,.btn-success:disabled{color:#fff;background-color:#2AA198;border-color:#2AA198;background-image:none}.btn-success:not(:disabled):not(.disabled):active,.btn-success:not(:disabled):not(.disabled).active,.show>.btn-success.dropdown-toggle{color:#fff;background-color:#1f7972;background-image:none;border-color:#1d6e68}.btn-success:not(:disabled):not(.disabled):active:focus,.btn-success:not(:disabled):not(.disabled).active:focus,.show>.btn-success.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(74,175,167,0.5);box-shadow:0 0 0 0.2rem rgba(74,175,167,0.5)}.btn-info{color:#fff;background:#268BD2 -webkit-gradient(linear, left top, left bottom, from(#207dbb), to(#268BD2)) repeat-x;background:#268BD2 linear-gradient(180deg, #207dbb, #268BD2) repeat-x;border-color:#268BD2}.btn-info:hover{color:#fff;background:#2076b2 -webkit-gradient(linear, left top, left bottom, from(#1b6a9f), to(#2076b2)) repeat-x;background:#2076b2 linear-gradient(180deg, #1b6a9f, #2076b2) repeat-x;border-color:#1e6ea7}.btn-info:focus,.btn-info.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(71,156,217,0.5);box-shadow:0 0 0 0.2rem rgba(71,156,217,0.5)}.btn-info.disabled,.btn-info:disabled{color:#fff;background-color:#268BD2;border-color:#268BD2;background-image:none}.btn-info:not(:disabled):not(.disabled):active,.btn-info:not(:disabled):not(.disabled).active,.show>.btn-info.dropdown-toggle{color:#fff;background-color:#1e6ea7;background-image:none;border-color:#1c679c}.btn-info:not(:disabled):not(.disabled):active:focus,.btn-info:not(:disabled):not(.disabled).active:focus,.show>.btn-info.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(71,156,217,0.5);box-shadow:0 0 0 0.2rem rgba(71,156,217,0.5)}.btn-warning{color:#fff;background:#CB4B16 -webkit-gradient(linear, left top, left bottom, from(#ad461b), to(#CB4B16)) repeat-x;background:#CB4B16 linear-gradient(180deg, #ad461b, #CB4B16) repeat-x;border-color:#CB4B16}.btn-warning:hover{color:#fff;background:#a83e12 -webkit-gradient(linear, left top, left bottom, from(#8f3b18), to(#a83e12)) repeat-x;background:#a83e12 linear-gradient(180deg, #8f3b18, #a83e12) repeat-x;border-color:#9d3a11}.btn-warning:focus,.btn-warning.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(211,102,57,0.5);box-shadow:0 0 0 0.2rem rgba(211,102,57,0.5)}.btn-warning.disabled,.btn-warning:disabled{color:#fff;background-color:#CB4B16;border-color:#CB4B16;background-image:none}.btn-warning:not(:disabled):not(.disabled):active,.btn-warning:not(:disabled):not(.disabled).active,.show>.btn-warning.dropdown-toggle{color:#fff;background-color:#9d3a11;background-image:none;border-color:#913610}.btn-warning:not(:disabled):not(.disabled):active:focus,.btn-warning:not(:disabled):not(.disabled).active:focus,.show>.btn-warning.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(211,102,57,0.5);box-shadow:0 0 0 0.2rem rgba(211,102,57,0.5)}.btn-danger{color:#fff;background:#D33682 -webkit-gradient(linear, left top, left bottom, from(#b33477), to(#D33682)) repeat-x;background:#D33682 linear-gradient(180deg, #b33477, #D33682) repeat-x;border-color:#D33682}.btn-danger:hover{color:#fff;background:#ba296f -webkit-gradient(linear, left top, left bottom, from(#9e2967), to(#ba296f)) repeat-x;background:#ba296f linear-gradient(180deg, #9e2967, #ba296f) repeat-x;border-color:#b02669}.btn-danger:focus,.btn-danger.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(218,84,149,0.5);box-shadow:0 0 0 0.2rem rgba(218,84,149,0.5)}.btn-danger.disabled,.btn-danger:disabled{color:#fff;background-color:#D33682;border-color:#D33682;background-image:none}.btn-danger:not(:disabled):not(.disabled):active,.btn-danger:not(:disabled):not(.disabled).active,.show>.btn-danger.dropdown-toggle{color:#fff;background-color:#b02669;background-image:none;border-color:#a52463}.btn-danger:not(:disabled):not(.disabled):active:focus,.btn-danger:not(:disabled):not(.disabled).active:focus,.show>.btn-danger.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(218,84,149,0.5);box-shadow:0 0 0 0.2rem rgba(218,84,149,0.5)}.btn-light{color:#002B36;background:#FDF6E3 -webkit-gradient(linear, left top, left bottom, from(#d7d8c9), to(#FDF6E3)) repeat-x;background:#FDF6E3 linear-gradient(180deg, #d7d8c9, #FDF6E3) repeat-x;border-color:#FDF6E3}.btn-light:hover{color:#002B36;background:#faebbf -webkit-gradient(linear, left top, left bottom, from(#d5ceab), to(#faebbf)) repeat-x;background:#faebbf linear-gradient(180deg, #d5ceab, #faebbf) repeat-x;border-color:#fae7b3}.btn-light:focus,.btn-light.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(215,216,201,0.5);box-shadow:0 0 0 0.2rem rgba(215,216,201,0.5)}.btn-light.disabled,.btn-light:disabled{color:#002B36;background-color:#FDF6E3;border-color:#FDF6E3;background-image:none}.btn-light:not(:disabled):not(.disabled):active,.btn-light:not(:disabled):not(.disabled).active,.show>.btn-light.dropdown-toggle{color:#002B36;background-color:#fae7b3;background-image:none;border-color:#f9e3a8}.btn-light:not(:disabled):not(.disabled):active:focus,.btn-light:not(:disabled):not(.disabled).active:focus,.show>.btn-light.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(215,216,201,0.5);box-shadow:0 0 0 0.2rem rgba(215,216,201,0.5)}.btn-dark{color:#fff;background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x;border-color:#073642}.btn-dark:hover{color:#fff;background:#031a1f -webkit-gradient(linear, left top, left bottom, from(#031c23), to(#031a1f)) repeat-x;background:#031a1f linear-gradient(180deg, #031c23, #031a1f) repeat-x;border-color:#021014}.btn-dark:focus,.btn-dark.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(44,84,94,0.5);box-shadow:0 0 0 0.2rem rgba(44,84,94,0.5)}.btn-dark.disabled,.btn-dark:disabled{color:#fff;background-color:#073642;border-color:#073642;background-image:none}.btn-dark:not(:disabled):not(.disabled):active,.btn-dark:not(:disabled):not(.disabled).active,.show>.btn-dark.dropdown-toggle{color:#fff;background-color:#021014;background-image:none;border-color:#010708}.btn-dark:not(:disabled):not(.disabled):active:focus,.btn-dark:not(:disabled):not(.disabled).active:focus,.show>.btn-dark.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(44,84,94,0.5);box-shadow:0 0 0 0.2rem rgba(44,84,94,0.5)}.btn-outline-primary{color:#B58900;border-color:#B58900}.btn-outline-primary:hover{color:#fff;background-color:#B58900;border-color:#B58900}.btn-outline-primary:focus,.btn-outline-primary.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(181,137,0,0.5);box-shadow:0 0 0 0.2rem rgba(181,137,0,0.5)}.btn-outline-primary.disabled,.btn-outline-primary:disabled{color:#B58900;background-color:transparent}.btn-outline-primary:not(:disabled):not(.disabled):active,.btn-outline-primary:not(:disabled):not(.disabled).active,.show>.btn-outline-primary.dropdown-toggle{color:#fff;background-color:#B58900;border-color:#B58900}.btn-outline-primary:not(:disabled):not(.disabled):active:focus,.btn-outline-primary:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-primary.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(181,137,0,0.5);box-shadow:0 0 0 0.2rem rgba(181,137,0,0.5)}.btn-outline-secondary{color:#839496;border-color:#839496}.btn-outline-secondary:hover{color:#fff;background-color:#839496;border-color:#839496}.btn-outline-secondary:focus,.btn-outline-secondary.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(131,148,150,0.5);box-shadow:0 0 0 0.2rem rgba(131,148,150,0.5)}.btn-outline-secondary.disabled,.btn-outline-secondary:disabled{color:#839496;background-color:transparent}.btn-outline-secondary:not(:disabled):not(.disabled):active,.btn-outline-secondary:not(:disabled):not(.disabled).active,.show>.btn-outline-secondary.dropdown-toggle{color:#fff;background-color:#839496;border-color:#839496}.btn-outline-secondary:not(:disabled):not(.disabled):active:focus,.btn-outline-secondary:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-secondary.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(131,148,150,0.5);box-shadow:0 0 0 0.2rem rgba(131,148,150,0.5)}.btn-outline-success{color:#2AA198;border-color:#2AA198}.btn-outline-success:hover{color:#fff;background-color:#2AA198;border-color:#2AA198}.btn-outline-success:focus,.btn-outline-success.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(42,161,152,0.5);box-shadow:0 0 0 0.2rem rgba(42,161,152,0.5)}.btn-outline-success.disabled,.btn-outline-success:disabled{color:#2AA198;background-color:transparent}.btn-outline-success:not(:disabled):not(.disabled):active,.btn-outline-success:not(:disabled):not(.disabled).active,.show>.btn-outline-success.dropdown-toggle{color:#fff;background-color:#2AA198;border-color:#2AA198}.btn-outline-success:not(:disabled):not(.disabled):active:focus,.btn-outline-success:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-success.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(42,161,152,0.5);box-shadow:0 0 0 0.2rem rgba(42,161,152,0.5)}.btn-outline-info{color:#268BD2;border-color:#268BD2}.btn-outline-info:hover{color:#fff;background-color:#268BD2;border-color:#268BD2}.btn-outline-info:focus,.btn-outline-info.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(38,139,210,0.5);box-shadow:0 0 0 0.2rem rgba(38,139,210,0.5)}.btn-outline-info.disabled,.btn-outline-info:disabled{color:#268BD2;background-color:transparent}.btn-outline-info:not(:disabled):not(.disabled):active,.btn-outline-info:not(:disabled):not(.disabled).active,.show>.btn-outline-info.dropdown-toggle{color:#fff;background-color:#268BD2;border-color:#268BD2}.btn-outline-info:not(:disabled):not(.disabled):active:focus,.btn-outline-info:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-info.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(38,139,210,0.5);box-shadow:0 0 0 0.2rem rgba(38,139,210,0.5)}.btn-outline-warning{color:#CB4B16;border-color:#CB4B16}.btn-outline-warning:hover{color:#fff;background-color:#CB4B16;border-color:#CB4B16}.btn-outline-warning:focus,.btn-outline-warning.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(203,75,22,0.5);box-shadow:0 0 0 0.2rem rgba(203,75,22,0.5)}.btn-outline-warning.disabled,.btn-outline-warning:disabled{color:#CB4B16;background-color:transparent}.btn-outline-warning:not(:disabled):not(.disabled):active,.btn-outline-warning:not(:disabled):not(.disabled).active,.show>.btn-outline-warning.dropdown-toggle{color:#fff;background-color:#CB4B16;border-color:#CB4B16}.btn-outline-warning:not(:disabled):not(.disabled):active:focus,.btn-outline-warning:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-warning.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(203,75,22,0.5);box-shadow:0 0 0 0.2rem rgba(203,75,22,0.5)}.btn-outline-danger{color:#D33682;border-color:#D33682}.btn-outline-danger:hover{color:#fff;background-color:#D33682;border-color:#D33682}.btn-outline-danger:focus,.btn-outline-danger.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(211,54,130,0.5);box-shadow:0 0 0 0.2rem rgba(211,54,130,0.5)}.btn-outline-danger.disabled,.btn-outline-danger:disabled{color:#D33682;background-color:transparent}.btn-outline-danger:not(:disabled):not(.disabled):active,.btn-outline-danger:not(:disabled):not(.disabled).active,.show>.btn-outline-danger.dropdown-toggle{color:#fff;background-color:#D33682;border-color:#D33682}.btn-outline-danger:not(:disabled):not(.disabled):active:focus,.btn-outline-danger:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-danger.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(211,54,130,0.5);box-shadow:0 0 0 0.2rem rgba(211,54,130,0.5)}.btn-outline-light{color:#FDF6E3;border-color:#FDF6E3}.btn-outline-light:hover{color:#002B36;background-color:#FDF6E3;border-color:#FDF6E3}.btn-outline-light:focus,.btn-outline-light.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(253,246,227,0.5);box-shadow:0 0 0 0.2rem rgba(253,246,227,0.5)}.btn-outline-light.disabled,.btn-outline-light:disabled{color:#FDF6E3;background-color:transparent}.btn-outline-light:not(:disabled):not(.disabled):active,.btn-outline-light:not(:disabled):not(.disabled).active,.show>.btn-outline-light.dropdown-toggle{color:#002B36;background-color:#FDF6E3;border-color:#FDF6E3}.btn-outline-light:not(:disabled):not(.disabled):active:focus,.btn-outline-light:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-light.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(253,246,227,0.5);box-shadow:0 0 0 0.2rem rgba(253,246,227,0.5)}.btn-outline-dark{color:#073642;border-color:#073642}.btn-outline-dark:hover{color:#fff;background-color:#073642;border-color:#073642}.btn-outline-dark:focus,.btn-outline-dark.focus{-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.5);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.5)}.btn-outline-dark.disabled,.btn-outline-dark:disabled{color:#073642;background-color:transparent}.btn-outline-dark:not(:disabled):not(.disabled):active,.btn-outline-dark:not(:disabled):not(.disabled).active,.show>.btn-outline-dark.dropdown-toggle{color:#fff;background-color:#073642;border-color:#073642}.btn-outline-dark:not(:disabled):not(.disabled):active:focus,.btn-outline-dark:not(:disabled):not(.disabled).active:focus,.show>.btn-outline-dark.dropdown-toggle:focus{-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.5);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.5)}.btn-link{font-weight:400;color:#2AA198}.btn-link:hover{color:#2AA198;text-decoration:underline}.btn-link:focus,.btn-link.focus{text-decoration:underline;-webkit-box-shadow:none;box-shadow:none}.btn-link:disabled,.btn-link.disabled{color:#839496;pointer-events:none}.btn-lg,.btn-group-lg>.btn{padding:0.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:0.3rem}.btn-sm,.btn-group-sm>.btn{padding:0.25rem 0.5rem;font-size:0.875rem;line-height:1.5;border-radius:0.2rem}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:0.5rem}input[type="submit"].btn-block,input[type="reset"].btn-block,input[type="button"].btn-block{width:100%}.fade{-webkit-transition:opacity 0.15s linear;transition:opacity 0.15s linear}@media screen and (prefers-reduced-motion: reduce){.fade{-webkit-transition:none;transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{position:relative;height:0;overflow:hidden;-webkit-transition:height 0.35s ease;transition:height 0.35s ease}@media screen and (prefers-reduced-motion: reduce){.collapsing{-webkit-transition:none;transition:none}}.dropup,.dropright,.dropdown,.dropleft{position:relative}.dropdown-toggle::after{display:inline-block;margin-left:0.255em;vertical-align:0.255em;content:"";border-top:0.3em solid;border-right:0.3em solid transparent;border-bottom:0;border-left:0.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:10rem;padding:0.5rem 0;margin:0.125rem 0 0;font-size:1rem;color:#839496;text-align:left;list-style:none;background-color:#073642;background-clip:padding-box;border:1px solid rgba(0,0,0,0.15);border-radius:0.25rem}.dropdown-menu-right{right:0;left:auto}@media (min-width: 576px){.dropdown-menu-sm-right{right:0;left:auto}}@media (min-width: 768px){.dropdown-menu-md-right{right:0;left:auto}}@media (min-width: 992px){.dropdown-menu-lg-right{right:0;left:auto}}@media (min-width: 1200px){.dropdown-menu-xl-right{right:0;left:auto}}.dropdown-menu-left{right:auto;left:0}@media (min-width: 576px){.dropdown-menu-sm-left{right:auto;left:0}}@media (min-width: 768px){.dropdown-menu-md-left{right:auto;left:0}}@media (min-width: 992px){.dropdown-menu-lg-left{right:auto;left:0}}@media (min-width: 1200px){.dropdown-menu-xl-left{right:auto;left:0}}.dropup .dropdown-menu{top:auto;bottom:100%;margin-top:0;margin-bottom:0.125rem}.dropup .dropdown-toggle::after{display:inline-block;margin-left:0.255em;vertical-align:0.255em;content:"";border-top:0;border-right:0.3em solid transparent;border-bottom:0.3em solid;border-left:0.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-menu{top:0;right:auto;left:100%;margin-top:0;margin-left:0.125rem}.dropright .dropdown-toggle::after{display:inline-block;margin-left:0.255em;vertical-align:0.255em;content:"";border-top:0.3em solid transparent;border-right:0;border-bottom:0.3em solid transparent;border-left:0.3em solid}.dropright .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-toggle::after{vertical-align:0}.dropleft .dropdown-menu{top:0;right:100%;left:auto;margin-top:0;margin-right:0.125rem}.dropleft .dropdown-toggle::after{display:inline-block;margin-left:0.255em;vertical-align:0.255em;content:""}.dropleft .dropdown-toggle::after{display:none}.dropleft .dropdown-toggle::before{display:inline-block;margin-right:0.255em;vertical-align:0.255em;content:"";border-top:0.3em solid transparent;border-right:0.3em solid;border-bottom:0.3em solid transparent}.dropleft .dropdown-toggle:empty::after{margin-left:0}.dropleft .dropdown-toggle::before{vertical-align:0}.dropdown-menu[x-placement^="top"],.dropdown-menu[x-placement^="right"],.dropdown-menu[x-placement^="bottom"],.dropdown-menu[x-placement^="left"]{right:auto;bottom:auto}.dropdown-divider{height:0;margin:0.5rem 0;overflow:hidden;border-top:1px solid #002B36}.dropdown-item{display:block;width:100%;padding:0.25rem 1.5rem;clear:both;font-weight:400;color:#839496;text-align:inherit;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:first-child{border-top-left-radius:calc(0.25rem - 1px);border-top-right-radius:calc(0.25rem - 1px)}.dropdown-item:last-child{border-bottom-right-radius:calc(0.25rem - 1px);border-bottom-left-radius:calc(0.25rem - 1px)}.dropdown-item:hover,.dropdown-item:focus{color:rgba(255,255,255,0.75);text-decoration:none;background:#002B36 -webkit-gradient(linear, left top, left bottom, from(#002b36), to(#002B36)) repeat-x;background:#002B36 linear-gradient(180deg, #002b36, #002B36) repeat-x}.dropdown-item.active,.dropdown-item:active{color:rgba(255,255,255,0.75);text-decoration:none;background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x}.dropdown-item.disabled,.dropdown-item:disabled{color:#839496;pointer-events:none;background-color:transparent;background-image:none}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:0.5rem 1.5rem;margin-bottom:0;font-size:0.875rem;color:#839496;white-space:nowrap}.dropdown-item-text{display:block;padding:0.25rem 1.5rem;color:#839496}.btn-group,.btn-group-vertical{position:relative;display:-webkit-inline-box;display:-ms-inline-flexbox;display:inline-flex;vertical-align:middle}.btn-group>.btn,.btn-group-vertical>.btn{position:relative;-webkit-box-flex:1;-ms-flex:1 1 auto;flex:1 1 auto}.btn-group>.btn:hover,.btn-group-vertical>.btn:hover{z-index:1}.btn-group>.btn:focus,.btn-group>.btn:active,.btn-group>.btn.active,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn.active{z-index:1}.btn-toolbar{display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-webkit-box-pack:start;-ms-flex-pack:start;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group>.btn:not(:first-child),.btn-group>.btn-group:not(:first-child){margin-left:-1px}.btn-group>.btn:not(:last-child):not(.dropdown-toggle),.btn-group>.btn-group:not(:last-child)>.btn{border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn:not(:first-child),.btn-group>.btn-group:not(:first-child)>.btn{border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:0.5625rem;padding-left:0.5625rem}.dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after,.dropright .dropdown-toggle-split::after{margin-left:0}.dropleft .dropdown-toggle-split::before{margin-right:0}.btn-sm+.dropdown-toggle-split,.btn-group-sm>.btn+.dropdown-toggle-split{padding-right:0.375rem;padding-left:0.375rem}.btn-lg+.dropdown-toggle-split,.btn-group-lg>.btn+.dropdown-toggle-split{padding-right:0.75rem;padding-left:0.75rem}.btn-group-vertical{-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;-webkit-box-align:start;-ms-flex-align:start;align-items:flex-start;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn:not(:first-child),.btn-group-vertical>.btn-group:not(:first-child){margin-top:-1px}.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle),.btn-group-vertical>.btn-group:not(:last-child)>.btn{border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn:not(:first-child),.btn-group-vertical>.btn-group:not(:first-child)>.btn{border-top-left-radius:0;border-top-right-radius:0}.btn-group-toggle>.btn,.btn-group-toggle>.btn-group>.btn{margin-bottom:0}.btn-group-toggle>.btn input[type="radio"],.btn-group-toggle>.btn input[type="checkbox"],.btn-group-toggle>.btn-group>.btn input[type="radio"],.btn-group-toggle>.btn-group>.btn input[type="checkbox"]{position:absolute;clip:rect(0, 0, 0, 0);pointer-events:none}.input-group{position:relative;display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-webkit-box-align:stretch;-ms-flex-align:stretch;align-items:stretch;width:100%}.input-group>.form-control,.input-group>.form-control-plaintext,.input-group>.custom-select,.input-group>.custom-file{position:relative;-webkit-box-flex:1;-ms-flex:1 1 auto;flex:1 1 auto;width:1%;margin-bottom:0}.input-group>.form-control+.form-control,.input-group>.form-control+.custom-select,.input-group>.form-control+.custom-file,.input-group>.form-control-plaintext+.form-control,.input-group>.form-control-plaintext+.custom-select,.input-group>.form-control-plaintext+.custom-file,.input-group>.custom-select+.form-control,.input-group>.custom-select+.custom-select,.input-group>.custom-select+.custom-file,.input-group>.custom-file+.form-control,.input-group>.custom-file+.custom-select,.input-group>.custom-file+.custom-file{margin-left:-1px}.input-group>.form-control:focus,.input-group>.custom-select:focus,.input-group>.custom-file .custom-file-input:focus ~ .custom-file-label{z-index:3}.input-group>.custom-file .custom-file-input:focus{z-index:4}.input-group>.form-control:not(:last-child),.input-group>.custom-select:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.form-control:not(:first-child),.input-group>.custom-select:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.custom-file{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center}.input-group>.custom-file:not(:last-child) .custom-file-label,.input-group>.custom-file:not(:last-child) .custom-file-label::after{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-file:not(:first-child) .custom-file-label{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-prepend,.input-group-append{display:-webkit-box;display:-ms-flexbox;display:flex}.input-group-prepend .btn,.input-group-append .btn{position:relative;z-index:2}.input-group-prepend .btn:focus,.input-group-append .btn:focus{z-index:3}.input-group-prepend .btn+.btn,.input-group-prepend .btn+.input-group-text,.input-group-prepend .input-group-text+.input-group-text,.input-group-prepend .input-group-text+.btn,.input-group-append .btn+.btn,.input-group-append .btn+.input-group-text,.input-group-append .input-group-text+.input-group-text,.input-group-append .input-group-text+.btn{margin-left:-1px}.input-group-prepend{margin-right:-1px}.input-group-append{margin-left:-1px}.input-group-text{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;padding:0.375rem 0.75rem;margin-bottom:0;font-size:1rem;font-weight:400;line-height:1.5;color:#839496;text-align:center;white-space:nowrap;background-color:#073642;border:1px solid rgba(0,0,0,0.15);border-radius:0.25rem}.input-group-text input[type="radio"],.input-group-text input[type="checkbox"]{margin-top:0}.input-group-lg>.form-control:not(textarea),.input-group-lg>.custom-select{height:calc(2.875rem + 2px)}.input-group-lg>.form-control,.input-group-lg>.custom-select,.input-group-lg>.input-group-prepend>.input-group-text,.input-group-lg>.input-group-append>.input-group-text,.input-group-lg>.input-group-prepend>.btn,.input-group-lg>.input-group-append>.btn{padding:0.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:0.3rem}.input-group-sm>.form-control:not(textarea),.input-group-sm>.custom-select{height:calc(1.8125rem + 2px)}.input-group-sm>.form-control,.input-group-sm>.custom-select,.input-group-sm>.input-group-prepend>.input-group-text,.input-group-sm>.input-group-append>.input-group-text,.input-group-sm>.input-group-prepend>.btn,.input-group-sm>.input-group-append>.btn{padding:0.25rem 0.5rem;font-size:0.875rem;line-height:1.5;border-radius:0.2rem}.input-group-lg>.custom-select,.input-group-sm>.custom-select{padding-right:1.75rem}.input-group>.input-group-prepend>.btn,.input-group>.input-group-prepend>.input-group-text,.input-group>.input-group-append:not(:last-child)>.btn,.input-group>.input-group-append:not(:last-child)>.input-group-text,.input-group>.input-group-append:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group>.input-group-append:last-child>.input-group-text:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.input-group-append>.btn,.input-group>.input-group-append>.input-group-text,.input-group>.input-group-prepend:not(:first-child)>.btn,.input-group>.input-group-prepend:not(:first-child)>.input-group-text,.input-group>.input-group-prepend:first-child>.btn:not(:first-child),.input-group>.input-group-prepend:first-child>.input-group-text:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.custom-control{position:relative;display:block;min-height:1.5rem;padding-left:1.5rem}.custom-control-inline{display:-webkit-inline-box;display:-ms-inline-flexbox;display:inline-flex;margin-right:1rem}.custom-control-input{position:absolute;z-index:-1;opacity:0}.custom-control-input:checked ~ .custom-control-label::before{color:rgba(255,255,255,0.75);border-color:#073642;background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x}.custom-control-input:focus ~ .custom-control-label::before{-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25)}.custom-control-input:focus:not(:checked) ~ .custom-control-label::before{border-color:#1394b5}.custom-control-input:not(:disabled):active ~ .custom-control-label::before{color:rgba(255,255,255,0.75);background-color:#18bae3;border-color:#18bae3}.custom-control-input:disabled ~ .custom-control-label{color:#839496}.custom-control-input:disabled ~ .custom-control-label::before{background-color:#657B83}.custom-control-label{position:relative;margin-bottom:0;vertical-align:top}.custom-control-label::before{position:absolute;top:0.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;pointer-events:none;content:"";background-color:#A9BDBD;border:#adb5bd solid 1px}.custom-control-label::after{position:absolute;top:0.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;content:"";background-repeat:no-repeat;background-position:center center;background-size:50% 50%}.custom-checkbox .custom-control-label::before{border-radius:0.25rem}.custom-checkbox .custom-control-input:checked ~ .custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='rgba(255, 255, 255, 0.75)' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26 2.974 7.25 8 2.193z'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:indeterminate ~ .custom-control-label::before{border-color:#073642;background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x}.custom-checkbox .custom-control-input:indeterminate ~ .custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 4'%3e%3cpath stroke='rgba(255, 255, 255, 0.75)' d='M0 2h4'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:disabled:checked ~ .custom-control-label::before{background-color:rgba(181,137,0,0.5)}.custom-checkbox .custom-control-input:disabled:indeterminate ~ .custom-control-label::before{background-color:rgba(181,137,0,0.5)}.custom-radio .custom-control-label::before{border-radius:50%}.custom-radio .custom-control-input:checked ~ .custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba(255, 255, 255, 0.75)'/%3e%3c/svg%3e")}.custom-radio .custom-control-input:disabled:checked ~ .custom-control-label::before{background-color:rgba(181,137,0,0.5)}.custom-switch{padding-left:2.25rem}.custom-switch .custom-control-label::before{left:-2.25rem;width:1.75rem;pointer-events:all;border-radius:0.5rem}.custom-switch .custom-control-label::after{top:calc(0.25rem + 2px);left:calc(-2.25rem + 2px);width:calc(1rem - 4px);height:calc(1rem - 4px);background-color:#adb5bd;border-radius:0.5rem;-webkit-transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-transform 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-transform 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:transform 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:transform 0.15s ease-in-out, background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-transform 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out}@media screen and (prefers-reduced-motion: reduce){.custom-switch .custom-control-label::after{-webkit-transition:none;transition:none}}.custom-switch .custom-control-input:checked ~ .custom-control-label::after{background-color:#A9BDBD;-webkit-transform:translateX(0.75rem);transform:translateX(0.75rem)}.custom-switch .custom-control-input:disabled:checked ~ .custom-control-label::before{background-color:rgba(181,137,0,0.5)}.custom-select{display:inline-block;width:100%;height:calc(2.25rem + 2px);padding:0.375rem 1.75rem 0.375rem 0.75rem;font-weight:400;line-height:1.5;color:#495057;vertical-align:middle;background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 4 5'%3e%3cpath fill='%23073642' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right 0.75rem center/8px 10px;background-color:#A9BDBD;border:1px solid rgba(0,0,0,0.15);border-radius:0.25rem;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-select:focus{border-color:#1394b5;outline:0;-webkit-box-shadow:0 0 0 0.2rem rgba(19,148,181,0.5);box-shadow:0 0 0 0.2rem rgba(19,148,181,0.5)}.custom-select:focus::-ms-value{color:#495057;background-color:#A9BDBD}.custom-select[multiple],.custom-select[size]:not([size="1"]){height:auto;padding-right:0.75rem;background-image:none}.custom-select:disabled{color:#839496;background-color:#EEE8D5}.custom-select::-ms-expand{opacity:0}.custom-select-sm{height:calc(1.8125rem + 2px);padding-top:0.25rem;padding-bottom:0.25rem;padding-left:0.5rem;font-size:0.875rem}.custom-select-lg{height:calc(2.875rem + 2px);padding-top:0.5rem;padding-bottom:0.5rem;padding-left:1rem;font-size:1.25rem}.custom-file{position:relative;display:inline-block;width:100%;height:calc(2.25rem + 2px);margin-bottom:0}.custom-file-input{position:relative;z-index:2;width:100%;height:calc(2.25rem + 2px);margin:0;opacity:0}.custom-file-input:focus ~ .custom-file-label{border-color:#1394b5;-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25)}.custom-file-input:disabled ~ .custom-file-label{background-color:#657B83}.custom-file-input:lang(en) ~ .custom-file-label::after{content:"Browse"}.custom-file-input ~ .custom-file-label[data-browse]::after{content:attr(data-browse)}.custom-file-label{position:absolute;top:0;right:0;left:0;z-index:1;height:calc(2.25rem + 2px);padding:0.375rem 0.75rem;font-weight:400;line-height:1.5;color:#495057;background-color:#A9BDBD;border:1px solid rgba(0,0,0,0.15);border-radius:0.25rem}.custom-file-label::after{position:absolute;top:0;right:0;bottom:0;z-index:3;display:block;height:2.25rem;padding:0.375rem 0.75rem;line-height:1.5;color:#495057;content:"Browse";background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x;border-left:inherit;border-radius:0 0.25rem 0.25rem 0}.custom-range{width:100%;height:calc(1rem + 0.4rem);padding:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-range:focus{outline:none}.custom-range:focus::-webkit-slider-thumb{-webkit-box-shadow:0 0 0 1px #002B36,0 0 0 0.2rem rgba(7,54,66,0.25);box-shadow:0 0 0 1px #002B36,0 0 0 0.2rem rgba(7,54,66,0.25)}.custom-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #002B36,0 0 0 0.2rem rgba(7,54,66,0.25)}.custom-range:focus::-ms-thumb{box-shadow:0 0 0 1px #002B36,0 0 0 0.2rem rgba(7,54,66,0.25)}.custom-range::-moz-focus-outer{border:0}.custom-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-0.25rem;background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x;border:0;border-radius:1rem;-webkit-transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;-webkit-appearance:none;appearance:none}@media screen and (prefers-reduced-motion: reduce){.custom-range::-webkit-slider-thumb{-webkit-transition:none;transition:none}}.custom-range::-webkit-slider-thumb:active{background:#18bae3 -webkit-gradient(linear, left top, left bottom, from(#14a5c9), to(#18bae3)) repeat-x;background:#18bae3 linear-gradient(180deg, #14a5c9, #18bae3) repeat-x}.custom-range::-webkit-slider-runnable-track{width:100%;height:0.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-moz-range-thumb{width:1rem;height:1rem;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x;border:0;border-radius:1rem;-webkit-transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;-moz-appearance:none;appearance:none}@media screen and (prefers-reduced-motion: reduce){.custom-range::-moz-range-thumb{-webkit-transition:none;transition:none}}.custom-range::-moz-range-thumb:active{background:#18bae3 linear-gradient(180deg, #14a5c9, #18bae3) repeat-x}.custom-range::-moz-range-track{width:100%;height:0.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-ms-thumb{width:1rem;height:1rem;margin-top:0;margin-right:0.2rem;margin-left:0.2rem;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x;border:0;border-radius:1rem;-webkit-transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;appearance:none}@media screen and (prefers-reduced-motion: reduce){.custom-range::-ms-thumb{-webkit-transition:none;transition:none}}.custom-range::-ms-thumb:active{background:#18bae3 linear-gradient(180deg, #14a5c9, #18bae3) repeat-x}.custom-range::-ms-track{width:100%;height:0.5rem;color:transparent;cursor:pointer;background-color:transparent;border-color:transparent;border-width:0.5rem}.custom-range::-ms-fill-lower{background-color:#dee2e6;border-radius:1rem}.custom-range::-ms-fill-upper{margin-right:15px;background-color:#dee2e6;border-radius:1rem}.custom-range:disabled::-webkit-slider-thumb{background-color:#adb5bd}.custom-range:disabled::-webkit-slider-runnable-track{cursor:default}.custom-range:disabled::-moz-range-thumb{background-color:#adb5bd}.custom-range:disabled::-moz-range-track{cursor:default}.custom-range:disabled::-ms-thumb{background-color:#adb5bd}.custom-control-label::before,.custom-file-label,.custom-select{-webkit-transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;transition:background-color 0.15s ease-in-out, border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out, -webkit-box-shadow 0.15s ease-in-out}@media screen and (prefers-reduced-motion: reduce){.custom-control-label::before,.custom-file-label,.custom-select{-webkit-transition:none;transition:none}}.nav{display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:0.5rem 1rem}.nav-link:hover,.nav-link:focus{text-decoration:none}.nav-link.disabled{color:#839496;pointer-events:none;cursor:default}.nav-tabs{border-bottom:1px solid #073642}.nav-tabs .nav-item{margin-bottom:-1px}.nav-tabs .nav-link{border:1px solid transparent;border-top-left-radius:0.25rem;border-top-right-radius:0.25rem}.nav-tabs .nav-link:hover,.nav-tabs .nav-link:focus{border-color:#073642}.nav-tabs .nav-link.disabled{color:#839496;background-color:transparent;border-color:transparent}.nav-tabs .nav-link.active,.nav-tabs .nav-item.show .nav-link{color:rgba(255,255,255,0.75);background-color:#002B36;border-color:#073642}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.nav-pills .nav-link{border-radius:0.25rem}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:rgba(255,255,255,0.75);background-color:#073642}.nav-fill .nav-item{-webkit-box-flex:1;-ms-flex:1 1 auto;flex:1 1 auto;text-align:center}.nav-justified .nav-item{-ms-flex-preferred-size:0;flex-basis:0;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;text-align:center}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{position:relative;display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:justify;-ms-flex-pack:justify;justify-content:space-between;padding:0.5rem 1rem}.navbar>.container,.navbar>.container-fluid{display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:justify;-ms-flex-pack:justify;justify-content:space-between}.navbar-brand{display:inline-block;padding-top:0.3125rem;padding-bottom:0.3125rem;margin-right:1rem;font-size:1.25rem;line-height:inherit;white-space:nowrap}.navbar-brand:hover,.navbar-brand:focus{text-decoration:none}.navbar-nav{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link{padding-right:0;padding-left:0}.navbar-nav .dropdown-menu{position:static;float:none}.navbar-text{display:inline-block;padding-top:0.5rem;padding-bottom:0.5rem}.navbar-collapse{-ms-flex-preferred-size:100%;flex-basis:100%;-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;-webkit-box-align:center;-ms-flex-align:center;align-items:center}.navbar-toggler{padding:0.25rem 0.75rem;font-size:1.25rem;line-height:1;background-color:transparent;border:1px solid transparent;border-radius:0.25rem}.navbar-toggler:hover,.navbar-toggler:focus{text-decoration:none}.navbar-toggler:not(:disabled):not(.disabled){cursor:pointer}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;content:"";background:no-repeat center center;background-size:100% 100%}@media (max-width: 575.98px){.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid{padding-right:0;padding-left:0}}@media (min-width: 576px){.navbar-expand-sm{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-webkit-box-pack:start;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-sm .navbar-nav{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-direction:row;flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:0.5rem;padding-left:0.5rem}.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-sm .navbar-collapse{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}}@media (max-width: 767.98px){.navbar-expand-md>.container,.navbar-expand-md>.container-fluid{padding-right:0;padding-left:0}}@media (min-width: 768px){.navbar-expand-md{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-webkit-box-pack:start;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-md .navbar-nav{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-direction:row;flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:0.5rem;padding-left:0.5rem}.navbar-expand-md>.container,.navbar-expand-md>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-md .navbar-collapse{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}}@media (max-width: 991.98px){.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid{padding-right:0;padding-left:0}}@media (min-width: 992px){.navbar-expand-lg{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-webkit-box-pack:start;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-lg .navbar-nav{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-direction:row;flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:0.5rem;padding-left:0.5rem}.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-lg .navbar-collapse{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}}@media (max-width: 1199.98px){.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid{padding-right:0;padding-left:0}}@media (min-width: 1200px){.navbar-expand-xl{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-webkit-box-pack:start;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-xl .navbar-nav{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-direction:row;flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:0.5rem;padding-left:0.5rem}.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-xl .navbar-collapse{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}}.navbar-expand{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-webkit-box-pack:start;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand>.container,.navbar-expand>.container-fluid{padding-right:0;padding-left:0}.navbar-expand .navbar-nav{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-direction:row;flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:0.5rem;padding-left:0.5rem}.navbar-expand>.container,.navbar-expand>.container-fluid{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand .navbar-collapse{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-light .navbar-brand{color:rgba(0,0,0,0.7)}.navbar-light .navbar-brand:hover,.navbar-light .navbar-brand:focus{color:rgba(0,0,0,0.7)}.navbar-light .navbar-nav .nav-link{color:rgba(0,0,0,0.4)}.navbar-light .navbar-nav .nav-link:hover,.navbar-light .navbar-nav .nav-link:focus{color:rgba(0,0,0,0.7)}.navbar-light .navbar-nav .nav-link.disabled{color:rgba(0,0,0,0.3)}.navbar-light .navbar-nav .show>.nav-link,.navbar-light .navbar-nav .active>.nav-link,.navbar-light .navbar-nav .nav-link.show,.navbar-light .navbar-nav .nav-link.active{color:rgba(0,0,0,0.7)}.navbar-light .navbar-toggler{color:rgba(0,0,0,0.4);border-color:rgba(0,0,0,0.1)}.navbar-light .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg viewBox='0 0 30 30' xmlns='http://www.w3.org/2000/svg'%3e%3cpath stroke='rgba(0, 0, 0, 0.4)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-light .navbar-text{color:rgba(0,0,0,0.4)}.navbar-light .navbar-text a{color:rgba(0,0,0,0.7)}.navbar-light .navbar-text a:hover,.navbar-light .navbar-text a:focus{color:rgba(0,0,0,0.7)}.navbar-dark .navbar-brand{color:#fff}.navbar-dark .navbar-brand:hover,.navbar-dark .navbar-brand:focus{color:#fff}.navbar-dark .navbar-nav .nav-link{color:rgba(255,255,255,0.5)}.navbar-dark .navbar-nav .nav-link:hover,.navbar-dark .navbar-nav .nav-link:focus{color:rgba(255,255,255,0.75)}.navbar-dark .navbar-nav .nav-link.disabled{color:rgba(255,255,255,0.25)}.navbar-dark .navbar-nav .show>.nav-link,.navbar-dark .navbar-nav .active>.nav-link,.navbar-dark .navbar-nav .nav-link.show,.navbar-dark .navbar-nav .nav-link.active{color:#fff}.navbar-dark .navbar-toggler{color:rgba(255,255,255,0.5);border-color:rgba(255,255,255,0.1)}.navbar-dark .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg viewBox='0 0 30 30' xmlns='http://www.w3.org/2000/svg'%3e%3cpath stroke='rgba(255, 255, 255, 0.5)' stroke-width='2' stroke-linecap='round' stroke-miterlimit='10' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-dark .navbar-text{color:rgba(255,255,255,0.5)}.navbar-dark .navbar-text a{color:#fff}.navbar-dark .navbar-text a:hover,.navbar-dark .navbar-text a:focus{color:#fff}.card{position:relative;display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;min-width:0;word-wrap:break-word;background-color:rgba(238,232,213,0.125);background-clip:border-box;border:1px solid rgba(0,43,54,0.95);border-radius:0.25rem}.card>hr{margin-right:0;margin-left:0}.card>.list-group:first-child .list-group-item:first-child{border-top-left-radius:0.25rem;border-top-right-radius:0.25rem}.card>.list-group:last-child .list-group-item:last-child{border-bottom-right-radius:0.25rem;border-bottom-left-radius:0.25rem}.card-body{-webkit-box-flex:1;-ms-flex:1 1 auto;flex:1 1 auto;padding:1.25rem}.card-title{margin-bottom:0.75rem}.card-subtitle{margin-top:-0.375rem;margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link:hover{text-decoration:none}.card-link+.card-link{margin-left:1.25rem}.card-header{padding:0.75rem 1.25rem;margin-bottom:0;color:inherit;background-color:rgba(7,54,66,0.25);border-bottom:1px solid rgba(0,43,54,0.95)}.card-header:first-child{border-radius:calc(0.25rem - 1px) calc(0.25rem - 1px) 0 0}.card-header+.list-group .list-group-item:first-child{border-top:0}.card-footer{padding:0.75rem 1.25rem;background-color:rgba(7,54,66,0.25);border-top:1px solid rgba(0,43,54,0.95)}.card-footer:last-child{border-radius:0 0 calc(0.25rem - 1px) calc(0.25rem - 1px)}.card-header-tabs{margin-right:-0.625rem;margin-bottom:-0.75rem;margin-left:-0.625rem;border-bottom:0}.card-header-pills{margin-right:-0.625rem;margin-left:-0.625rem}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:1.25rem}.card-img{width:100%;border-radius:calc(0.25rem - 1px)}.card-img-top{width:100%;border-top-left-radius:calc(0.25rem - 1px);border-top-right-radius:calc(0.25rem - 1px)}.card-img-bottom{width:100%;border-bottom-right-radius:calc(0.25rem - 1px);border-bottom-left-radius:calc(0.25rem - 1px)}.card-deck{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column}.card-deck .card{margin-bottom:15px}@media (min-width: 576px){.card-deck{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row wrap;flex-flow:row wrap;margin-right:-15px;margin-left:-15px}.card-deck .card{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-flex:1;-ms-flex:1 0 0%;flex:1 0 0%;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;margin-right:15px;margin-bottom:0;margin-left:15px}}.card-group{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column}.card-group>.card{margin-bottom:15px}@media (min-width: 576px){.card-group{-webkit-box-orient:horizontal;-webkit-box-direction:normal;-ms-flex-flow:row wrap;flex-flow:row wrap}.card-group>.card{-webkit-box-flex:1;-ms-flex:1 0 0%;flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:first-child{border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:first-child .card-img-top,.card-group>.card:first-child .card-header{border-top-right-radius:0}.card-group>.card:first-child .card-img-bottom,.card-group>.card:first-child .card-footer{border-bottom-right-radius:0}.card-group>.card:last-child{border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:last-child .card-img-top,.card-group>.card:last-child .card-header{border-top-left-radius:0}.card-group>.card:last-child .card-img-bottom,.card-group>.card:last-child .card-footer{border-bottom-left-radius:0}.card-group>.card:only-child{border-radius:0.25rem}.card-group>.card:only-child .card-img-top,.card-group>.card:only-child .card-header{border-top-left-radius:0.25rem;border-top-right-radius:0.25rem}.card-group>.card:only-child .card-img-bottom,.card-group>.card:only-child .card-footer{border-bottom-right-radius:0.25rem;border-bottom-left-radius:0.25rem}.card-group>.card:not(:first-child):not(:last-child):not(:only-child){border-radius:0}.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-img-top,.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-img-bottom,.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-header,.card-group>.card:not(:first-child):not(:last-child):not(:only-child) .card-footer{border-radius:0}}.card-columns .card{margin-bottom:0.75rem}@media (min-width: 576px){.card-columns{-webkit-column-count:3;column-count:3;-webkit-column-gap:1.25rem;column-gap:1.25rem;orphans:1;widows:1}.card-columns .card{display:inline-block;width:100%}}.accordion .card{overflow:hidden}.accordion .card:not(:first-of-type) .card-header:first-child{border-radius:0}.accordion .card:not(:first-of-type):not(:last-of-type){border-bottom:0;border-radius:0}.accordion .card:first-of-type{border-bottom:0;border-bottom-right-radius:0;border-bottom-left-radius:0}.accordion .card:last-of-type{border-top-left-radius:0;border-top-right-radius:0}.accordion .card .card-header{margin-bottom:-1px}.breadcrumb{display:-webkit-box;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding:0.75rem 1rem;margin-bottom:1rem;list-style:none;background-color:#073642;border-radius:0.25rem}.breadcrumb-item+.breadcrumb-item{padding-left:0.5rem}.breadcrumb-item+.breadcrumb-item::before{display:inline-block;padding-right:0.5rem;color:#839496;content:"/"}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:underline}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:none}.breadcrumb-item.active{color:#839496}.pagination{display:-webkit-box;display:-ms-flexbox;display:flex;padding-left:0;list-style:none;border-radius:0.25rem}.page-link{position:relative;display:block;padding:0.5rem 0.75rem;margin-left:-1px;line-height:1.25;color:#2AA198;background-color:transparent;border:1px solid #073642}.page-link:hover{z-index:2;color:#2AA198;text-decoration:none;background-color:#073642;border-color:#073642}.page-link:focus{z-index:2;outline:0;-webkit-box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25);box-shadow:0 0 0 0.2rem rgba(7,54,66,0.25)}.page-link:not(:disabled):not(.disabled){cursor:pointer}.page-item:first-child .page-link{margin-left:0;border-top-left-radius:0.25rem;border-bottom-left-radius:0.25rem}.page-item:last-child .page-link{border-top-right-radius:0.25rem;border-bottom-right-radius:0.25rem}.page-item.active .page-link{z-index:1;color:rgba(255,255,255,0.75);background-color:#073642;border-color:#073642}.page-item.disabled .page-link{color:#073642;pointer-events:none;cursor:auto;background-color:transparent;border-color:#073642}.pagination-lg .page-link{padding:0.75rem 1.5rem;font-size:1.25rem;line-height:1.5}.pagination-lg .page-item:first-child .page-link{border-top-left-radius:0.3rem;border-bottom-left-radius:0.3rem}.pagination-lg .page-item:last-child .page-link{border-top-right-radius:0.3rem;border-bottom-right-radius:0.3rem}.pagination-sm .page-link{padding:0.25rem 0.5rem;font-size:0.875rem;line-height:1.5}.pagination-sm .page-item:first-child .page-link{border-top-left-radius:0.2rem;border-bottom-left-radius:0.2rem}.pagination-sm .page-item:last-child .page-link{border-top-right-radius:0.2rem;border-bottom-right-radius:0.2rem}.badge{display:inline-block;padding:0.25em 0.4em;font-size:75%;font-weight:700;line-height:1;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:0.25rem}a.badge:hover,a.badge:focus{text-decoration:none}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.badge-pill{padding-right:0.6em;padding-left:0.6em;border-radius:10rem}.badge-primary{color:#fff;background-color:#B58900}a.badge-primary:hover,a.badge-primary:focus{color:#fff;background-color:#826200}.badge-secondary{color:#fff;background-color:#839496}a.badge-secondary:hover,a.badge-secondary:focus{color:#fff;background-color:#697b7d}.badge-success{color:#fff;background-color:#2AA198}a.badge-success:hover,a.badge-success:focus{color:#fff;background-color:#1f7972}.badge-info{color:#fff;background-color:#268BD2}a.badge-info:hover,a.badge-info:focus{color:#fff;background-color:#1e6ea7}.badge-warning{color:#fff;background-color:#CB4B16}a.badge-warning:hover,a.badge-warning:focus{color:#fff;background-color:#9d3a11}.badge-danger{color:#fff;background-color:#D33682}a.badge-danger:hover,a.badge-danger:focus{color:#fff;background-color:#b02669}.badge-light{color:#002B36;background-color:#FDF6E3}a.badge-light:hover,a.badge-light:focus{color:#002B36;background-color:#fae7b3}.badge-dark{color:#fff;background-color:#073642}a.badge-dark:hover,a.badge-dark:focus{color:#fff;background-color:#021014}.jumbotron{padding:2rem 1rem;margin-bottom:2rem;background-color:#073642;border-radius:0.3rem}@media (min-width: 576px){.jumbotron{padding:4rem 2rem}}.jumbotron-fluid{padding-right:0;padding-left:0;border-radius:0}.alert{position:relative;padding:0.75rem 1.25rem;margin-bottom:1rem;border:1px solid transparent;border-radius:0.25rem}.alert-heading{color:inherit}.alert-link{font-weight:700}.alert-dismissible{padding-right:4rem}.alert-dismissible .close{position:absolute;top:0;right:0;padding:0.75rem 1.25rem;color:inherit}.alert-primary{color:#5e4700;background:#f0e7cc -webkit-gradient(linear, left top, left bottom, from(#cccbb6), to(#f0e7cc)) repeat-x;background:#f0e7cc linear-gradient(180deg, #cccbb6, #f0e7cc) repeat-x;border-color:#eadeb8}.alert-primary hr{border-top-color:#e4d5a4}.alert-primary .alert-link{color:#2b2000}.alert-secondary{color:#444d4e;background:#e6eaea -webkit-gradient(linear, left top, left bottom, from(#c4cdcf), to(#e6eaea)) repeat-x;background:#e6eaea linear-gradient(180deg, #c4cdcf, #e6eaea) repeat-x;border-color:#dce1e2}.alert-secondary hr{border-top-color:#ced5d6}.alert-secondary .alert-link{color:#2c3233}.alert-success{color:#16544f;background:#d4ecea -webkit-gradient(linear, left top, left bottom, from(#b4cfcf), to(#d4ecea)) repeat-x;background:#d4ecea linear-gradient(180deg, #b4cfcf, #d4ecea) repeat-x;border-color:#c3e5e2}.alert-success hr{border-top-color:#b1ddd9}.alert-success .alert-link{color:#0b2c29}.alert-info{color:#14486d;background:#d4e8f6 -webkit-gradient(linear, left top, left bottom, from(#b4ccd9), to(#d4e8f6)) repeat-x;background:#d4e8f6 linear-gradient(180deg, #b4ccd9, #d4e8f6) repeat-x;border-color:#c2dff2}.alert-info hr{border-top-color:#add4ee}.alert-info .alert-link{color:#0c2c42}.alert-warning{color:#6a270b;background:#f5dbd0 -webkit-gradient(linear, left top, left bottom, from(#d0c1b9), to(#f5dbd0)) repeat-x;background:#f5dbd0 linear-gradient(180deg, #d0c1b9, #f5dbd0) repeat-x;border-color:#f0cdbe}.alert-warning hr{border-top-color:#ebbda9}.alert-warning .alert-link{color:#3c1606}.alert-danger{color:#6e1c44;background:#f6d7e6 -webkit-gradient(linear, left top, left bottom, from(#d1bdcc), to(#f6d7e6)) repeat-x;background:#f6d7e6 linear-gradient(180deg, #d1bdcc, #f6d7e6) repeat-x;border-color:#f3c7dc}.alert-danger hr{border-top-color:#efb2cf}.alert-danger .alert-link{color:#45122b}.alert-light{color:#848076;background:#fffdf9 -webkit-gradient(linear, left top, left bottom, from(#d9dedc), to(#fffdf9)) repeat-x;background:#fffdf9 linear-gradient(180deg, #d9dedc, #fffdf9) repeat-x;border-color:#fefcf7}.alert-light hr{border-top-color:#fbf4e0}.alert-light .alert-link{color:#69665e}.alert-dark{color:#041c22;background:#cdd7d9 -webkit-gradient(linear, left top, left bottom, from(#aebdc1), to(#cdd7d9)) repeat-x;background:#cdd7d9 linear-gradient(180deg, #aebdc1, #cdd7d9) repeat-x;border-color:#bac7ca}.alert-dark hr{border-top-color:#acbbbf}.alert-dark .alert-link{color:black}@-webkit-keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}.progress{display:-webkit-box;display:-ms-flexbox;display:flex;height:1rem;overflow:hidden;font-size:0.75rem;background-color:#073642;border-radius:0.25rem}.progress-bar{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center;color:#B58900;text-align:center;white-space:nowrap;background-color:#B58900;-webkit-transition:width 0.6s ease;transition:width 0.6s ease}@media screen and (prefers-reduced-motion: reduce){.progress-bar{-webkit-transition:none;transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg, rgba(255,255,255,0.15) 25%, transparent 25%, transparent 50%, rgba(255,255,255,0.15) 50%, rgba(255,255,255,0.15) 75%, transparent 75%, transparent);background-size:1rem 1rem}.progress-bar-animated{-webkit-animation:progress-bar-stripes 1s linear infinite;animation:progress-bar-stripes 1s linear infinite}.media{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:start;-ms-flex-align:start;align-items:flex-start}.media-body{-webkit-box-flex:1;-ms-flex:1;flex:1}.list-group{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0}.list-group-item-action{width:100%;color:#839496;text-align:inherit}.list-group-item-action:hover,.list-group-item-action:focus{color:rgba(255,255,255,0.75);text-decoration:none;background-color:#073642}.list-group-item-action:active{color:#839496;background-color:#EEE8D5}.list-group-item{position:relative;display:block;padding:0.75rem 1.25rem;margin-bottom:-1px;background-color:transparent;border:1px solid #073642}.list-group-item:first-child{border-top-left-radius:0.25rem;border-top-right-radius:0.25rem}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:0.25rem;border-bottom-left-radius:0.25rem}.list-group-item:hover,.list-group-item:focus{z-index:1;text-decoration:none}.list-group-item.disabled,.list-group-item:disabled{color:#073642;pointer-events:none;background-color:transparent}.list-group-item.active{z-index:2;color:rgba(255,255,255,0.75);background-color:#073642;border-color:#073642}.list-group-flush .list-group-item{border-right:0;border-left:0;border-radius:0}.list-group-flush .list-group-item:last-child{margin-bottom:-1px}.list-group-flush:first-child .list-group-item:first-child{border-top:0}.list-group-flush:last-child .list-group-item:last-child{margin-bottom:0;border-bottom:0}.list-group-item-primary{color:#5e4700;background-color:#eadeb8}.list-group-item-primary.list-group-item-action:hover,.list-group-item-primary.list-group-item-action:focus{color:#5e4700;background-color:#e4d5a4}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#5e4700;border-color:#5e4700}.list-group-item-secondary{color:#444d4e;background-color:#dce1e2}.list-group-item-secondary.list-group-item-action:hover,.list-group-item-secondary.list-group-item-action:focus{color:#444d4e;background-color:#ced5d6}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#444d4e;border-color:#444d4e}.list-group-item-success{color:#16544f;background-color:#c3e5e2}.list-group-item-success.list-group-item-action:hover,.list-group-item-success.list-group-item-action:focus{color:#16544f;background-color:#b1ddd9}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#16544f;border-color:#16544f}.list-group-item-info{color:#14486d;background-color:#c2dff2}.list-group-item-info.list-group-item-action:hover,.list-group-item-info.list-group-item-action:focus{color:#14486d;background-color:#add4ee}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#14486d;border-color:#14486d}.list-group-item-warning{color:#6a270b;background-color:#f0cdbe}.list-group-item-warning.list-group-item-action:hover,.list-group-item-warning.list-group-item-action:focus{color:#6a270b;background-color:#ebbda9}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#6a270b;border-color:#6a270b}.list-group-item-danger{color:#6e1c44;background-color:#f3c7dc}.list-group-item-danger.list-group-item-action:hover,.list-group-item-danger.list-group-item-action:focus{color:#6e1c44;background-color:#efb2cf}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#6e1c44;border-color:#6e1c44}.list-group-item-light{color:#848076;background-color:#fefcf7}.list-group-item-light.list-group-item-action:hover,.list-group-item-light.list-group-item-action:focus{color:#848076;background-color:#fbf4e0}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#848076;border-color:#848076}.list-group-item-dark{color:#041c22;background-color:#bac7ca}.list-group-item-dark.list-group-item-action:hover,.list-group-item-dark.list-group-item-action:focus{color:#041c22;background-color:#acbbbf}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#041c22;border-color:#041c22}.close{float:right;font-size:1.5rem;font-weight:700;line-height:1;color:#839496;text-shadow:none;opacity:.5}.close:hover{color:#839496;text-decoration:none}.close:not(:disabled):not(.disabled){cursor:pointer}.close:not(:disabled):not(.disabled):hover,.close:not(:disabled):not(.disabled):focus{opacity:.75}button.close{padding:0;background-color:transparent;border:0;-webkit-appearance:none;-moz-appearance:none;appearance:none}a.close.disabled{pointer-events:none}.toast{max-width:350px;overflow:hidden;font-size:0.875rem;background-color:rgba(255,255,255,0.85);background-clip:padding-box;border:1px solid rgba(0,0,0,0.1);border-radius:0.25rem;-webkit-box-shadow:0 0.25rem 0.75rem rgba(0,0,0,0.1);box-shadow:0 0.25rem 0.75rem rgba(0,0,0,0.1);-webkit-backdrop-filter:blur(10px);backdrop-filter:blur(10px);opacity:0}.toast:not(:last-child){margin-bottom:0.75rem}.toast.showing{opacity:1}.toast.show{display:block;opacity:1}.toast.hide{display:none}.toast-header{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;padding:0.25rem 0.75rem;color:#839496;background-color:rgba(255,255,255,0.85);background-clip:padding-box;border-bottom:1px solid rgba(0,0,0,0.05)}.toast-body{padding:0.75rem}.modal-open{overflow:hidden}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal{position:fixed;top:0;left:0;z-index:1050;display:none;width:100%;height:100%;overflow:hidden;outline:0}.modal-dialog{position:relative;width:auto;margin:0.5rem;pointer-events:none}.modal.fade .modal-dialog{-webkit-transition:-webkit-transform 0.3s ease-out;transition:-webkit-transform 0.3s ease-out;transition:transform 0.3s ease-out;transition:transform 0.3s ease-out, -webkit-transform 0.3s ease-out;-webkit-transform:translate(0, -50px);transform:translate(0, -50px)}@media screen and (prefers-reduced-motion: reduce){.modal.fade .modal-dialog{-webkit-transition:none;transition:none}}.modal.show .modal-dialog{-webkit-transform:none;transform:none}.modal-dialog-centered{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;min-height:calc(100% - (0.5rem * 2))}.modal-dialog-centered::before{display:block;height:calc(100vh - (0.5rem * 2));content:""}.modal-content{position:relative;display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;width:100%;pointer-events:auto;background-color:#073642;background-clip:padding-box;border:1px solid #002B36;border-radius:0.3rem;outline:0}.modal-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:0.5}.modal-header{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:start;-ms-flex-align:start;align-items:flex-start;-webkit-box-pack:justify;-ms-flex-pack:justify;justify-content:space-between;padding:1rem 1rem;border-bottom:1px solid #002B36;border-top-left-radius:0.3rem;border-top-right-radius:0.3rem}.modal-header .close{padding:1rem 1rem;margin:-1rem -1rem -1rem auto}.modal-title{margin-bottom:0;line-height:1.5}.modal-body{position:relative;-webkit-box-flex:1;-ms-flex:1 1 auto;flex:1 1 auto;padding:1rem}.modal-footer{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:end;-ms-flex-pack:end;justify-content:flex-end;padding:1rem;border-top:1px solid #002B36;border-bottom-right-radius:0.3rem;border-bottom-left-radius:0.3rem}.modal-footer>:not(:first-child){margin-left:.25rem}.modal-footer>:not(:last-child){margin-right:.25rem}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width: 576px){.modal-dialog{max-width:500px;margin:1.75rem auto}.modal-dialog-centered{min-height:calc(100% - (1.75rem * 2))}.modal-dialog-centered::before{height:calc(100vh - (1.75rem * 2))}.modal-sm{max-width:300px}}@media (min-width: 992px){.modal-lg,.modal-xl{max-width:800px}}@media (min-width: 1200px){.modal-xl{max-width:1140px}}.tooltip{position:absolute;z-index:1070;display:block;margin:0;font-family:"Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:0.875rem;word-wrap:break-word;opacity:0}.tooltip.show{opacity:0.9}.tooltip .arrow{position:absolute;display:block;width:0.8rem;height:0.4rem}.tooltip .arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-top,.bs-tooltip-auto[x-placement^="top"]{padding:0.4rem 0}.bs-tooltip-top .arrow,.bs-tooltip-auto[x-placement^="top"] .arrow{bottom:0}.bs-tooltip-top .arrow::before,.bs-tooltip-auto[x-placement^="top"] .arrow::before{top:0;border-width:0.4rem 0.4rem 0;border-top-color:#000}.bs-tooltip-right,.bs-tooltip-auto[x-placement^="right"]{padding:0 0.4rem}.bs-tooltip-right .arrow,.bs-tooltip-auto[x-placement^="right"] .arrow{left:0;width:0.4rem;height:0.8rem}.bs-tooltip-right .arrow::before,.bs-tooltip-auto[x-placement^="right"] .arrow::before{right:0;border-width:0.4rem 0.4rem 0.4rem 0;border-right-color:#000}.bs-tooltip-bottom,.bs-tooltip-auto[x-placement^="bottom"]{padding:0.4rem 0}.bs-tooltip-bottom .arrow,.bs-tooltip-auto[x-placement^="bottom"] .arrow{top:0}.bs-tooltip-bottom .arrow::before,.bs-tooltip-auto[x-placement^="bottom"] .arrow::before{bottom:0;border-width:0 0.4rem 0.4rem;border-bottom-color:#000}.bs-tooltip-left,.bs-tooltip-auto[x-placement^="left"]{padding:0 0.4rem}.bs-tooltip-left .arrow,.bs-tooltip-auto[x-placement^="left"] .arrow{right:0;width:0.4rem;height:0.8rem}.bs-tooltip-left .arrow::before,.bs-tooltip-auto[x-placement^="left"] .arrow::before{left:0;border-width:0.4rem 0 0.4rem 0.4rem;border-left-color:#000}.tooltip-inner{max-width:200px;padding:0.25rem 0.5rem;color:#fff;text-align:center;background-color:#000;border-radius:0.25rem}.popover{position:absolute;top:0;left:0;z-index:1060;display:block;max-width:276px;font-family:"Source Sans Pro", -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, "Helvetica Neue", Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:0.875rem;word-wrap:break-word;background-color:#073642;background-clip:padding-box;border:1px solid #002B36;border-radius:0.3rem}.popover .arrow{position:absolute;display:block;width:1rem;height:0.5rem;margin:0 0.3rem}.popover .arrow::before,.popover .arrow::after{position:absolute;display:block;content:"";border-color:transparent;border-style:solid}.bs-popover-top,.bs-popover-auto[x-placement^="top"]{margin-bottom:0.5rem}.bs-popover-top .arrow,.bs-popover-auto[x-placement^="top"] .arrow{bottom:calc((0.5rem + 1px) * -1)}.bs-popover-top .arrow::before,.bs-popover-auto[x-placement^="top"] .arrow::before,.bs-popover-top .arrow::after,.bs-popover-auto[x-placement^="top"] .arrow::after{border-width:0.5rem 0.5rem 0}.bs-popover-top .arrow::before,.bs-popover-auto[x-placement^="top"] .arrow::before{bottom:0;border-top-color:#002b36}.bs-popover-top .arrow::after,.bs-popover-auto[x-placement^="top"] .arrow::after{bottom:1px;border-top-color:#073642}.bs-popover-right,.bs-popover-auto[x-placement^="right"]{margin-left:0.5rem}.bs-popover-right .arrow,.bs-popover-auto[x-placement^="right"] .arrow{left:calc((0.5rem + 1px) * -1);width:0.5rem;height:1rem;margin:0.3rem 0}.bs-popover-right .arrow::before,.bs-popover-auto[x-placement^="right"] .arrow::before,.bs-popover-right .arrow::after,.bs-popover-auto[x-placement^="right"] .arrow::after{border-width:0.5rem 0.5rem 0.5rem 0}.bs-popover-right .arrow::before,.bs-popover-auto[x-placement^="right"] .arrow::before{left:0;border-right-color:#002b36}.bs-popover-right .arrow::after,.bs-popover-auto[x-placement^="right"] .arrow::after{left:1px;border-right-color:#073642}.bs-popover-bottom,.bs-popover-auto[x-placement^="bottom"]{margin-top:0.5rem}.bs-popover-bottom .arrow,.bs-popover-auto[x-placement^="bottom"] .arrow{top:calc((0.5rem + 1px) * -1)}.bs-popover-bottom .arrow::before,.bs-popover-auto[x-placement^="bottom"] .arrow::before,.bs-popover-bottom .arrow::after,.bs-popover-auto[x-placement^="bottom"] .arrow::after{border-width:0 0.5rem 0.5rem 0.5rem}.bs-popover-bottom .arrow::before,.bs-popover-auto[x-placement^="bottom"] .arrow::before{top:0;border-bottom-color:#002b36}.bs-popover-bottom .arrow::after,.bs-popover-auto[x-placement^="bottom"] .arrow::after{top:1px;border-bottom-color:#073642}.bs-popover-bottom .popover-header::before,.bs-popover-auto[x-placement^="bottom"] .popover-header::before{position:absolute;top:0;left:50%;display:block;width:1rem;margin-left:-0.5rem;content:"";border-bottom:1px solid #073642}.bs-popover-left,.bs-popover-auto[x-placement^="left"]{margin-right:0.5rem}.bs-popover-left .arrow,.bs-popover-auto[x-placement^="left"] .arrow{right:calc((0.5rem + 1px) * -1);width:0.5rem;height:1rem;margin:0.3rem 0}.bs-popover-left .arrow::before,.bs-popover-auto[x-placement^="left"] .arrow::before,.bs-popover-left .arrow::after,.bs-popover-auto[x-placement^="left"] .arrow::after{border-width:0.5rem 0 0.5rem 0.5rem}.bs-popover-left .arrow::before,.bs-popover-auto[x-placement^="left"] .arrow::before{right:0;border-left-color:#002b36}.bs-popover-left .arrow::after,.bs-popover-auto[x-placement^="left"] .arrow::after{right:1px;border-left-color:#073642}.popover-header{padding:0.5rem 0.75rem;margin-bottom:0;font-size:1rem;color:inherit;background-color:#073642;border-bottom:1px solid #05232b;border-top-left-radius:calc(0.3rem - 1px);border-top-right-radius:calc(0.3rem - 1px)}.popover-header:empty{display:none}.popover-body{padding:0.5rem 0.75rem;color:#839496}.carousel{position:relative}.carousel.pointer-event{-ms-touch-action:pan-y;touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden;-webkit-transition:-webkit-transform 0.6s ease-in-out;transition:-webkit-transform 0.6s ease-in-out;transition:transform 0.6s ease-in-out;transition:transform 0.6s ease-in-out, -webkit-transform 0.6s ease-in-out}@media screen and (prefers-reduced-motion: reduce){.carousel-item{-webkit-transition:none;transition:none}}.carousel-item.active,.carousel-item-next,.carousel-item-prev{display:block}.carousel-item-next:not(.carousel-item-left),.active.carousel-item-right{-webkit-transform:translateX(100%);transform:translateX(100%)}.carousel-item-prev:not(.carousel-item-right),.active.carousel-item-left{-webkit-transform:translateX(-100%);transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;-webkit-transition-property:opacity;transition-property:opacity;-webkit-transform:none;transform:none}.carousel-fade .carousel-item.active,.carousel-fade .carousel-item-next.carousel-item-left,.carousel-fade .carousel-item-prev.carousel-item-right{z-index:1;opacity:1}.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{z-index:0;opacity:0;-webkit-transition:0s 0.6s opacity;transition:0s 0.6s opacity}@media screen and (prefers-reduced-motion: reduce){.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{-webkit-transition:none;transition:none}}.carousel-control-prev,.carousel-control-next{position:absolute;top:0;bottom:0;z-index:1;display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center;width:15%;color:#fff;text-align:center;opacity:0.5;-webkit-transition:opacity 0.15s ease;transition:opacity 0.15s ease}@media screen and (prefers-reduced-motion: reduce){.carousel-control-prev,.carousel-control-next{-webkit-transition:none;transition:none}}.carousel-control-prev:hover,.carousel-control-prev:focus,.carousel-control-next:hover,.carousel-control-next:focus{color:#fff;text-decoration:none;outline:0;opacity:0.9}.carousel-control-prev{left:0;background:-webkit-gradient(linear, left top, right top, from(rgba(0,0,0,0.25)), to(rgba(0,0,0,0.001)));background:linear-gradient(90deg, rgba(0,0,0,0.25), rgba(0,0,0,0.001))}.carousel-control-next{right:0;background:-webkit-gradient(linear, right top, left top, from(rgba(0,0,0,0.25)), to(rgba(0,0,0,0.001)));background:linear-gradient(270deg, rgba(0,0,0,0.25), rgba(0,0,0,0.001))}.carousel-control-prev-icon,.carousel-control-next-icon{display:inline-block;width:20px;height:20px;background:transparent no-repeat center center;background-size:100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 8 8'%3e%3cpath d='M5.25 0l-4 4 4 4 1.5-1.5-2.5-2.5 2.5-2.5-1.5-1.5z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' viewBox='0 0 8 8'%3e%3cpath d='M2.75 0l-1.5 1.5 2.5 2.5-2.5 2.5 1.5 1.5 4-4-4-4z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:15;display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center;padding-left:0;margin-right:15%;margin-left:15%;list-style:none}.carousel-indicators li{-webkit-box-sizing:content-box;box-sizing:content-box;-webkit-box-flex:0;-ms-flex:0 1 auto;flex:0 1 auto;width:30px;height:3px;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5;-webkit-transition:opacity 0.6s ease;transition:opacity 0.6s ease}@media screen and (prefers-reduced-motion: reduce){.carousel-indicators li{-webkit-transition:none;transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center}@-webkit-keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}.spinner-border{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;border:0.25em solid currentColor;border-right-color:transparent;border-radius:50%;-webkit-animation:spinner-border .75s linear infinite;animation:spinner-border .75s linear infinite}.spinner-border-sm{width:1rem;height:1rem;border-width:0.2em}@-webkit-keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1}}@keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1}}.spinner-grow{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;background-color:currentColor;border-radius:50%;opacity:0;-webkit-animation:spinner-grow .75s linear infinite;animation:spinner-grow .75s linear infinite}.spinner-grow-sm{width:1rem;height:1rem}.align-baseline{vertical-align:baseline !important}.align-top{vertical-align:top !important}.align-middle{vertical-align:middle !important}.align-bottom{vertical-align:bottom !important}.align-text-bottom{vertical-align:text-bottom !important}.align-text-top{vertical-align:text-top !important}.bg-primary{background-color:#B58900 !important}a.bg-primary:hover,a.bg-primary:focus,button.bg-primary:hover,button.bg-primary:focus{background-color:#826200 !important}.bg-secondary{background-color:#839496 !important}a.bg-secondary:hover,a.bg-secondary:focus,button.bg-secondary:hover,button.bg-secondary:focus{background-color:#697b7d !important}.bg-success{background-color:#2AA198 !important}a.bg-success:hover,a.bg-success:focus,button.bg-success:hover,button.bg-success:focus{background-color:#1f7972 !important}.bg-info{background-color:#268BD2 !important}a.bg-info:hover,a.bg-info:focus,button.bg-info:hover,button.bg-info:focus{background-color:#1e6ea7 !important}.bg-warning{background-color:#CB4B16 !important}a.bg-warning:hover,a.bg-warning:focus,button.bg-warning:hover,button.bg-warning:focus{background-color:#9d3a11 !important}.bg-danger{background-color:#D33682 !important}a.bg-danger:hover,a.bg-danger:focus,button.bg-danger:hover,button.bg-danger:focus{background-color:#b02669 !important}.bg-light{background-color:#FDF6E3 !important}a.bg-light:hover,a.bg-light:focus,button.bg-light:hover,button.bg-light:focus{background-color:#fae7b3 !important}.bg-dark{background-color:#073642 !important}a.bg-dark:hover,a.bg-dark:focus,button.bg-dark:hover,button.bg-dark:focus{background-color:#021014 !important}.bg-gradient-primary{background:#B58900 -webkit-gradient(linear, left top, left bottom, from(#9a7b08), to(#B58900)) repeat-x !important;background:#B58900 linear-gradient(180deg, #9a7b08, #B58900) repeat-x !important}.bg-gradient-secondary{background:#839496 -webkit-gradient(linear, left top, left bottom, from(#6f8488), to(#839496)) repeat-x !important;background:#839496 linear-gradient(180deg, #6f8488, #839496) repeat-x !important}.bg-gradient-success{background:#2AA198 -webkit-gradient(linear, left top, left bottom, from(#248f89), to(#2AA198)) repeat-x !important;background:#2AA198 linear-gradient(180deg, #248f89, #2AA198) repeat-x !important}.bg-gradient-info{background:#268BD2 -webkit-gradient(linear, left top, left bottom, from(#207dbb), to(#268BD2)) repeat-x !important;background:#268BD2 linear-gradient(180deg, #207dbb, #268BD2) repeat-x !important}.bg-gradient-warning{background:#CB4B16 -webkit-gradient(linear, left top, left bottom, from(#ad461b), to(#CB4B16)) repeat-x !important;background:#CB4B16 linear-gradient(180deg, #ad461b, #CB4B16) repeat-x !important}.bg-gradient-danger{background:#D33682 -webkit-gradient(linear, left top, left bottom, from(#b33477), to(#D33682)) repeat-x !important;background:#D33682 linear-gradient(180deg, #b33477, #D33682) repeat-x !important}.bg-gradient-light{background:#FDF6E3 -webkit-gradient(linear, left top, left bottom, from(#d7d8c9), to(#FDF6E3)) repeat-x !important;background:#FDF6E3 linear-gradient(180deg, #d7d8c9, #FDF6E3) repeat-x !important}.bg-gradient-dark{background:#073642 -webkit-gradient(linear, left top, left bottom, from(#063440), to(#073642)) repeat-x !important;background:#073642 linear-gradient(180deg, #063440, #073642) repeat-x !important}.bg-white{background-color:#fff !important}.bg-transparent{background-color:transparent !important}.border{border:1px solid #dee2e6 !important}.border-top{border-top:1px solid #dee2e6 !important}.border-right{border-right:1px solid #dee2e6 !important}.border-bottom{border-bottom:1px solid #dee2e6 !important}.border-left{border-left:1px solid #dee2e6 !important}.border-0{border:0 !important}.border-top-0{border-top:0 !important}.border-right-0{border-right:0 !important}.border-bottom-0{border-bottom:0 !important}.border-left-0{border-left:0 !important}.border-primary{border-color:#B58900 !important}.border-secondary{border-color:#839496 !important}.border-success{border-color:#2AA198 !important}.border-info{border-color:#268BD2 !important}.border-warning{border-color:#CB4B16 !important}.border-danger{border-color:#D33682 !important}.border-light{border-color:#FDF6E3 !important}.border-dark{border-color:#073642 !important}.border-white{border-color:#fff !important}.rounded{border-radius:0.25rem !important}.rounded-top{border-top-left-radius:0.25rem !important;border-top-right-radius:0.25rem !important}.rounded-right{border-top-right-radius:0.25rem !important;border-bottom-right-radius:0.25rem !important}.rounded-bottom{border-bottom-right-radius:0.25rem !important;border-bottom-left-radius:0.25rem !important}.rounded-left{border-top-left-radius:0.25rem !important;border-bottom-left-radius:0.25rem !important}.rounded-circle{border-radius:50% !important}.rounded-pill{border-radius:50rem !important}.rounded-0{border-radius:0 !important}.clearfix::after{display:block;clear:both;content:""}.d-none{display:none !important}.d-inline{display:inline !important}.d-inline-block{display:inline-block !important}.d-block{display:block !important}.d-table{display:table !important}.d-table-row{display:table-row !important}.d-table-cell{display:table-cell !important}.d-flex{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important}.d-inline-flex{display:-webkit-inline-box !important;display:-ms-inline-flexbox !important;display:inline-flex !important}@media (min-width: 576px){.d-sm-none{display:none !important}.d-sm-inline{display:inline !important}.d-sm-inline-block{display:inline-block !important}.d-sm-block{display:block !important}.d-sm-table{display:table !important}.d-sm-table-row{display:table-row !important}.d-sm-table-cell{display:table-cell !important}.d-sm-flex{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important}.d-sm-inline-flex{display:-webkit-inline-box !important;display:-ms-inline-flexbox !important;display:inline-flex !important}}@media (min-width: 768px){.d-md-none{display:none !important}.d-md-inline{display:inline !important}.d-md-inline-block{display:inline-block !important}.d-md-block{display:block !important}.d-md-table{display:table !important}.d-md-table-row{display:table-row !important}.d-md-table-cell{display:table-cell !important}.d-md-flex{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important}.d-md-inline-flex{display:-webkit-inline-box !important;display:-ms-inline-flexbox !important;display:inline-flex !important}}@media (min-width: 992px){.d-lg-none{display:none !important}.d-lg-inline{display:inline !important}.d-lg-inline-block{display:inline-block !important}.d-lg-block{display:block !important}.d-lg-table{display:table !important}.d-lg-table-row{display:table-row !important}.d-lg-table-cell{display:table-cell !important}.d-lg-flex{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important}.d-lg-inline-flex{display:-webkit-inline-box !important;display:-ms-inline-flexbox !important;display:inline-flex !important}}@media (min-width: 1200px){.d-xl-none{display:none !important}.d-xl-inline{display:inline !important}.d-xl-inline-block{display:inline-block !important}.d-xl-block{display:block !important}.d-xl-table{display:table !important}.d-xl-table-row{display:table-row !important}.d-xl-table-cell{display:table-cell !important}.d-xl-flex{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important}.d-xl-inline-flex{display:-webkit-inline-box !important;display:-ms-inline-flexbox !important;display:inline-flex !important}}@media print{.d-print-none{display:none !important}.d-print-inline{display:inline !important}.d-print-inline-block{display:inline-block !important}.d-print-block{display:block !important}.d-print-table{display:table !important}.d-print-table-row{display:table-row !important}.d-print-table-cell{display:table-cell !important}.d-print-flex{display:-webkit-box !important;display:-ms-flexbox !important;display:flex !important}.d-print-inline-flex{display:-webkit-inline-box !important;display:-ms-inline-flexbox !important;display:inline-flex !important}}.embed-responsive{position:relative;display:block;width:100%;padding:0;overflow:hidden}.embed-responsive::before{display:block;content:""}.embed-responsive .embed-responsive-item,.embed-responsive iframe,.embed-responsive embed,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-21by9::before{padding-top:42.8571428571%}.embed-responsive-16by9::before{padding-top:56.25%}.embed-responsive-3by4::before{padding-top:133.333333333%}.embed-responsive-1by1::before{padding-top:100%}.flex-row{-webkit-box-orient:horizontal !important;-webkit-box-direction:normal !important;-ms-flex-direction:row !important;flex-direction:row !important}.flex-column{-webkit-box-orient:vertical !important;-webkit-box-direction:normal !important;-ms-flex-direction:column !important;flex-direction:column !important}.flex-row-reverse{-webkit-box-orient:horizontal !important;-webkit-box-direction:reverse !important;-ms-flex-direction:row-reverse !important;flex-direction:row-reverse !important}.flex-column-reverse{-webkit-box-orient:vertical !important;-webkit-box-direction:reverse !important;-ms-flex-direction:column-reverse !important;flex-direction:column-reverse !important}.flex-wrap{-ms-flex-wrap:wrap !important;flex-wrap:wrap !important}.flex-nowrap{-ms-flex-wrap:nowrap !important;flex-wrap:nowrap !important}.flex-wrap-reverse{-ms-flex-wrap:wrap-reverse !important;flex-wrap:wrap-reverse !important}.flex-fill{-webkit-box-flex:1 !important;-ms-flex:1 1 auto !important;flex:1 1 auto !important}.flex-grow-0{-webkit-box-flex:0 !important;-ms-flex-positive:0 !important;flex-grow:0 !important}.flex-grow-1{-webkit-box-flex:1 !important;-ms-flex-positive:1 !important;flex-grow:1 !important}.flex-shrink-0{-ms-flex-negative:0 !important;flex-shrink:0 !important}.flex-shrink-1{-ms-flex-negative:1 !important;flex-shrink:1 !important}.justify-content-start{-webkit-box-pack:start !important;-ms-flex-pack:start !important;justify-content:flex-start !important}.justify-content-end{-webkit-box-pack:end !important;-ms-flex-pack:end !important;justify-content:flex-end !important}.justify-content-center{-webkit-box-pack:center !important;-ms-flex-pack:center !important;justify-content:center !important}.justify-content-between{-webkit-box-pack:justify !important;-ms-flex-pack:justify !important;justify-content:space-between !important}.justify-content-around{-ms-flex-pack:distribute !important;justify-content:space-around !important}.align-items-start{-webkit-box-align:start !important;-ms-flex-align:start !important;align-items:flex-start !important}.align-items-end{-webkit-box-align:end !important;-ms-flex-align:end !important;align-items:flex-end !important}.align-items-center{-webkit-box-align:center !important;-ms-flex-align:center !important;align-items:center !important}.align-items-baseline{-webkit-box-align:baseline !important;-ms-flex-align:baseline !important;align-items:baseline !important}.align-items-stretch{-webkit-box-align:stretch !important;-ms-flex-align:stretch !important;align-items:stretch !important}.align-content-start{-ms-flex-line-pack:start !important;align-content:flex-start !important}.align-content-end{-ms-flex-line-pack:end !important;align-content:flex-end !important}.align-content-center{-ms-flex-line-pack:center !important;align-content:center !important}.align-content-between{-ms-flex-line-pack:justify !important;align-content:space-between !important}.align-content-around{-ms-flex-line-pack:distribute !important;align-content:space-around !important}.align-content-stretch{-ms-flex-line-pack:stretch !important;align-content:stretch !important}.align-self-auto{-ms-flex-item-align:auto !important;align-self:auto !important}.align-self-start{-ms-flex-item-align:start !important;align-self:flex-start !important}.align-self-end{-ms-flex-item-align:end !important;align-self:flex-end !important}.align-self-center{-ms-flex-item-align:center !important;align-self:center !important}.align-self-baseline{-ms-flex-item-align:baseline !important;align-self:baseline !important}.align-self-stretch{-ms-flex-item-align:stretch !important;align-self:stretch !important}@media (min-width: 576px){.flex-sm-row{-webkit-box-orient:horizontal !important;-webkit-box-direction:normal !important;-ms-flex-direction:row !important;flex-direction:row !important}.flex-sm-column{-webkit-box-orient:vertical !important;-webkit-box-direction:normal !important;-ms-flex-direction:column !important;flex-direction:column !important}.flex-sm-row-reverse{-webkit-box-orient:horizontal !important;-webkit-box-direction:reverse !important;-ms-flex-direction:row-reverse !important;flex-direction:row-reverse !important}.flex-sm-column-reverse{-webkit-box-orient:vertical !important;-webkit-box-direction:reverse !important;-ms-flex-direction:column-reverse !important;flex-direction:column-reverse !important}.flex-sm-wrap{-ms-flex-wrap:wrap !important;flex-wrap:wrap !important}.flex-sm-nowrap{-ms-flex-wrap:nowrap !important;flex-wrap:nowrap !important}.flex-sm-wrap-reverse{-ms-flex-wrap:wrap-reverse !important;flex-wrap:wrap-reverse !important}.flex-sm-fill{-webkit-box-flex:1 !important;-ms-flex:1 1 auto !important;flex:1 1 auto !important}.flex-sm-grow-0{-webkit-box-flex:0 !important;-ms-flex-positive:0 !important;flex-grow:0 !important}.flex-sm-grow-1{-webkit-box-flex:1 !important;-ms-flex-positive:1 !important;flex-grow:1 !important}.flex-sm-shrink-0{-ms-flex-negative:0 !important;flex-shrink:0 !important}.flex-sm-shrink-1{-ms-flex-negative:1 !important;flex-shrink:1 !important}.justify-content-sm-start{-webkit-box-pack:start !important;-ms-flex-pack:start !important;justify-content:flex-start !important}.justify-content-sm-end{-webkit-box-pack:end !important;-ms-flex-pack:end !important;justify-content:flex-end !important}.justify-content-sm-center{-webkit-box-pack:center !important;-ms-flex-pack:center !important;justify-content:center !important}.justify-content-sm-between{-webkit-box-pack:justify !important;-ms-flex-pack:justify !important;justify-content:space-between !important}.justify-content-sm-around{-ms-flex-pack:distribute !important;justify-content:space-around !important}.align-items-sm-start{-webkit-box-align:start !important;-ms-flex-align:start !important;align-items:flex-start !important}.align-items-sm-end{-webkit-box-align:end !important;-ms-flex-align:end !important;align-items:flex-end !important}.align-items-sm-center{-webkit-box-align:center !important;-ms-flex-align:center !important;align-items:center !important}.align-items-sm-baseline{-webkit-box-align:baseline !important;-ms-flex-align:baseline !important;align-items:baseline !important}.align-items-sm-stretch{-webkit-box-align:stretch !important;-ms-flex-align:stretch !important;align-items:stretch !important}.align-content-sm-start{-ms-flex-line-pack:start !important;align-content:flex-start !important}.align-content-sm-end{-ms-flex-line-pack:end !important;align-content:flex-end !important}.align-content-sm-center{-ms-flex-line-pack:center !important;align-content:center !important}.align-content-sm-between{-ms-flex-line-pack:justify !important;align-content:space-between !important}.align-content-sm-around{-ms-flex-line-pack:distribute !important;align-content:space-around !important}.align-content-sm-stretch{-ms-flex-line-pack:stretch !important;align-content:stretch !important}.align-self-sm-auto{-ms-flex-item-align:auto !important;align-self:auto !important}.align-self-sm-start{-ms-flex-item-align:start !important;align-self:flex-start !important}.align-self-sm-end{-ms-flex-item-align:end !important;align-self:flex-end !important}.align-self-sm-center{-ms-flex-item-align:center !important;align-self:center !important}.align-self-sm-baseline{-ms-flex-item-align:baseline !important;align-self:baseline !important}.align-self-sm-stretch{-ms-flex-item-align:stretch !important;align-self:stretch !important}}@media (min-width: 768px){.flex-md-row{-webkit-box-orient:horizontal !important;-webkit-box-direction:normal !important;-ms-flex-direction:row !important;flex-direction:row !important}.flex-md-column{-webkit-box-orient:vertical !important;-webkit-box-direction:normal !important;-ms-flex-direction:column !important;flex-direction:column !important}.flex-md-row-reverse{-webkit-box-orient:horizontal !important;-webkit-box-direction:reverse !important;-ms-flex-direction:row-reverse !important;flex-direction:row-reverse !important}.flex-md-column-reverse{-webkit-box-orient:vertical !important;-webkit-box-direction:reverse !important;-ms-flex-direction:column-reverse !important;flex-direction:column-reverse !important}.flex-md-wrap{-ms-flex-wrap:wrap !important;flex-wrap:wrap !important}.flex-md-nowrap{-ms-flex-wrap:nowrap !important;flex-wrap:nowrap !important}.flex-md-wrap-reverse{-ms-flex-wrap:wrap-reverse !important;flex-wrap:wrap-reverse !important}.flex-md-fill{-webkit-box-flex:1 !important;-ms-flex:1 1 auto !important;flex:1 1 auto !important}.flex-md-grow-0{-webkit-box-flex:0 !important;-ms-flex-positive:0 !important;flex-grow:0 !important}.flex-md-grow-1{-webkit-box-flex:1 !important;-ms-flex-positive:1 !important;flex-grow:1 !important}.flex-md-shrink-0{-ms-flex-negative:0 !important;flex-shrink:0 !important}.flex-md-shrink-1{-ms-flex-negative:1 !important;flex-shrink:1 !important}.justify-content-md-start{-webkit-box-pack:start !important;-ms-flex-pack:start !important;justify-content:flex-start !important}.justify-content-md-end{-webkit-box-pack:end !important;-ms-flex-pack:end !important;justify-content:flex-end !important}.justify-content-md-center{-webkit-box-pack:center !important;-ms-flex-pack:center !important;justify-content:center !important}.justify-content-md-between{-webkit-box-pack:justify !important;-ms-flex-pack:justify !important;justify-content:space-between !important}.justify-content-md-around{-ms-flex-pack:distribute !important;justify-content:space-around !important}.align-items-md-start{-webkit-box-align:start !important;-ms-flex-align:start !important;align-items:flex-start !important}.align-items-md-end{-webkit-box-align:end !important;-ms-flex-align:end !important;align-items:flex-end !important}.align-items-md-center{-webkit-box-align:center !important;-ms-flex-align:center !important;align-items:center !important}.align-items-md-baseline{-webkit-box-align:baseline !important;-ms-flex-align:baseline !important;align-items:baseline !important}.align-items-md-stretch{-webkit-box-align:stretch !important;-ms-flex-align:stretch !important;align-items:stretch !important}.align-content-md-start{-ms-flex-line-pack:start !important;align-content:flex-start !important}.align-content-md-end{-ms-flex-line-pack:end !important;align-content:flex-end !important}.align-content-md-center{-ms-flex-line-pack:center !important;align-content:center !important}.align-content-md-between{-ms-flex-line-pack:justify !important;align-content:space-between !important}.align-content-md-around{-ms-flex-line-pack:distribute !important;align-content:space-around !important}.align-content-md-stretch{-ms-flex-line-pack:stretch !important;align-content:stretch !important}.align-self-md-auto{-ms-flex-item-align:auto !important;align-self:auto !important}.align-self-md-start{-ms-flex-item-align:start !important;align-self:flex-start !important}.align-self-md-end{-ms-flex-item-align:end !important;align-self:flex-end !important}.align-self-md-center{-ms-flex-item-align:center !important;align-self:center !important}.align-self-md-baseline{-ms-flex-item-align:baseline !important;align-self:baseline !important}.align-self-md-stretch{-ms-flex-item-align:stretch !important;align-self:stretch !important}}@media (min-width: 992px){.flex-lg-row{-webkit-box-orient:horizontal !important;-webkit-box-direction:normal !important;-ms-flex-direction:row !important;flex-direction:row !important}.flex-lg-column{-webkit-box-orient:vertical !important;-webkit-box-direction:normal !important;-ms-flex-direction:column !important;flex-direction:column !important}.flex-lg-row-reverse{-webkit-box-orient:horizontal !important;-webkit-box-direction:reverse !important;-ms-flex-direction:row-reverse !important;flex-direction:row-reverse !important}.flex-lg-column-reverse{-webkit-box-orient:vertical !important;-webkit-box-direction:reverse !important;-ms-flex-direction:column-reverse !important;flex-direction:column-reverse !important}.flex-lg-wrap{-ms-flex-wrap:wrap !important;flex-wrap:wrap !important}.flex-lg-nowrap{-ms-flex-wrap:nowrap !important;flex-wrap:nowrap !important}.flex-lg-wrap-reverse{-ms-flex-wrap:wrap-reverse !important;flex-wrap:wrap-reverse !important}.flex-lg-fill{-webkit-box-flex:1 !important;-ms-flex:1 1 auto !important;flex:1 1 auto !important}.flex-lg-grow-0{-webkit-box-flex:0 !important;-ms-flex-positive:0 !important;flex-grow:0 !important}.flex-lg-grow-1{-webkit-box-flex:1 !important;-ms-flex-positive:1 !important;flex-grow:1 !important}.flex-lg-shrink-0{-ms-flex-negative:0 !important;flex-shrink:0 !important}.flex-lg-shrink-1{-ms-flex-negative:1 !important;flex-shrink:1 !important}.justify-content-lg-start{-webkit-box-pack:start !important;-ms-flex-pack:start !important;justify-content:flex-start !important}.justify-content-lg-end{-webkit-box-pack:end !important;-ms-flex-pack:end !important;justify-content:flex-end !important}.justify-content-lg-center{-webkit-box-pack:center !important;-ms-flex-pack:center !important;justify-content:center !important}.justify-content-lg-between{-webkit-box-pack:justify !important;-ms-flex-pack:justify !important;justify-content:space-between !important}.justify-content-lg-around{-ms-flex-pack:distribute !important;justify-content:space-around !important}.align-items-lg-start{-webkit-box-align:start !important;-ms-flex-align:start !important;align-items:flex-start !important}.align-items-lg-end{-webkit-box-align:end !important;-ms-flex-align:end !important;align-items:flex-end !important}.align-items-lg-center{-webkit-box-align:center !important;-ms-flex-align:center !important;align-items:center !important}.align-items-lg-baseline{-webkit-box-align:baseline !important;-ms-flex-align:baseline !important;align-items:baseline !important}.align-items-lg-stretch{-webkit-box-align:stretch !important;-ms-flex-align:stretch !important;align-items:stretch !important}.align-content-lg-start{-ms-flex-line-pack:start !important;align-content:flex-start !important}.align-content-lg-end{-ms-flex-line-pack:end !important;align-content:flex-end !important}.align-content-lg-center{-ms-flex-line-pack:center !important;align-content:center !important}.align-content-lg-between{-ms-flex-line-pack:justify !important;align-content:space-between !important}.align-content-lg-around{-ms-flex-line-pack:distribute !important;align-content:space-around !important}.align-content-lg-stretch{-ms-flex-line-pack:stretch !important;align-content:stretch !important}.align-self-lg-auto{-ms-flex-item-align:auto !important;align-self:auto !important}.align-self-lg-start{-ms-flex-item-align:start !important;align-self:flex-start !important}.align-self-lg-end{-ms-flex-item-align:end !important;align-self:flex-end !important}.align-self-lg-center{-ms-flex-item-align:center !important;align-self:center !important}.align-self-lg-baseline{-ms-flex-item-align:baseline !important;align-self:baseline !important}.align-self-lg-stretch{-ms-flex-item-align:stretch !important;align-self:stretch !important}}@media (min-width: 1200px){.flex-xl-row{-webkit-box-orient:horizontal !important;-webkit-box-direction:normal !important;-ms-flex-direction:row !important;flex-direction:row !important}.flex-xl-column{-webkit-box-orient:vertical !important;-webkit-box-direction:normal !important;-ms-flex-direction:column !important;flex-direction:column !important}.flex-xl-row-reverse{-webkit-box-orient:horizontal !important;-webkit-box-direction:reverse !important;-ms-flex-direction:row-reverse !important;flex-direction:row-reverse !important}.flex-xl-column-reverse{-webkit-box-orient:vertical !important;-webkit-box-direction:reverse !important;-ms-flex-direction:column-reverse !important;flex-direction:column-reverse !important}.flex-xl-wrap{-ms-flex-wrap:wrap !important;flex-wrap:wrap !important}.flex-xl-nowrap{-ms-flex-wrap:nowrap !important;flex-wrap:nowrap !important}.flex-xl-wrap-reverse{-ms-flex-wrap:wrap-reverse !important;flex-wrap:wrap-reverse !important}.flex-xl-fill{-webkit-box-flex:1 !important;-ms-flex:1 1 auto !important;flex:1 1 auto !important}.flex-xl-grow-0{-webkit-box-flex:0 !important;-ms-flex-positive:0 !important;flex-grow:0 !important}.flex-xl-grow-1{-webkit-box-flex:1 !important;-ms-flex-positive:1 !important;flex-grow:1 !important}.flex-xl-shrink-0{-ms-flex-negative:0 !important;flex-shrink:0 !important}.flex-xl-shrink-1{-ms-flex-negative:1 !important;flex-shrink:1 !important}.justify-content-xl-start{-webkit-box-pack:start !important;-ms-flex-pack:start !important;justify-content:flex-start !important}.justify-content-xl-end{-webkit-box-pack:end !important;-ms-flex-pack:end !important;justify-content:flex-end !important}.justify-content-xl-center{-webkit-box-pack:center !important;-ms-flex-pack:center !important;justify-content:center !important}.justify-content-xl-between{-webkit-box-pack:justify !important;-ms-flex-pack:justify !important;justify-content:space-between !important}.justify-content-xl-around{-ms-flex-pack:distribute !important;justify-content:space-around !important}.align-items-xl-start{-webkit-box-align:start !important;-ms-flex-align:start !important;align-items:flex-start !important}.align-items-xl-end{-webkit-box-align:end !important;-ms-flex-align:end !important;align-items:flex-end !important}.align-items-xl-center{-webkit-box-align:center !important;-ms-flex-align:center !important;align-items:center !important}.align-items-xl-baseline{-webkit-box-align:baseline !important;-ms-flex-align:baseline !important;align-items:baseline !important}.align-items-xl-stretch{-webkit-box-align:stretch !important;-ms-flex-align:stretch !important;align-items:stretch !important}.align-content-xl-start{-ms-flex-line-pack:start !important;align-content:flex-start !important}.align-content-xl-end{-ms-flex-line-pack:end !important;align-content:flex-end !important}.align-content-xl-center{-ms-flex-line-pack:center !important;align-content:center !important}.align-content-xl-between{-ms-flex-line-pack:justify !important;align-content:space-between !important}.align-content-xl-around{-ms-flex-line-pack:distribute !important;align-content:space-around !important}.align-content-xl-stretch{-ms-flex-line-pack:stretch !important;align-content:stretch !important}.align-self-xl-auto{-ms-flex-item-align:auto !important;align-self:auto !important}.align-self-xl-start{-ms-flex-item-align:start !important;align-self:flex-start !important}.align-self-xl-end{-ms-flex-item-align:end !important;align-self:flex-end !important}.align-self-xl-center{-ms-flex-item-align:center !important;align-self:center !important}.align-self-xl-baseline{-ms-flex-item-align:baseline !important;align-self:baseline !important}.align-self-xl-stretch{-ms-flex-item-align:stretch !important;align-self:stretch !important}}.float-left{float:left !important}.float-right{float:right !important}.float-none{float:none !important}@media (min-width: 576px){.float-sm-left{float:left !important}.float-sm-right{float:right !important}.float-sm-none{float:none !important}}@media (min-width: 768px){.float-md-left{float:left !important}.float-md-right{float:right !important}.float-md-none{float:none !important}}@media (min-width: 992px){.float-lg-left{float:left !important}.float-lg-right{float:right !important}.float-lg-none{float:none !important}}@media (min-width: 1200px){.float-xl-left{float:left !important}.float-xl-right{float:right !important}.float-xl-none{float:none !important}}.overflow-auto{overflow:auto !important}.overflow-hidden{overflow:hidden !important}.position-static{position:static !important}.position-relative{position:relative !important}.position-absolute{position:absolute !important}.position-fixed{position:fixed !important}.position-sticky{position:-webkit-sticky !important;position:sticky !important}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}@supports (position: -webkit-sticky) or (position: sticky){.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}.sr-only{position:absolute;width:1px;height:1px;padding:0;overflow:hidden;clip:rect(0, 0, 0, 0);white-space:nowrap;border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;overflow:visible;clip:auto;white-space:normal}.shadow-sm{-webkit-box-shadow:0 0.125rem 0.25rem rgba(0,0,0,0.075) !important;box-shadow:0 0.125rem 0.25rem rgba(0,0,0,0.075) !important}.shadow{-webkit-box-shadow:0 0.5rem 1rem rgba(0,0,0,0.15) !important;box-shadow:0 0.5rem 1rem rgba(0,0,0,0.15) !important}.shadow-lg{-webkit-box-shadow:0 1rem 3rem rgba(0,0,0,0.175) !important;box-shadow:0 1rem 3rem rgba(0,0,0,0.175) !important}.shadow-none{-webkit-box-shadow:none !important;box-shadow:none !important}.w-25{width:25% !important}.w-50{width:50% !important}.w-75{width:75% !important}.w-100{width:100% !important}.w-auto{width:auto !important}.h-25{height:25% !important}.h-50{height:50% !important}.h-75{height:75% !important}.h-100{height:100% !important}.h-auto{height:auto !important}.mw-100{max-width:100% !important}.mh-100{max-height:100% !important}.min-vw-100{min-width:100vw !important}.min-vh-100{min-height:100vh !important}.vw-100{width:100vw !important}.vh-100{height:100vh !important}.m-0{margin:0 !important}.mt-0,.my-0{margin-top:0 !important}.mr-0,.mx-0{margin-right:0 !important}.mb-0,.my-0{margin-bottom:0 !important}.ml-0,.mx-0{margin-left:0 !important}.m-1{margin:0.25rem !important}.mt-1,.my-1{margin-top:0.25rem !important}.mr-1,.mx-1{margin-right:0.25rem !important}.mb-1,.my-1{margin-bottom:0.25rem !important}.ml-1,.mx-1{margin-left:0.25rem !important}.m-2{margin:0.5rem !important}.mt-2,.my-2{margin-top:0.5rem !important}.mr-2,.mx-2{margin-right:0.5rem !important}.mb-2,.my-2{margin-bottom:0.5rem !important}.ml-2,.mx-2{margin-left:0.5rem !important}.m-3{margin:1rem !important}.mt-3,.my-3{margin-top:1rem !important}.mr-3,.mx-3{margin-right:1rem !important}.mb-3,.my-3{margin-bottom:1rem !important}.ml-3,.mx-3{margin-left:1rem !important}.m-4{margin:1.5rem !important}.mt-4,.my-4{margin-top:1.5rem !important}.mr-4,.mx-4{margin-right:1.5rem !important}.mb-4,.my-4{margin-bottom:1.5rem !important}.ml-4,.mx-4{margin-left:1.5rem !important}.m-5{margin:3rem !important}.mt-5,.my-5{margin-top:3rem !important}.mr-5,.mx-5{margin-right:3rem !important}.mb-5,.my-5{margin-bottom:3rem !important}.ml-5,.mx-5{margin-left:3rem !important}.p-0{padding:0 !important}.pt-0,.py-0{padding-top:0 !important}.pr-0,.px-0{padding-right:0 !important}.pb-0,.py-0{padding-bottom:0 !important}.pl-0,.px-0{padding-left:0 !important}.p-1{padding:0.25rem !important}.pt-1,.py-1{padding-top:0.25rem !important}.pr-1,.px-1{padding-right:0.25rem !important}.pb-1,.py-1{padding-bottom:0.25rem !important}.pl-1,.px-1{padding-left:0.25rem !important}.p-2{padding:0.5rem !important}.pt-2,.py-2{padding-top:0.5rem !important}.pr-2,.px-2{padding-right:0.5rem !important}.pb-2,.py-2{padding-bottom:0.5rem !important}.pl-2,.px-2{padding-left:0.5rem !important}.p-3{padding:1rem !important}.pt-3,.py-3{padding-top:1rem !important}.pr-3,.px-3{padding-right:1rem !important}.pb-3,.py-3{padding-bottom:1rem !important}.pl-3,.px-3{padding-left:1rem !important}.p-4{padding:1.5rem !important}.pt-4,.py-4{padding-top:1.5rem !important}.pr-4,.px-4{padding-right:1.5rem !important}.pb-4,.py-4{padding-bottom:1.5rem !important}.pl-4,.px-4{padding-left:1.5rem !important}.p-5{padding:3rem !important}.pt-5,.py-5{padding-top:3rem !important}.pr-5,.px-5{padding-right:3rem !important}.pb-5,.py-5{padding-bottom:3rem !important}.pl-5,.px-5{padding-left:3rem !important}.m-n1{margin:-0.25rem !important}.mt-n1,.my-n1{margin-top:-0.25rem !important}.mr-n1,.mx-n1{margin-right:-0.25rem !important}.mb-n1,.my-n1{margin-bottom:-0.25rem !important}.ml-n1,.mx-n1{margin-left:-0.25rem !important}.m-n2{margin:-0.5rem !important}.mt-n2,.my-n2{margin-top:-0.5rem !important}.mr-n2,.mx-n2{margin-right:-0.5rem !important}.mb-n2,.my-n2{margin-bottom:-0.5rem !important}.ml-n2,.mx-n2{margin-left:-0.5rem !important}.m-n3{margin:-1rem !important}.mt-n3,.my-n3{margin-top:-1rem !important}.mr-n3,.mx-n3{margin-right:-1rem !important}.mb-n3,.my-n3{margin-bottom:-1rem !important}.ml-n3,.mx-n3{margin-left:-1rem !important}.m-n4{margin:-1.5rem !important}.mt-n4,.my-n4{margin-top:-1.5rem !important}.mr-n4,.mx-n4{margin-right:-1.5rem !important}.mb-n4,.my-n4{margin-bottom:-1.5rem !important}.ml-n4,.mx-n4{margin-left:-1.5rem !important}.m-n5{margin:-3rem !important}.mt-n5,.my-n5{margin-top:-3rem !important}.mr-n5,.mx-n5{margin-right:-3rem !important}.mb-n5,.my-n5{margin-bottom:-3rem !important}.ml-n5,.mx-n5{margin-left:-3rem !important}.m-auto{margin:auto !important}.mt-auto,.my-auto{margin-top:auto !important}.mr-auto,.mx-auto{margin-right:auto !important}.mb-auto,.my-auto{margin-bottom:auto !important}.ml-auto,.mx-auto{margin-left:auto !important}@media (min-width: 576px){.m-sm-0{margin:0 !important}.mt-sm-0,.my-sm-0{margin-top:0 !important}.mr-sm-0,.mx-sm-0{margin-right:0 !important}.mb-sm-0,.my-sm-0{margin-bottom:0 !important}.ml-sm-0,.mx-sm-0{margin-left:0 !important}.m-sm-1{margin:0.25rem !important}.mt-sm-1,.my-sm-1{margin-top:0.25rem !important}.mr-sm-1,.mx-sm-1{margin-right:0.25rem !important}.mb-sm-1,.my-sm-1{margin-bottom:0.25rem !important}.ml-sm-1,.mx-sm-1{margin-left:0.25rem !important}.m-sm-2{margin:0.5rem !important}.mt-sm-2,.my-sm-2{margin-top:0.5rem !important}.mr-sm-2,.mx-sm-2{margin-right:0.5rem !important}.mb-sm-2,.my-sm-2{margin-bottom:0.5rem !important}.ml-sm-2,.mx-sm-2{margin-left:0.5rem !important}.m-sm-3{margin:1rem !important}.mt-sm-3,.my-sm-3{margin-top:1rem !important}.mr-sm-3,.mx-sm-3{margin-right:1rem !important}.mb-sm-3,.my-sm-3{margin-bottom:1rem !important}.ml-sm-3,.mx-sm-3{margin-left:1rem !important}.m-sm-4{margin:1.5rem !important}.mt-sm-4,.my-sm-4{margin-top:1.5rem !important}.mr-sm-4,.mx-sm-4{margin-right:1.5rem !important}.mb-sm-4,.my-sm-4{margin-bottom:1.5rem !important}.ml-sm-4,.mx-sm-4{margin-left:1.5rem !important}.m-sm-5{margin:3rem !important}.mt-sm-5,.my-sm-5{margin-top:3rem !important}.mr-sm-5,.mx-sm-5{margin-right:3rem !important}.mb-sm-5,.my-sm-5{margin-bottom:3rem !important}.ml-sm-5,.mx-sm-5{margin-left:3rem !important}.p-sm-0{padding:0 !important}.pt-sm-0,.py-sm-0{padding-top:0 !important}.pr-sm-0,.px-sm-0{padding-right:0 !important}.pb-sm-0,.py-sm-0{padding-bottom:0 !important}.pl-sm-0,.px-sm-0{padding-left:0 !important}.p-sm-1{padding:0.25rem !important}.pt-sm-1,.py-sm-1{padding-top:0.25rem !important}.pr-sm-1,.px-sm-1{padding-right:0.25rem !important}.pb-sm-1,.py-sm-1{padding-bottom:0.25rem !important}.pl-sm-1,.px-sm-1{padding-left:0.25rem !important}.p-sm-2{padding:0.5rem !important}.pt-sm-2,.py-sm-2{padding-top:0.5rem !important}.pr-sm-2,.px-sm-2{padding-right:0.5rem !important}.pb-sm-2,.py-sm-2{padding-bottom:0.5rem !important}.pl-sm-2,.px-sm-2{padding-left:0.5rem !important}.p-sm-3{padding:1rem !important}.pt-sm-3,.py-sm-3{padding-top:1rem !important}.pr-sm-3,.px-sm-3{padding-right:1rem !important}.pb-sm-3,.py-sm-3{padding-bottom:1rem !important}.pl-sm-3,.px-sm-3{padding-left:1rem !important}.p-sm-4{padding:1.5rem !important}.pt-sm-4,.py-sm-4{padding-top:1.5rem !important}.pr-sm-4,.px-sm-4{padding-right:1.5rem !important}.pb-sm-4,.py-sm-4{padding-bottom:1.5rem !important}.pl-sm-4,.px-sm-4{padding-left:1.5rem !important}.p-sm-5{padding:3rem !important}.pt-sm-5,.py-sm-5{padding-top:3rem !important}.pr-sm-5,.px-sm-5{padding-right:3rem !important}.pb-sm-5,.py-sm-5{padding-bottom:3rem !important}.pl-sm-5,.px-sm-5{padding-left:3rem !important}.m-sm-n1{margin:-0.25rem !important}.mt-sm-n1,.my-sm-n1{margin-top:-0.25rem !important}.mr-sm-n1,.mx-sm-n1{margin-right:-0.25rem !important}.mb-sm-n1,.my-sm-n1{margin-bottom:-0.25rem !important}.ml-sm-n1,.mx-sm-n1{margin-left:-0.25rem !important}.m-sm-n2{margin:-0.5rem !important}.mt-sm-n2,.my-sm-n2{margin-top:-0.5rem !important}.mr-sm-n2,.mx-sm-n2{margin-right:-0.5rem !important}.mb-sm-n2,.my-sm-n2{margin-bottom:-0.5rem !important}.ml-sm-n2,.mx-sm-n2{margin-left:-0.5rem !important}.m-sm-n3{margin:-1rem !important}.mt-sm-n3,.my-sm-n3{margin-top:-1rem !important}.mr-sm-n3,.mx-sm-n3{margin-right:-1rem !important}.mb-sm-n3,.my-sm-n3{margin-bottom:-1rem !important}.ml-sm-n3,.mx-sm-n3{margin-left:-1rem !important}.m-sm-n4{margin:-1.5rem !important}.mt-sm-n4,.my-sm-n4{margin-top:-1.5rem !important}.mr-sm-n4,.mx-sm-n4{margin-right:-1.5rem !important}.mb-sm-n4,.my-sm-n4{margin-bottom:-1.5rem !important}.ml-sm-n4,.mx-sm-n4{margin-left:-1.5rem !important}.m-sm-n5{margin:-3rem !important}.mt-sm-n5,.my-sm-n5{margin-top:-3rem !important}.mr-sm-n5,.mx-sm-n5{margin-right:-3rem !important}.mb-sm-n5,.my-sm-n5{margin-bottom:-3rem !important}.ml-sm-n5,.mx-sm-n5{margin-left:-3rem !important}.m-sm-auto{margin:auto !important}.mt-sm-auto,.my-sm-auto{margin-top:auto !important}.mr-sm-auto,.mx-sm-auto{margin-right:auto !important}.mb-sm-auto,.my-sm-auto{margin-bottom:auto !important}.ml-sm-auto,.mx-sm-auto{margin-left:auto !important}}@media (min-width: 768px){.m-md-0{margin:0 !important}.mt-md-0,.my-md-0{margin-top:0 !important}.mr-md-0,.mx-md-0{margin-right:0 !important}.mb-md-0,.my-md-0{margin-bottom:0 !important}.ml-md-0,.mx-md-0{margin-left:0 !important}.m-md-1{margin:0.25rem !important}.mt-md-1,.my-md-1{margin-top:0.25rem !important}.mr-md-1,.mx-md-1{margin-right:0.25rem !important}.mb-md-1,.my-md-1{margin-bottom:0.25rem !important}.ml-md-1,.mx-md-1{margin-left:0.25rem !important}.m-md-2{margin:0.5rem !important}.mt-md-2,.my-md-2{margin-top:0.5rem !important}.mr-md-2,.mx-md-2{margin-right:0.5rem !important}.mb-md-2,.my-md-2{margin-bottom:0.5rem !important}.ml-md-2,.mx-md-2{margin-left:0.5rem !important}.m-md-3{margin:1rem !important}.mt-md-3,.my-md-3{margin-top:1rem !important}.mr-md-3,.mx-md-3{margin-right:1rem !important}.mb-md-3,.my-md-3{margin-bottom:1rem !important}.ml-md-3,.mx-md-3{margin-left:1rem !important}.m-md-4{margin:1.5rem !important}.mt-md-4,.my-md-4{margin-top:1.5rem !important}.mr-md-4,.mx-md-4{margin-right:1.5rem !important}.mb-md-4,.my-md-4{margin-bottom:1.5rem !important}.ml-md-4,.mx-md-4{margin-left:1.5rem !important}.m-md-5{margin:3rem !important}.mt-md-5,.my-md-5{margin-top:3rem !important}.mr-md-5,.mx-md-5{margin-right:3rem !important}.mb-md-5,.my-md-5{margin-bottom:3rem !important}.ml-md-5,.mx-md-5{margin-left:3rem !important}.p-md-0{padding:0 !important}.pt-md-0,.py-md-0{padding-top:0 !important}.pr-md-0,.px-md-0{padding-right:0 !important}.pb-md-0,.py-md-0{padding-bottom:0 !important}.pl-md-0,.px-md-0{padding-left:0 !important}.p-md-1{padding:0.25rem !important}.pt-md-1,.py-md-1{padding-top:0.25rem !important}.pr-md-1,.px-md-1{padding-right:0.25rem !important}.pb-md-1,.py-md-1{padding-bottom:0.25rem !important}.pl-md-1,.px-md-1{padding-left:0.25rem !important}.p-md-2{padding:0.5rem !important}.pt-md-2,.py-md-2{padding-top:0.5rem !important}.pr-md-2,.px-md-2{padding-right:0.5rem !important}.pb-md-2,.py-md-2{padding-bottom:0.5rem !important}.pl-md-2,.px-md-2{padding-left:0.5rem !important}.p-md-3{padding:1rem !important}.pt-md-3,.py-md-3{padding-top:1rem !important}.pr-md-3,.px-md-3{padding-right:1rem !important}.pb-md-3,.py-md-3{padding-bottom:1rem !important}.pl-md-3,.px-md-3{padding-left:1rem !important}.p-md-4{padding:1.5rem !important}.pt-md-4,.py-md-4{padding-top:1.5rem !important}.pr-md-4,.px-md-4{padding-right:1.5rem !important}.pb-md-4,.py-md-4{padding-bottom:1.5rem !important}.pl-md-4,.px-md-4{padding-left:1.5rem !important}.p-md-5{padding:3rem !important}.pt-md-5,.py-md-5{padding-top:3rem !important}.pr-md-5,.px-md-5{padding-right:3rem !important}.pb-md-5,.py-md-5{padding-bottom:3rem !important}.pl-md-5,.px-md-5{padding-left:3rem !important}.m-md-n1{margin:-0.25rem !important}.mt-md-n1,.my-md-n1{margin-top:-0.25rem !important}.mr-md-n1,.mx-md-n1{margin-right:-0.25rem !important}.mb-md-n1,.my-md-n1{margin-bottom:-0.25rem !important}.ml-md-n1,.mx-md-n1{margin-left:-0.25rem !important}.m-md-n2{margin:-0.5rem !important}.mt-md-n2,.my-md-n2{margin-top:-0.5rem !important}.mr-md-n2,.mx-md-n2{margin-right:-0.5rem !important}.mb-md-n2,.my-md-n2{margin-bottom:-0.5rem !important}.ml-md-n2,.mx-md-n2{margin-left:-0.5rem !important}.m-md-n3{margin:-1rem !important}.mt-md-n3,.my-md-n3{margin-top:-1rem !important}.mr-md-n3,.mx-md-n3{margin-right:-1rem !important}.mb-md-n3,.my-md-n3{margin-bottom:-1rem !important}.ml-md-n3,.mx-md-n3{margin-left:-1rem !important}.m-md-n4{margin:-1.5rem !important}.mt-md-n4,.my-md-n4{margin-top:-1.5rem !important}.mr-md-n4,.mx-md-n4{margin-right:-1.5rem !important}.mb-md-n4,.my-md-n4{margin-bottom:-1.5rem !important}.ml-md-n4,.mx-md-n4{margin-left:-1.5rem !important}.m-md-n5{margin:-3rem !important}.mt-md-n5,.my-md-n5{margin-top:-3rem !important}.mr-md-n5,.mx-md-n5{margin-right:-3rem !important}.mb-md-n5,.my-md-n5{margin-bottom:-3rem !important}.ml-md-n5,.mx-md-n5{margin-left:-3rem !important}.m-md-auto{margin:auto !important}.mt-md-auto,.my-md-auto{margin-top:auto !important}.mr-md-auto,.mx-md-auto{margin-right:auto !important}.mb-md-auto,.my-md-auto{margin-bottom:auto !important}.ml-md-auto,.mx-md-auto{margin-left:auto !important}}@media (min-width: 992px){.m-lg-0{margin:0 !important}.mt-lg-0,.my-lg-0{margin-top:0 !important}.mr-lg-0,.mx-lg-0{margin-right:0 !important}.mb-lg-0,.my-lg-0{margin-bottom:0 !important}.ml-lg-0,.mx-lg-0{margin-left:0 !important}.m-lg-1{margin:0.25rem !important}.mt-lg-1,.my-lg-1{margin-top:0.25rem !important}.mr-lg-1,.mx-lg-1{margin-right:0.25rem !important}.mb-lg-1,.my-lg-1{margin-bottom:0.25rem !important}.ml-lg-1,.mx-lg-1{margin-left:0.25rem !important}.m-lg-2{margin:0.5rem !important}.mt-lg-2,.my-lg-2{margin-top:0.5rem !important}.mr-lg-2,.mx-lg-2{margin-right:0.5rem !important}.mb-lg-2,.my-lg-2{margin-bottom:0.5rem !important}.ml-lg-2,.mx-lg-2{margin-left:0.5rem !important}.m-lg-3{margin:1rem !important}.mt-lg-3,.my-lg-3{margin-top:1rem !important}.mr-lg-3,.mx-lg-3{margin-right:1rem !important}.mb-lg-3,.my-lg-3{margin-bottom:1rem !important}.ml-lg-3,.mx-lg-3{margin-left:1rem !important}.m-lg-4{margin:1.5rem !important}.mt-lg-4,.my-lg-4{margin-top:1.5rem !important}.mr-lg-4,.mx-lg-4{margin-right:1.5rem !important}.mb-lg-4,.my-lg-4{margin-bottom:1.5rem !important}.ml-lg-4,.mx-lg-4{margin-left:1.5rem !important}.m-lg-5{margin:3rem !important}.mt-lg-5,.my-lg-5{margin-top:3rem !important}.mr-lg-5,.mx-lg-5{margin-right:3rem !important}.mb-lg-5,.my-lg-5{margin-bottom:3rem !important}.ml-lg-5,.mx-lg-5{margin-left:3rem !important}.p-lg-0{padding:0 !important}.pt-lg-0,.py-lg-0{padding-top:0 !important}.pr-lg-0,.px-lg-0{padding-right:0 !important}.pb-lg-0,.py-lg-0{padding-bottom:0 !important}.pl-lg-0,.px-lg-0{padding-left:0 !important}.p-lg-1{padding:0.25rem !important}.pt-lg-1,.py-lg-1{padding-top:0.25rem !important}.pr-lg-1,.px-lg-1{padding-right:0.25rem !important}.pb-lg-1,.py-lg-1{padding-bottom:0.25rem !important}.pl-lg-1,.px-lg-1{padding-left:0.25rem !important}.p-lg-2{padding:0.5rem !important}.pt-lg-2,.py-lg-2{padding-top:0.5rem !important}.pr-lg-2,.px-lg-2{padding-right:0.5rem !important}.pb-lg-2,.py-lg-2{padding-bottom:0.5rem !important}.pl-lg-2,.px-lg-2{padding-left:0.5rem !important}.p-lg-3{padding:1rem !important}.pt-lg-3,.py-lg-3{padding-top:1rem !important}.pr-lg-3,.px-lg-3{padding-right:1rem !important}.pb-lg-3,.py-lg-3{padding-bottom:1rem !important}.pl-lg-3,.px-lg-3{padding-left:1rem !important}.p-lg-4{padding:1.5rem !important}.pt-lg-4,.py-lg-4{padding-top:1.5rem !important}.pr-lg-4,.px-lg-4{padding-right:1.5rem !important}.pb-lg-4,.py-lg-4{padding-bottom:1.5rem !important}.pl-lg-4,.px-lg-4{padding-left:1.5rem !important}.p-lg-5{padding:3rem !important}.pt-lg-5,.py-lg-5{padding-top:3rem !important}.pr-lg-5,.px-lg-5{padding-right:3rem !important}.pb-lg-5,.py-lg-5{padding-bottom:3rem !important}.pl-lg-5,.px-lg-5{padding-left:3rem !important}.m-lg-n1{margin:-0.25rem !important}.mt-lg-n1,.my-lg-n1{margin-top:-0.25rem !important}.mr-lg-n1,.mx-lg-n1{margin-right:-0.25rem !important}.mb-lg-n1,.my-lg-n1{margin-bottom:-0.25rem !important}.ml-lg-n1,.mx-lg-n1{margin-left:-0.25rem !important}.m-lg-n2{margin:-0.5rem !important}.mt-lg-n2,.my-lg-n2{margin-top:-0.5rem !important}.mr-lg-n2,.mx-lg-n2{margin-right:-0.5rem !important}.mb-lg-n2,.my-lg-n2{margin-bottom:-0.5rem !important}.ml-lg-n2,.mx-lg-n2{margin-left:-0.5rem !important}.m-lg-n3{margin:-1rem !important}.mt-lg-n3,.my-lg-n3{margin-top:-1rem !important}.mr-lg-n3,.mx-lg-n3{margin-right:-1rem !important}.mb-lg-n3,.my-lg-n3{margin-bottom:-1rem !important}.ml-lg-n3,.mx-lg-n3{margin-left:-1rem !important}.m-lg-n4{margin:-1.5rem !important}.mt-lg-n4,.my-lg-n4{margin-top:-1.5rem !important}.mr-lg-n4,.mx-lg-n4{margin-right:-1.5rem !important}.mb-lg-n4,.my-lg-n4{margin-bottom:-1.5rem !important}.ml-lg-n4,.mx-lg-n4{margin-left:-1.5rem !important}.m-lg-n5{margin:-3rem !important}.mt-lg-n5,.my-lg-n5{margin-top:-3rem !important}.mr-lg-n5,.mx-lg-n5{margin-right:-3rem !important}.mb-lg-n5,.my-lg-n5{margin-bottom:-3rem !important}.ml-lg-n5,.mx-lg-n5{margin-left:-3rem !important}.m-lg-auto{margin:auto !important}.mt-lg-auto,.my-lg-auto{margin-top:auto !important}.mr-lg-auto,.mx-lg-auto{margin-right:auto !important}.mb-lg-auto,.my-lg-auto{margin-bottom:auto !important}.ml-lg-auto,.mx-lg-auto{margin-left:auto !important}}@media (min-width: 1200px){.m-xl-0{margin:0 !important}.mt-xl-0,.my-xl-0{margin-top:0 !important}.mr-xl-0,.mx-xl-0{margin-right:0 !important}.mb-xl-0,.my-xl-0{margin-bottom:0 !important}.ml-xl-0,.mx-xl-0{margin-left:0 !important}.m-xl-1{margin:0.25rem !important}.mt-xl-1,.my-xl-1{margin-top:0.25rem !important}.mr-xl-1,.mx-xl-1{margin-right:0.25rem !important}.mb-xl-1,.my-xl-1{margin-bottom:0.25rem !important}.ml-xl-1,.mx-xl-1{margin-left:0.25rem !important}.m-xl-2{margin:0.5rem !important}.mt-xl-2,.my-xl-2{margin-top:0.5rem !important}.mr-xl-2,.mx-xl-2{margin-right:0.5rem !important}.mb-xl-2,.my-xl-2{margin-bottom:0.5rem !important}.ml-xl-2,.mx-xl-2{margin-left:0.5rem !important}.m-xl-3{margin:1rem !important}.mt-xl-3,.my-xl-3{margin-top:1rem !important}.mr-xl-3,.mx-xl-3{margin-right:1rem !important}.mb-xl-3,.my-xl-3{margin-bottom:1rem !important}.ml-xl-3,.mx-xl-3{margin-left:1rem !important}.m-xl-4{margin:1.5rem !important}.mt-xl-4,.my-xl-4{margin-top:1.5rem !important}.mr-xl-4,.mx-xl-4{margin-right:1.5rem !important}.mb-xl-4,.my-xl-4{margin-bottom:1.5rem !important}.ml-xl-4,.mx-xl-4{margin-left:1.5rem !important}.m-xl-5{margin:3rem !important}.mt-xl-5,.my-xl-5{margin-top:3rem !important}.mr-xl-5,.mx-xl-5{margin-right:3rem !important}.mb-xl-5,.my-xl-5{margin-bottom:3rem !important}.ml-xl-5,.mx-xl-5{margin-left:3rem !important}.p-xl-0{padding:0 !important}.pt-xl-0,.py-xl-0{padding-top:0 !important}.pr-xl-0,.px-xl-0{padding-right:0 !important}.pb-xl-0,.py-xl-0{padding-bottom:0 !important}.pl-xl-0,.px-xl-0{padding-left:0 !important}.p-xl-1{padding:0.25rem !important}.pt-xl-1,.py-xl-1{padding-top:0.25rem !important}.pr-xl-1,.px-xl-1{padding-right:0.25rem !important}.pb-xl-1,.py-xl-1{padding-bottom:0.25rem !important}.pl-xl-1,.px-xl-1{padding-left:0.25rem !important}.p-xl-2{padding:0.5rem !important}.pt-xl-2,.py-xl-2{padding-top:0.5rem !important}.pr-xl-2,.px-xl-2{padding-right:0.5rem !important}.pb-xl-2,.py-xl-2{padding-bottom:0.5rem !important}.pl-xl-2,.px-xl-2{padding-left:0.5rem !important}.p-xl-3{padding:1rem !important}.pt-xl-3,.py-xl-3{padding-top:1rem !important}.pr-xl-3,.px-xl-3{padding-right:1rem !important}.pb-xl-3,.py-xl-3{padding-bottom:1rem !important}.pl-xl-3,.px-xl-3{padding-left:1rem !important}.p-xl-4{padding:1.5rem !important}.pt-xl-4,.py-xl-4{padding-top:1.5rem !important}.pr-xl-4,.px-xl-4{padding-right:1.5rem !important}.pb-xl-4,.py-xl-4{padding-bottom:1.5rem !important}.pl-xl-4,.px-xl-4{padding-left:1.5rem !important}.p-xl-5{padding:3rem !important}.pt-xl-5,.py-xl-5{padding-top:3rem !important}.pr-xl-5,.px-xl-5{padding-right:3rem !important}.pb-xl-5,.py-xl-5{padding-bottom:3rem !important}.pl-xl-5,.px-xl-5{padding-left:3rem !important}.m-xl-n1{margin:-0.25rem !important}.mt-xl-n1,.my-xl-n1{margin-top:-0.25rem !important}.mr-xl-n1,.mx-xl-n1{margin-right:-0.25rem !important}.mb-xl-n1,.my-xl-n1{margin-bottom:-0.25rem !important}.ml-xl-n1,.mx-xl-n1{margin-left:-0.25rem !important}.m-xl-n2{margin:-0.5rem !important}.mt-xl-n2,.my-xl-n2{margin-top:-0.5rem !important}.mr-xl-n2,.mx-xl-n2{margin-right:-0.5rem !important}.mb-xl-n2,.my-xl-n2{margin-bottom:-0.5rem !important}.ml-xl-n2,.mx-xl-n2{margin-left:-0.5rem !important}.m-xl-n3{margin:-1rem !important}.mt-xl-n3,.my-xl-n3{margin-top:-1rem !important}.mr-xl-n3,.mx-xl-n3{margin-right:-1rem !important}.mb-xl-n3,.my-xl-n3{margin-bottom:-1rem !important}.ml-xl-n3,.mx-xl-n3{margin-left:-1rem !important}.m-xl-n4{margin:-1.5rem !important}.mt-xl-n4,.my-xl-n4{margin-top:-1.5rem !important}.mr-xl-n4,.mx-xl-n4{margin-right:-1.5rem !important}.mb-xl-n4,.my-xl-n4{margin-bottom:-1.5rem !important}.ml-xl-n4,.mx-xl-n4{margin-left:-1.5rem !important}.m-xl-n5{margin:-3rem !important}.mt-xl-n5,.my-xl-n5{margin-top:-3rem !important}.mr-xl-n5,.mx-xl-n5{margin-right:-3rem !important}.mb-xl-n5,.my-xl-n5{margin-bottom:-3rem !important}.ml-xl-n5,.mx-xl-n5{margin-left:-3rem !important}.m-xl-auto{margin:auto !important}.mt-xl-auto,.my-xl-auto{margin-top:auto !important}.mr-xl-auto,.mx-xl-auto{margin-right:auto !important}.mb-xl-auto,.my-xl-auto{margin-bottom:auto !important}.ml-xl-auto,.mx-xl-auto{margin-left:auto !important}}.text-monospace{font-family:SFMono-Regular, Menlo, Monaco, Consolas, "Liberation Mono", "Courier New", monospace}.text-justify{text-align:justify !important}.text-wrap{white-space:normal !important}.text-nowrap{white-space:nowrap !important}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.text-left{text-align:left !important}.text-right{text-align:right !important}.text-center{text-align:center !important}@media (min-width: 576px){.text-sm-left{text-align:left !important}.text-sm-right{text-align:right !important}.text-sm-center{text-align:center !important}}@media (min-width: 768px){.text-md-left{text-align:left !important}.text-md-right{text-align:right !important}.text-md-center{text-align:center !important}}@media (min-width: 992px){.text-lg-left{text-align:left !important}.text-lg-right{text-align:right !important}.text-lg-center{text-align:center !important}}@media (min-width: 1200px){.text-xl-left{text-align:left !important}.text-xl-right{text-align:right !important}.text-xl-center{text-align:center !important}}.text-lowercase{text-transform:lowercase !important}.text-uppercase{text-transform:uppercase !important}.text-capitalize{text-transform:capitalize !important}.font-weight-light{font-weight:300 !important}.font-weight-lighter{font-weight:lighter !important}.font-weight-normal{font-weight:400 !important}.font-weight-bold{font-weight:700 !important}.font-weight-bolder{font-weight:bolder !important}.font-italic{font-style:italic !important}.text-white{color:#fff !important}.text-primary{color:#B58900 !important}a.text-primary:hover,a.text-primary:focus{color:#694f00 !important}.text-secondary{color:#839496 !important}a.text-secondary:hover,a.text-secondary:focus{color:#5e6d6f !important}.text-success{color:#2AA198 !important}a.text-success:hover,a.text-success:focus{color:#1a645f !important}.text-info{color:#268BD2 !important}a.text-info:hover,a.text-info:focus{color:#1a6091 !important}.text-warning{color:#CB4B16 !important}a.text-warning:hover,a.text-warning:focus{color:#86320f !important}.text-danger{color:#D33682 !important}a.text-danger:hover,a.text-danger:focus{color:#9b225c !important}.text-light{color:#FDF6E3 !important}a.text-light:hover,a.text-light:focus{color:#f8df9c !important}.text-dark{color:#073642 !important}a.text-dark:hover,a.text-dark:focus{color:black !important}.text-body{color:#839496 !important}.text-muted{color:#839496 !important}.text-black-50{color:rgba(0,0,0,0.5) !important}.text-white-50{color:rgba(255,255,255,0.5) !important}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.text-decoration-none{text-decoration:none !important}.text-reset{color:inherit !important}.visible{visibility:visible !important}.invisible{visibility:hidden !important}@media print{*,*::before,*::after{text-shadow:none !important;-webkit-box-shadow:none !important;box-shadow:none !important}a:not(.btn){text-decoration:underline}abbr[title]::after{content:" (" attr(title) ")"}pre{white-space:pre-wrap !important}pre,blockquote{border:1px solid #adb5bd;page-break-inside:avoid}thead{display:table-header-group}tr,img{page-break-inside:avoid}p,h2,h3{orphans:3;widows:3}h2,h3{page-break-after:avoid}@page{size:a3}body{min-width:992px !important}.container{min-width:992px !important}.navbar{display:none}.badge{border:1px solid #000}.table{border-collapse:collapse !important}.table td,.table th{background-color:#fff !important}.table-bordered th,.table-bordered td{border:1px solid #dee2e6 !important}.table-dark{color:inherit}.table-dark th,.table-dark td,.table-dark thead th,.table-dark tbody+tbody{border-color:#073642}.table .thead-dark th{color:inherit;border-color:#073642}}.btn-primary{background:#B58900 -webkit-gradient(linear, left top, left bottom, from(#c09b26), to(#B58900)) repeat-x;background:#B58900 linear-gradient(180deg, #c09b26, #B58900) repeat-x}.btn-secondary{background:#839496 -webkit-gradient(linear, left top, left bottom, from(#96a4a6), to(#839496)) repeat-x;background:#839496 linear-gradient(180deg, #96a4a6, #839496) repeat-x}.btn-success{background:#2AA198 -webkit-gradient(linear, left top, left bottom, from(#4aafa7), to(#2AA198)) repeat-x;background:#2AA198 linear-gradient(180deg, #4aafa7, #2AA198) repeat-x}.btn-info{background:#268BD2 -webkit-gradient(linear, left top, left bottom, from(#479cd9), to(#268BD2)) repeat-x;background:#268BD2 linear-gradient(180deg, #479cd9, #268BD2) repeat-x}.btn-warning{background:#CB4B16 -webkit-gradient(linear, left top, left bottom, from(#d36639), to(#CB4B16)) repeat-x;background:#CB4B16 linear-gradient(180deg, #d36639, #CB4B16) repeat-x}.btn-danger{background:#D33682 -webkit-gradient(linear, left top, left bottom, from(#da5495), to(#D33682)) repeat-x;background:#D33682 linear-gradient(180deg, #da5495, #D33682) repeat-x}.btn-light{background:#FDF6E3 -webkit-gradient(linear, left top, left bottom, from(#fdf7e7), to(#FDF6E3)) repeat-x;background:#FDF6E3 linear-gradient(180deg, #fdf7e7, #FDF6E3) repeat-x}.btn-dark{background:#073642 -webkit-gradient(linear, left top, left bottom, from(#2c545e), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #2c545e, #073642) repeat-x}.table-primary,.table-secondary,.table-dark,.table-success,.table-info,.table-warning,.table-danger{color:#fff}.table-primary,.table-primary>th,.table-primary>td{background-color:#B58900}.table-secondary,.table-secondary>th,.table-secondary>td{background-color:#839496}.table-light,.table-light>th,.table-light>td{background-color:#FDF6E3}.table-dark,.table-dark>th,.table-dark>td{background-color:#073642}.table-success,.table-success>th,.table-success>td{background-color:#2AA198}.table-info,.table-info>th,.table-info>td{background-color:#268BD2}.table-danger,.table-danger>th,.table-danger>td{background-color:#D33682}.table-warning,.table-warning>th,.table-warning>td{background-color:#CB4B16}.table-active,.table-active>th,.table-active>td{background-color:rgba(255,255,255,0.075)}.table-hover .table-primary:hover,.table-hover .table-primary:hover>th,.table-hover .table-primary:hover>td{background-color:#9c7600}.table-hover .table-secondary:hover,.table-hover .table-secondary:hover>th,.table-hover .table-secondary:hover>td{background-color:#75888a}.table-hover .table-light:hover,.table-hover .table-light:hover>th,.table-hover .table-light:hover>td{background-color:#fbeecb}.table-hover .table-dark:hover,.table-hover .table-dark:hover>th,.table-hover .table-dark:hover>td{background-color:#05232b}.table-hover .table-success:hover,.table-hover .table-success:hover>th,.table-hover .table-success:hover>td{background-color:#258d85}.table-hover .table-info:hover,.table-hover .table-info:hover>th,.table-hover .table-info:hover>td{background-color:#227dbc}.table-hover .table-danger:hover,.table-hover .table-danger:hover>th,.table-hover .table-danger:hover>td{background-color:#c42b75}.table-hover .table-warning:hover,.table-hover .table-warning:hover>th,.table-hover .table-warning:hover>td{background-color:#b44314}.table-hover .table-active:hover,.table-hover .table-active:hover>th,.table-hover .table-active:hover>td{background-color:rgba(255,255,255,0.075)}.alert{border:none;color:#fff}.alert a,.alert .alert-link{color:#fff;text-decoration:underline}.alert-primary{background:#B58900 -webkit-gradient(linear, left top, left bottom, from(#c09b26), to(#B58900)) repeat-x;background:#B58900 linear-gradient(180deg, #c09b26, #B58900) repeat-x}.alert-secondary{background:#839496 -webkit-gradient(linear, left top, left bottom, from(#96a4a6), to(#839496)) repeat-x;background:#839496 linear-gradient(180deg, #96a4a6, #839496) repeat-x}.alert-success{background:#2AA198 -webkit-gradient(linear, left top, left bottom, from(#4aafa7), to(#2AA198)) repeat-x;background:#2AA198 linear-gradient(180deg, #4aafa7, #2AA198) repeat-x}.alert-info{background:#268BD2 -webkit-gradient(linear, left top, left bottom, from(#479cd9), to(#268BD2)) repeat-x;background:#268BD2 linear-gradient(180deg, #479cd9, #268BD2) repeat-x}.alert-warning{background:#CB4B16 -webkit-gradient(linear, left top, left bottom, from(#d36639), to(#CB4B16)) repeat-x;background:#CB4B16 linear-gradient(180deg, #d36639, #CB4B16) repeat-x}.alert-danger{background:#D33682 -webkit-gradient(linear, left top, left bottom, from(#da5495), to(#D33682)) repeat-x;background:#D33682 linear-gradient(180deg, #da5495, #D33682) repeat-x}.alert-light{background:#FDF6E3 -webkit-gradient(linear, left top, left bottom, from(#fdf7e7), to(#FDF6E3)) repeat-x;background:#FDF6E3 linear-gradient(180deg, #fdf7e7, #FDF6E3) repeat-x}.alert-dark{background:#073642 -webkit-gradient(linear, left top, left bottom, from(#2c545e), to(#073642)) repeat-x;background:#073642 linear-gradient(180deg, #2c545e, #073642) repeat-x}.alert-light,.alert-light a:not(.btn),.alert-light .alert-link{color:#002B36} diff --git a/StaticRoot/dn42_logo.png b/StaticRoot/dn42_logo.png new file mode 100644 index 0000000..a4f2f8f Binary files /dev/null and b/StaticRoot/dn42_logo.png differ diff --git a/StaticRoot/explorer.js b/StaticRoot/explorer.js new file mode 100644 index 0000000..a336e86 --- /dev/null +++ b/StaticRoot/explorer.js @@ -0,0 +1,352 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry Explorer +////////////////////////////////////////////////////////////////////////// + +// global store for data that is loaded once +const GlobalStore = { + data: { + RegStats: null, + Index: null + } +} + + +////////////////////////////////////////////////////////////////////////// +// registry object component + +Vue.component('reg-object', { + template: '#reg-object-template', + props: [ 'link', 'content' ], + data() { + return { } + }, + methods: { + updateSearch: function(str) { + vm.updateSearch(str) + } + }, + computed: { + rtype: function() { + var ix = this.link.indexOf("/") + return this.link.substring(0, ix) + }, + obj: function() { + var ix = this.link.indexOf("/") + return this.link.substring(ix + 1) + } + } +}) + +////////////////////////////////////////////////////////////////////////// +// reg-attribute component + +Vue.component('reg-attribute', { + template: '#reg-attribute-template', + props: [ 'content' ], + data() { + return { } + }, + methods: { + isRegObject: function(str) { + return (this.content.match(/^\[.*?\]\(.*?\)/) != null) + } + }, + computed: { + objectLink: function() { + reg = this.content.match(/^\[(.*?)\]\((.*?)\)/) + return reg[2] + }, + decorated: function() { + var c = this.content + + // an attribute terminated with \n indicates a blank + // trailing line, however a single trailing
will + // not be rendered in HTML, this hack doubles up the + // trailing newline so it creates a
pair + // which renders as the blank line + if (c.substr(c.length-1) == "\n") { + c = c + "\n" + } + + // replace newlines with line breaks + c = c.replace(/\n/g, "
") + + // decorate + c = anchorme(c, { + truncate: 40, + ips: false, + attributes: [ { name: "target", value: "_blank" } ] + }) + + // and return the final decorated content + return c + } + } +}) + +////////////////////////////////////////////////////////////////////////// +// search input component + +Vue.component('search-input', { + template: '#search-input-template', + data() { + return { + search: '', + searchTimeout: 0 + } + }, + methods: { + + debounceSearch: function(value) { + if (this.searchTimeout) { + clearTimeout(this.searchTimeout) + } + + // link should be an absolute path + value = '/' + value + + this.searchTimeout = setTimeout( + this.$router.push.bind(this.$router, value), 500 + ) + } + }, + mounted() { + + // listen to search updates and set the search text appropriately + this.$root.$on('SearchChanged', value => { + this.search = value + }) + } +}) + +////////////////////////////////////////////////////////////////////////// +// registry-stats component + +Vue.component('registry-stats', { + template: '#registry-stats-template', + data() { + return { + state: null, + error: null, + store: GlobalStore.data + } + }, + methods: { + + // just fetch the stats from the API server via axios + reload(event) { + this.store.RegStats = null + this.state = "loading" + + axios + .get('/api/registry/') + .then(response => { + this.store.RegStats = response.data + this.state = 'complete' + }) + .catch(error => { + this.error = error + this.state = 'error' + console.log(error) + }) + } + }, + mounted() { + if (this.store.RegStats == null) { + this.reload() + } + } +}) + +////////////////////////////////////////////////////////////////////////// +// root component + +Vue.component('app-root', { + template: '#app-root-template', + mounted() { + this.$root.$emit('SearchChanged', '') + } +}) + +////////////////////////////////////////////////////////////////////////// +// search results + +Vue.component('app-search', { + template: '#app-search-template', + data() { + return { + state: null, + search: null, + results: null, + store: GlobalStore.data + } + }, + + // trigger a search on route update and on mount + beforeRouteUpdate(to, from, next) { + this.search = to.params.pathMatch + this.doSearch() + next() + }, + mounted() { + // store the search for later + this.search = this.$route.params.pathMatch + + // the index must be loaded before any real work takes place + if (this.store.Index == null) { + this.loadIndex() + } + else { + // index was already loaded, go search + this.doSearch() + } + }, + + methods: { + + // load the search index from the API + loadIndex() { + + this.state = 'loading' + this.$root.$emit('SearchChanged', 'Initialising ...') + + axios + .get('/api/registry/*') + .then(response => { + this.store.Index = response.data + + // if a query parameter has been passed, + // then go search + if (this.search != null) { + this.$nextTick(this.doSearch()) + } + }) + .catch(error => { + this.state = 'error' + console.log(error) + }) + }, + + // substring match object names against a search + matchObjects(objects, rtype, term) { + var results = [ ] + + // check each object + for (const obj in objects) { + + // matches are all lower case + var s = objects[obj].toLowerCase() + + var pos = s.indexOf(term) + if (pos != -1) { + if ((pos == 0) && (s == term)) { + // exact match, return just this result + return [[ rtype, objects[obj] ]] + } + results.push([ rtype, objects[obj] ]) + } + } + + return results + }, + + + // filter the index to find matches + searchFilter() { + + var results = [ ] + var index = this.store.Index + + // comparisons are lowercase + var term = this.search.toLowerCase() + + // check if search includes a '/' + var slash = term.indexOf('/') + if (slash != -1) { + // match only on the specific type + var rtype = term.substring(0, slash) + var term = term.substring(slash + 1) + objects = index[rtype] + + if (objects != null) { + results = this.matchObjects(objects, rtype, term) + } + } + else { + + // walk though the entire index + for (const rtype in index) { + var objlist = this.matchObjects(index[rtype], rtype, term) + results = results.concat(objlist) + } + } + + return results + + }, + + // perform the search and present results + doSearch() { + // notify other components that the search is updated + this.$root.$emit('SearchChanged', this.search) + + // filter matches against the index + filtered = this.searchFilter() + + // got nothing ? + if (filtered.length == 0) { + this.state = "noresults" + } + + // just one result + else if (filtered.length == 1) { + var objname = filtered[0] + + // load the object from the API + this.state = 'loading' + query = '/api/registry/' + objname[0] + '/' + objname[1] + + axios + .get(query) + .then(response => { + this.state = 'result' + this.result = response.data + }) + .catch(error => { + this.error = error + this.state = 'error' + }) + } + + // lots of results + else { + this.state = "resultlist" + this.result = filtered + } + } + } +}) + +////////////////////////////////////////////////////////////////////////// +// main vue application starts here + +// initialise the Vue Router +const router = new VueRouter({ + routes: [ + { path: '/', component: Vue.component('app-root') }, + { path: '/*', component: Vue.component('app-search') } + ] +}) + +// and the main app instance +const vm = new Vue({ + el: '#explorer_app', + data: { + + }, + router +}) + + +////////////////////////////////////////////////////////////////////////// +// end of code diff --git a/StaticRoot/index.html b/StaticRoot/index.html new file mode 100644 index 0000000..71bb188 --- /dev/null +++ b/StaticRoot/index.html @@ -0,0 +1,218 @@ + + + + DN42 Registry Explorer + + + + + + + + + +
+ + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/contrib/build.sh b/contrib/build.sh new file mode 100755 index 0000000..f5d2a45 --- /dev/null +++ b/contrib/build.sh @@ -0,0 +1,26 @@ +#!/bin/bash +########################################################################## +# A small script to build a static dn42regsrv image +# using the golang container image +# +# the binary will be built in to the current directory +########################################################################## +RUNTIME=$(which podman || which docker) +echo "Using container runtime: ${RUNTIME}" + +# find the source directory +SCRIPTPATH="$(cd "$(dirname "$0")" ; pwd -P)" +SOURCEPATH="$(cd "${SCRIPTPATH}/../"; pwd -P)" +echo "Source is in: ${SOURCEPATH}" + +# do the thing +${RUNTIME} run --rm \ + -e CGO_ENABLED=0 \ + -v "${SOURCEPATH}:/go/src/dn42regsrv" \ + -v "${PWD}:/go/bin" \ + -w "/go/src/dn42regsrv" \ + docker.io/golang:1.12 \ + go get + +########################################################################## +# end of code diff --git a/contrib/buildah.sh b/contrib/buildah.sh new file mode 100755 index 0000000..bb474e0 --- /dev/null +++ b/contrib/buildah.sh @@ -0,0 +1,65 @@ +#!/bin/bash +########################################################################## +echo "Building dn42regsrv container" + +# find the source directory +SCRIPTPATH="$(cd "$(dirname "$0")" ; pwd -P)" +SOURCEPATH="$(cd "${SCRIPTPATH}/../"; pwd -P)" +echo "Source is in: ${SOURCEPATH}" + +########################################################################## + +DEPS='git' +B=$(which buildah) + +# initialise container +c=$(buildah from --name dn42regsrv-working docker.io/debian:buster) + +########################################################################## + +# install dependencies and initialise directories +$B run $c -- bash < +function replace_rr { + local rr_name=$1; shift + local rr_type=$1; shift + local rr_content=$* + + echo "Replace: ${rr_name} ${rr_type} '${rr_content}'" + + if [ ${DEBUG} -eq 0 ] + then + ${PDNSUTIL} replace-rrset . ${rr_name} ${rr_type} "${rr_content}" + fi +} + + +# delete_rr +function delete_rr { + local rr_name=$1 + local rr_type=$2 + + echo "Delete: ${rr_name} ${rr_type}" + + if [ ${DEBUG} -eq 0 ] + then + ${PDNSUTIL} delete-rrset . ${rr_name} ${rr_type} + fi +} + +# list the current contents of the root zone +function get_current_root_zone { + local rra rr_name rr_type rr_content + while read -r -a rra + do + rr_name=${rra[0]} + rr_type=${rra[3]} + rr_content=${rra[@]:4} + + current_rz["${rr_name} ${rr_type}"]=${rr_content} + + done < <(${PDNSUTIL} list-zone .) +} + +# update the . SOA record after a change +function update_soa { + echo "Incrementing SOA serial" + if [ ${DEBUG} -eq 0 ] + then + ${PDNSUTIL} increase-serial . + fi +} + +# used to trigger a notify to any slaves of this server +function notify_slaves { + echo "Notfying slaves" + if [ ${DEBUG} -eq 0 ] + then + ${PDNSCTRL} notify . + fi +} + +########################################################################## +# No further local customisation should be needed from here +########################################################################## + +# initialise script parameters and global vars + +function usage { + echo "Usage: $0 [-h] [-d] [-a URL] [-c FILE]" + echo " -h: this help" + echo " -d: enable debugging and don't action changes" + echo " -a: URL to dn42regsrv API" + echo " -c: file in which to store previous commit number" +} + +# default options +DEBUG=0 +APIURL="http://explorer.burble.dn42/api" +COMMITFILE="/tmp/.sync_rz_commit" + +# parse any arguments passed to the script +while getopts ":hda:" opt +do + case ${opt} in + d) + DEBUG=1 + ;; + a) + APIURL=${OPTARG} + ;; + *) + usage + exit 0 + ;; + esac +done + +# global vars +declare -A current_rz +declare -A new_rz +current_commit='' +new_commit='' +deleted_records=0 +updated_records=0 + +########################################################################## +# fetch and parse the root zone data from the API + +function fetch_new_root_zone { + local line fields rr_name rr_type rr_content + while read -r line + do + if [[ ${line} == ';; Commit Reference:'* ]] + then + new_commit=${line#;; Commit Reference: } + else + # strip out comments and create array + fields=( ${line%%;*} ) + + # if the line is a valid record + if [ ${#fields[@]} -ge 4 ] + then + rr_name=${fields[0]} + rr_type=${fields[2]} + rr_content=${fields[@]:3} + new_rz["${rr_name} ${rr_type}"]=${rr_content} + fi + fi + done < <(/usr/bin/wget -O - -q "${APIURL}/dns/root-zone?format=bind") + + if [ ${DEBUG} -eq 1 ]; then echo "New Commit: ${new_commit}"; fi +} + +########################################################################## +# load and store the previous commit + +function load_current_commit { + read -r current_commit < "${COMMITFILE}" + if [ ${DEBUG} -eq 1 ]; then echo "Current Commit: ${current_commit}"; fi +} + +function store_current_commit { + if [ ${DEBUG} -eq 0 ] + then + echo "${1}" > "${COMMITFILE}" + fi +} + +########################################################################## +# remove records that have been deleted + +function is_ignored { + local rr_name=$1 + for i in "${IGNORE_RECORDS[@]}" + do + [ "${i}" == "${rr_name}" ] && echo "ignored" + done + echo "" +} + +function remove_deleted_records { + local key rr_name ignored + deleted_records=0 + + # check each record in the old root zone + for key in "${!current_rz[@]}" + do + ignored=$(is_ignored ${key% }) + + # if record is not ignored, and no new record exists + if [ "${ignored}" == '' -a "${new_rz[${key}]}" == '' ] + then + # then get rid of it + delete_rr ${key} + deleted_records=$((deleted_records + 1)) + fi + done + + echo "Deleted ${deleted_records} records" +} + +########################################################################## +# update records that have been added or changed + +function update_new_records { + local key content + updated_records=0 + + # check each new record + for key in "${!new_rz[@]}" + do + content="${new_rz[${key}]}" + + # if old record didn't exist, or content differs + if [ "${current_rz[${key}]}" != "${content}" ] + then + # update the record + replace_rr $key ${content} + updated_records=$((updated_records + 1)) + fi + + done + + echo "Updated ${updated_records} records" +} + +########################################################################## +# main flow of the script starts here + +echo "DN42 Root Zone Sync" +date +echo + +fetch_new_root_zone + +# check that the commit was populated +if [ "${new_commit}" == '' ] +then + echo "Unable to fetch new root zone, aborting" + exit 1 +fi + +load_current_commit + +# now check if anything actually needs to be done +if [ "${new_commit}" == "${current_commit}" ] +then + echo "Commits are equal, nothing to do" + exit 0 +fi + +get_current_root_zone + +# apply changes +remove_deleted_records +update_new_records + +# bail out if there were no actual differences +if [ $((deleted_records + updated_records)) -eq 0 ] +then + echo "No records were updated, exiting" +else + + # update the SOA and send out a notification to slaves + update_soa + notify_slaves + +fi + +# finally store the new commit to show it's been updated +store_current_commit "${new_commit}" + +echo "All done" + +########################################################################## +# end of code diff --git a/dn42regsrv.go b/dn42regsrv.go new file mode 100644 index 0000000..f5edcfb --- /dev/null +++ b/dn42regsrv.go @@ -0,0 +1,202 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry API Server +////////////////////////////////////////////////////////////////////////// + +package main + +////////////////////////////////////////////////////////////////////////// + +import ( + "context" + "encoding/json" + "github.com/gorilla/mux" + log "github.com/sirupsen/logrus" + flag "github.com/spf13/pflag" + "net/http" + "os" + "os/signal" + "time" +) + +////////////////////////////////////////////////////////////////////////// +// simple event bus + +type NotifyFunc func(...interface{}) +type SimpleEventBus map[string][]NotifyFunc + +var EventBus = make(SimpleEventBus) + +// add a listener to an event +func (bus SimpleEventBus) Listen(event string, nfunc NotifyFunc) { + bus[event] = append(bus[event], nfunc) +} + +// fire notifications for an event +func (bus SimpleEventBus) Fire(event string, params ...interface{}) { + funcs := bus[event] + if funcs != nil { + for _, nfunc := range funcs { + nfunc(params...) + } + } +} + +////////////////////////////////////////////////////////////////////////// +// utility func for returning JSON from an API endpoint + +func ResponseJSON(w http.ResponseWriter, v interface{}) { + + // for response time testing + //time.Sleep(time.Second) + + // marshal the JSON string + data, err := json.Marshal(v) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + }).Error("Failed to marshal JSON") + } + + // write back to http handler + w.Header().Set("Content-Type", "application/json") + w.Header().Set("Access-Control-Allow-Origin", "*") + w.Write(data) +} + +////////////////////////////////////////////////////////////////////////// +// utility function to set the log level + +func setLogLevel(levelStr string) { + + if level, err := log.ParseLevel(levelStr); err != nil { + // failed to set the level + + // set a sensible default and, of course, log the error + log.SetLevel(log.InfoLevel) + log.WithFields(log.Fields{ + "loglevel": levelStr, + "error": err, + }).Error("Failed to set requested log level") + + } else { + + // set the requested level + log.SetLevel(level) + + } +} + +////////////////////////////////////////////////////////////////////////// +// http request logger + +func requestLogger(next http.Handler) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + + log.WithFields(log.Fields{ + "method": r.Method, + "URL": r.URL.String(), + "Remote": r.RemoteAddr, + }).Debug("HTTP Request") + + next.ServeHTTP(w, r) + }) +} + +////////////////////////////////////////////////////////////////////////// +// everything starts here + +func main() { + + // set a default log level, so that logging can be used immediately + // the level will be overidden later on once the command line + // options are loaded + log.SetLevel(log.InfoLevel) + log.Info("DN42 Registry API Server Starting") + + // declare cmd line options + var ( + logLevel = flag.StringP("LogLevel", "l", "Info", "Log level") + regDir = flag.StringP("RegDir", "d", "registry", "Registry data directory") + bindAddress = flag.StringP("BindAddress", "b", "[::]:8042", "Server bind address") + staticRoot = flag.StringP("StaticRoot", "s", "StaticRoot", "Static page directory") + refreshInterval = flag.StringP("Refresh", "i", "60m", "Refresh interval") + gitPath = flag.StringP("GitPath", "g", "/usr/bin/git", "Path to git executable") + autoPull = flag.BoolP("AutoPull", "a", true, "Automatically pull the registry") + branch = flag.StringP("Branch", "p", "master", "git branch to pull") + ) + flag.Parse() + + // now initialise logging properly based on the cmd line options + setLogLevel(*logLevel) + + // parse the refreshInterval and start data collection + interval, err := time.ParseDuration(*refreshInterval) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "interval": *refreshInterval, + }).Fatal("Unable to parse registry refresh interval") + } + + InitialiseRegistryData(*regDir, interval, + *gitPath, *autoPull, *branch) + + // initialise router + router := mux.NewRouter() + // log all access + router.Use(requestLogger) + + // add API routes + subr := router.PathPrefix("/api").Subrouter() + EventBus.Fire("APIEndpoint", subr) + + // initialise static routes + InstallStaticRoutes(router, *staticRoot) + + // initialise http server + server := &http.Server{ + Addr: *bindAddress, + WriteTimeout: time.Second * 15, + ReadTimeout: time.Second * 15, + IdleTimeout: time.Second * 60, + Handler: router, + } + + // run the server in a non-blocking goroutine + + log.WithFields(log.Fields{ + "BindAddress": *bindAddress, + }).Info("Starting server") + + go func() { + if err := server.ListenAndServe(); err != nil { + log.WithFields(log.Fields{ + "error": err, + "BindAddress": *bindAddress, + }).Fatal("Unable to start server") + } + }() + + // graceful shutdown via SIGINT (^C) + csig := make(chan os.Signal, 1) + signal.Notify(csig, os.Interrupt) + + // and block + <-csig + + log.Info("Server shutting down") + + // deadline for server to shutdown + ctx, cancel := context.WithTimeout(context.Background(), 10) + defer cancel() + + // shutdown the server + server.Shutdown(ctx) + + // nothing left to do + log.Info("Shutdown complete, all done") + os.Exit(0) +} + +////////////////////////////////////////////////////////////////////////// +// end of code diff --git a/dnsapi.go b/dnsapi.go new file mode 100644 index 0000000..da01c4c --- /dev/null +++ b/dnsapi.go @@ -0,0 +1,234 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry API Server +////////////////////////////////////////////////////////////////////////// + +package main + +////////////////////////////////////////////////////////////////////////// + +import ( + "fmt" + "github.com/gorilla/mux" + log "github.com/sirupsen/logrus" + // "math/big" + "net/http" + "strings" + "time" +) + +////////////////////////////////////////////////////////////////////////// +// register the api + +func init() { + EventBus.Listen("APIEndpoint", InitDNSAPI) + EventBus.Listen("RegistryUpdate", DNSUpdate) +} + +////////////////////////////////////////////////////////////////////////// +// data model + +// very simple DNS record data +type DNSRecord struct { + Name string + Type string + Content string + Comment string `json:",omitempty"` +} + +type DNSZone struct { + Records []*DNSRecord + Commit string + Generated time.Time +} + +var DNSRootZone *DNSZone + +////////////////////////////////////////////////////////////////////////// +// fixed set of authoritative zones + +var DNSRootAuthZones = map[string]string{ + "dn42": "domain/dn42", + "recursive-servers.dn42": "domain/recursive-servers.dn42", + "delegation-servers.dn42": "domain/delegation-servers.dn42", + "d.f.ip6.arpa": "inet6num/fd00::_8", + "20.172.in-addr.arpa": "inetnum/172.20.0.0_16", + "21.172.in-addr.arpa": "inetnum/172.21.0.0_16", + "22.172.in-addr.arpa": "inetnum/172.22.0.0_16", + "23.172.in-addr.arpa": "inetnum/172.23.0.0_16", + "31.172.in-addr.arpa": "inetnum/172.31.0.0_16", + "10.in-addr.arpa": "inetnum/10.0.0.0_8", +} + +////////////////////////////////////////////////////////////////////////// +// called from main to initialise the API routing + +func InitDNSAPI(params ...interface{}) { + + router := params[0].(*mux.Router) + + s := router. + Methods("GET"). + PathPrefix("/dns"). + Subrouter() + + s.HandleFunc("/root-zone", dnsRZoneHandler) + + log.Info("DNS API installed") +} + +////////////////////////////////////////////////////////////////////////// +// api handlers + +// return records that should be included in a DN42 root zone +func dnsRZoneHandler(w http.ResponseWriter, r *http.Request) { + + var format []string + query := r.URL.Query() + format = query["format"] + if format == nil || len(format) != 1 { + format = []string{"json"} + } + + switch format[0] { + case "bind": + DNSRootZone.WriteBindFormat(w) + + case "json": + ResponseJSON(w, DNSRootZone) + + default: + ResponseJSON(w, DNSRootZone) + } +} + +////////////////////////////////////////////////////////////////////////// +// called whenever the registry is updated + +func DNSUpdate(params ...interface{}) { + + registry := params[0].(*Registry) + // path := params[1].(string) + + zone := &DNSZone{ + Generated: time.Now(), + Commit: registry.Commit, + } + + // add zones that are authoritative within DN42 + for name, object := range DNSRootAuthZones { + zone.AddRecords(registry, name, object, "DN42 Authoritative Zone") + } + + // search all domain objects and add stub records for each TLD + rtype := registry.Types["domain"] + for name, object := range rtype.Objects { + // domain is a TLD if it doesn't contain a '.' + if strings.IndexRune(name, '.') == -1 { + // don't include zones which are authoritative within DN42 + if DNSRootAuthZones[name] == "" { + zone.AddRecords(registry, name, object.Ref, "Forward Zone") + } + } + } + + DNSRootZone = zone +} + +////////////////////////////////////////////////////////////////////////// +// utility function to add a DNS record to a zone + +func (zone *DNSZone) AddRecord(name string, t string, + content string, comment string) { + record := &DNSRecord{ + Name: name, + Type: t, + Content: content, + Comment: comment, + } + zone.Records = append(zone.Records, record) +} + +////////////////////////////////////////////////////////////////////////// +// add nserver and ds-rdata records from a registry object + +func (zone *DNSZone) AddRecords(registry *Registry, name string, + path string, comment string) { + + // use the registry metadata key index to find the appropriate values + object := registry.GetObject(path) + if object == nil { + log.WithFields(log.Fields{ + "zone": name, + "path": path, + }).Error("DNS: unable to find object in registry") + return + } + + nserver := object.GetKey("nserver") + for _, ns := range nserver { + // check if stub record needs to be added + fields := strings.Split(ns.RawValue, " ") + if len(fields) == 2 { + // add a record for the NS, together with a stub A or AAAA record + + var stubtype string + if strings.IndexRune(fields[1], ':') == -1 { + // no : so IPv4 + stubtype = "A" + } else { + // has : so IPv6 + stubtype = "AAAA" + } + + zone.AddRecord(name, "NS", fields[0]+".", comment) + zone.AddRecord(fields[0], stubtype, fields[1], comment) + + } else { + // no, just add an NS record as it was presented + zone.AddRecord(name, "NS", ns.RawValue+".", comment) + } + + } + + dsrdata := object.GetKey("ds-rdata") + for _, ds := range dsrdata { + zone.AddRecord(name, "DS", ds.RawValue, comment) + } + +} + +////////////////////////////////////////////////////////////////////////// +// Functions for outputting zone records in different formats + +func (r *DNSRecord) ToBindString() string { + var comment string + if r.Comment == "" { + comment = "" + } else { + comment = "\t; " + r.Comment + } + + return fmt.Sprintf("%s\tIN\t%s\t%s%s", + r.Name, r.Type, r.Content, comment, + ) +} + +func (zone *DNSZone) WriteBindFormat(w http.ResponseWriter) { + + w.Header().Set("Content-Type", "text/plain") + w.Header().Set("Access-Control-Allow-Origin", "*") + + // provide a header + fmt.Fprintf(w, ";; DN42 Root Zone Records\n"+ + ";; Commit Reference: %s\n;; Generated: %s\n", + zone.Commit, zone.Generated) + + // then simply output each record in turn + for _, record := range zone.Records { + fmt.Fprintln(w, record.ToBindString()) + } + +} + +////////////////////////////////////////////////////////////////////////// +// end of code diff --git a/regapi.go b/regapi.go new file mode 100644 index 0000000..97eb369 --- /dev/null +++ b/regapi.go @@ -0,0 +1,536 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry API Server +////////////////////////////////////////////////////////////////////////// + +package main + +////////////////////////////////////////////////////////////////////////// + +import ( + // "fmt" + "github.com/gorilla/mux" + log "github.com/sirupsen/logrus" + "net/http" + "strings" + // "time" +) + +////////////////////////////////////////////////////////////////////////// +// data structures + +type RegMetaReturn struct { + Commit string +} + +////////////////////////////////////////////////////////////////////////// +// register the api + +func init() { + EventBus.Listen("APIEndpoint", InitRegistryAPI) +} + +////////////////////////////////////////////////////////////////////////// +// called from main to initialise the API routing + +func InitRegistryAPI(params ...interface{}) { + + router := params[0].(*mux.Router) + + s := router. + Methods("GET"). + PathPrefix("/registry"). + Subrouter() + + s.HandleFunc("/", regRootHandler) + //s.HandleFunc("/.schema", rTypeListHandler) + //s.HandleFunc("/.meta/", rTypeListHandler) + + s.HandleFunc("/.meta", regMetaHandler) + s.HandleFunc("/{type}", regTypeHandler) + s.HandleFunc("/{type}/{object}", regObjectHandler) + s.HandleFunc("/{type}/{object}/{key}", regKeyHandler) + s.HandleFunc("/{type}/{object}/{key}/{attribute}", regAttributeHandler) + + log.Info("Registry API installed") +} + +////////////////////////////////////////////////////////////////////////// +// return registry metadata + +func regMetaHandler(w http.ResponseWriter, r *http.Request) { + + rv := RegMetaReturn{ + Commit: RegistryData.Commit, + } + + ResponseJSON(w, rv) +} + +////////////////////////////////////////////////////////////////////////// +// filter functions + +// return a list of types that match the filter +func filterTypes(filter string) []*RegType { + + var rtypes []*RegType = nil + + // check if filter starts with '*' + if filter[0] == '*' { + // try and match the filter against all reg types + + filter = strings.ToLower(filter[1:]) + + // special case, if the filter was '*' return all types + if len(filter) == 0 { + + rtypes = make([]*RegType, 0, len(RegistryData.Types)) + for _, rtype := range RegistryData.Types { + rtypes = append(rtypes, rtype) + } + + } else { + + // otherwise substring match the types + for _, rtype := range RegistryData.Types { + lname := strings.ToLower(rtype.Ref) + if strings.Contains(lname, filter) { + // matched, add it to the list + rtypes = append(rtypes, rtype) + } + } + + } + + } else { + // perform an exact match with one entry + + rtype := RegistryData.Types[filter] + if rtype != nil { + // return a single answer + rtypes = []*RegType{rtype} + } + + } + + return rtypes +} + +// return a list of objects from a set of types that match a filter +func filterObjects(rtypes []*RegType, filter string) []*RegObject { + + var objects []*RegObject = nil + + // check if filter starts with '*' + if filter[0] == '*' { + // try and match objects against the filter + + filter = strings.ToLower(filter[1:]) + + // for each type + for _, rtype := range rtypes { + + // special case, if the filter was '*' return all objects + if len(filter) == 0 { + + objs := make([]*RegObject, 0, len(rtype.Objects)) + for _, object := range rtype.Objects { + objs = append(objs, object) + } + objects = append(objects, objs...) + + } else { + // otherwise substring match the object names + + for _, object := range rtype.Objects { + lname := strings.ToLower(object.Ref) + if strings.Contains(lname, filter) { + // matched, add it to the list + objects = append(objects, object) + } + } + + } + + } + + } else { + // perform an exact match against one object for each type + + for _, rtype := range rtypes { + + object := rtype.Objects[filter] + if object != nil { + // add the object + objects = append(objects, object) + } + } + + } + + return objects +} + +// return a list of key indices matching the filter +func filterKeys(rtypes []*RegType, filter string) []*RegKeyIndex { + + var ix []*RegKeyIndex = nil + + // check if filter starts with '*' + if filter[0] == '*' { + // try and match keys against the filter + + filter = strings.ToLower(filter[1:]) + + // for each type + for _, rtype := range rtypes { + ref := rtype.Ref + schema := RegistryData.Schema[ref] + + // special case, if the filter was '*' return all indices + if len(filter) == 0 { + + tmp := make([]*RegKeyIndex, 0, len(schema.KeyIndex)) + for _, keyix := range schema.KeyIndex { + tmp = append(tmp, keyix) + } + ix = append(ix, tmp...) + + } else { + // otherwise substring match the key names + + for kname, keyix := range schema.KeyIndex { + kname = strings.ToLower(kname) + if strings.Contains(kname, filter) { + ix = append(ix, keyix) + } + } + + } + } + + } else { + // perform an exact match, one key for each type + + for _, rtype := range rtypes { + ref := rtype.Ref + schema := RegistryData.Schema[ref] + keyix := schema.KeyIndex[filter] + if keyix != nil { + // add the index + ix = append(ix, keyix) + } + } + + } + + return ix +} + +// helper func to determine if an attribute matches a filter +func matchAttribute(attribute *RegAttribute, + filter string, isExact bool) bool { + + if isExact { + + return filter == attribute.RawValue + + } else { + + l := strings.ToLower(attribute.RawValue) + return strings.Contains(l, filter) + + } +} + +// return a map of objects and attribute values that match the filter +func filterAttributes(ix []*RegKeyIndex, objects []*RegObject, + filter string, raw bool) map[string]map[string][]string { + + result := make(map[string]map[string][]string) + + // pre-calculate the search type + isExact := true + isAll := false + + if filter[0] == '*' { + isExact = false + filter = strings.ToLower(filter[1:]) + if len(filter) == 0 { + isAll = true + } + } + + // for each key index + for _, keyix := range ix { + + // for each object + for _, object := range objects { + + // attributes in this object that match this key + attributes := keyix.Objects[object] + if attributes != nil { + // this object has at least one relevant key + + // match the attributes + for _, attribute := range attributes { + if isAll || matchAttribute(attribute, filter, isExact) { + // match found ! + + objmap := result[object.Ref] + if objmap == nil { + objmap = make(map[string][]string) + result[object.Ref] = objmap + } + + // append the result + var value *string + if raw { + value = &attribute.RawValue + } else { + value = &attribute.Value + } + + objmap[keyix.Ref] = append(objmap[keyix.Ref], *value) + } + } + } + + } + + } + + return result +} + +////////////////////////////////////////////////////////////////////////// +// root handler, lists all types within the registry + +func regRootHandler(w http.ResponseWriter, r *http.Request) { + + response := make(map[string]int) + for _, rType := range RegistryData.Types { + response[rType.Ref] = len(rType.Objects) + } + ResponseJSON(w, response) + +} + +////////////////////////////////////////////////////////////////////////// +// type handler returns list of objects that match the type + +func regTypeHandler(w http.ResponseWriter, r *http.Request) { + + // request parameters + vars := mux.Vars(r) + tFilter := vars["type"] // type filter + + // match registry types against the filter + rtypes := filterTypes(tFilter) + if rtypes == nil { + http.Error(w, "No objects matching '"+tFilter+"' found", + http.StatusNotFound) + return + } + + // construct the response + response := make(map[string][]string) + for _, rtype := range rtypes { + + objects := make([]string, 0, len(rtype.Objects)) + for key := range rtype.Objects { + objects = append(objects, key) + } + + response[rtype.Ref] = objects + } + + ResponseJSON(w, response) +} + +////////////////////////////////////////////////////////////////////////// +// object handler returns object data + +// per object response structure +type RegObjectResponse struct { + Attributes [][2]string + Backlinks []string +} + +func regObjectHandler(w http.ResponseWriter, r *http.Request) { + + // request parameters + vars := mux.Vars(r) + query := r.URL.Query() + + tFilter := vars["type"] // type filter + oFilter := vars["object"] // object filter + raw := query["raw"] // raw or decorated results + + // select the type(s) + rtypes := filterTypes(tFilter) + if rtypes == nil { + http.Error(w, "No objects matching '"+tFilter+"' found", + http.StatusNotFound) + return + } + + // then select the objects + objects := filterObjects(rtypes, oFilter) + if objects == nil { + http.Error(w, "No objects matching '"+tFilter+ + "/"+oFilter+"' found", http.StatusNotFound) + return + } + + // collate the results in to the response data + if raw == nil { + // provide a decorated response + response := make(map[string]RegObjectResponse) + + // for each object in the results + for _, object := range objects { + + // copy the raw attributes + attributes := make([][2]string, len(object.Data)) + for ix, attribute := range object.Data { + attributes[ix] = [2]string{attribute.Key, attribute.Value} + } + + // construct the backlinks + backlinks := make([]string, len(object.Backlinks)) + for ix, object := range object.Backlinks { + backlinks[ix] = object.Ref + } + + // add to the response + response[object.Ref] = RegObjectResponse{ + Attributes: attributes, + Backlinks: backlinks, + } + } + + ResponseJSON(w, response) + + } else { + // provide a response with just the raw registry data + response := make(map[string][][2]string) + + // for each object in the results + for _, object := range objects { + + attributes := make([][2]string, len(object.Data)) + response[object.Ref] = attributes + + // copy the raw attributes + for ix, attribute := range object.Data { + attributes[ix] = [2]string{attribute.Key, attribute.RawValue} + } + } + + ResponseJSON(w, response) + } + +} + +////////////////////////////////////////////////////////////////////////// +// key handler returns attribute data matching the key + +func regKeyHandler(w http.ResponseWriter, r *http.Request) { + + // request parameters + vars := mux.Vars(r) + query := r.URL.Query() + + tFilter := vars["type"] // type filter + oFilter := vars["object"] // object filter + kFilter := vars["key"] // key filter + raw := query["raw"] // raw or decorated results + + // select the type(s) + rtypes := filterTypes(tFilter) + if rtypes == nil { + http.Error(w, "No objects matching '"+tFilter+"' found", + http.StatusNotFound) + return + } + + // select the key indices + ix := filterKeys(rtypes, kFilter) + if rtypes == nil { + http.Error(w, "No objects matching '"+tFilter+"/*/"+ + kFilter+"' found", http.StatusNotFound) + return + } + + // select the objects + objects := filterObjects(rtypes, oFilter) + if objects == nil { + http.Error(w, "No objects matching '"+tFilter+ + "/"+oFilter+"' found", http.StatusNotFound) + return + } + + // select objects that match the keys + amap := filterAttributes(ix, objects, "*", (raw != nil)) + if len(amap) == 0 { + http.Error(w, "No attributes matching '"+tFilter+"/"+ + oFilter+"/"+kFilter+"' found", http.StatusNotFound) + return + } + + ResponseJSON(w, amap) +} + +////////////////////////////////////////////////////////////////////////// +// attribute handler returns attribute data matching the attribute + +func regAttributeHandler(w http.ResponseWriter, r *http.Request) { + + // request parameters + vars := mux.Vars(r) + query := r.URL.Query() + + tFilter := vars["type"] // type filter + oFilter := vars["object"] // object filter + kFilter := vars["key"] // key filter + aFilter := vars["attribute"] // attribute filter + raw := query["raw"] // raw or decorated results + + // select the type(s) + rtypes := filterTypes(tFilter) + if rtypes == nil { + http.Error(w, "No objects matching '"+tFilter+"' found", + http.StatusNotFound) + return + } + + // select the key indices + ix := filterKeys(rtypes, kFilter) + if rtypes == nil { + http.Error(w, "No objects matching '"+tFilter+"/*/"+ + kFilter+"' found", http.StatusNotFound) + return + } + + // then select the objects + objects := filterObjects(rtypes, oFilter) + if objects == nil { + http.Error(w, "No objects matching '"+tFilter+ + "/"+oFilter+"' found", http.StatusNotFound) + return + } + + // select objects that match the keys + amap := filterAttributes(ix, objects, aFilter, (raw != nil)) + if len(amap) == 0 { + http.Error(w, "No attributes matching '"+tFilter+"/"+ + oFilter+"/"+kFilter+"/"+aFilter+"' found", http.StatusNotFound) + return + } + + ResponseJSON(w, amap) + +} + +////////////////////////////////////////////////////////////////////////// +// end of code diff --git a/registry.go b/registry.go new file mode 100644 index 0000000..7c56622 --- /dev/null +++ b/registry.go @@ -0,0 +1,717 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry API Server +////////////////////////////////////////////////////////////////////////// + +package main + +////////////////////////////////////////////////////////////////////////// + +import ( + "bufio" + // "errors" + "fmt" + log "github.com/sirupsen/logrus" + "io/ioutil" + "os" + "os/exec" + "strings" + "time" +) + +////////////////////////////////////////////////////////////////////////// +// registry data model + +// registry data + +// Attributes within Objects +type RegAttribute struct { + Key string + Value string // this is a post-processed, or decorated value + RawValue string // the raw value as read from the registry +} + +type RegObject struct { + Ref string // the ref contains the full path for this object + Data []*RegAttribute // the key/value data for this object + Backlinks []*RegObject // other objects that reference this one +} + +// types are collections of objects +type RegType struct { + Ref string // full path for this type + Objects map[string]*RegObject // the objects in this type +} + +// registry meta data + +type RegAttributeSchema struct { + Fields []string + Relations []*RegType +} + +type RegKeyIndex struct { + Ref string + Objects map[*RegObject][]*RegAttribute +} + +type RegTypeSchema struct { + Ref string + Attributes map[string]*RegAttributeSchema + KeyIndex map[string]*RegKeyIndex +} + +// the registry itself + +type Registry struct { + Commit string + Schema map[string]*RegTypeSchema + Types map[string]*RegType +} + +// and a variable for the actual data +var RegistryData *Registry + +// store the current commit has +var previousCommit string + +////////////////////////////////////////////////////////////////////////// +// utility and manipulation functions + +// general functions + +func RegistryMakePath(t string, o string) string { + return t + "/" + o +} + +func RegistrySplitPath(p string) (string, string) { + tmp := strings.Split(p, "/") + if len(tmp) != 2 { + return "", "" + } + return tmp[0], tmp[1] +} + +func (registry *Registry) GetObject(path string) *RegObject { + rtname, objname := RegistrySplitPath(path) + rtype := registry.Types[rtname] + if rtype == nil { + return nil + } + return rtype.Objects[objname] +} + +// attribute functions + +// nothing here + +// object functions + +// return attributes exactly matching a specific key +func (object *RegObject) GetKey(key string) []*RegAttribute { + + attributes := make([]*RegAttribute, 0) + for _, a := range object.Data { + if a.Key == key { + attributes = append(attributes, a) + } + } + + return attributes +} + +// return a single key +func (object *RegObject) GetSingleKey(key string) *RegAttribute { + + attributes := object.GetKey(key) + if len(attributes) != 1 { + log.WithFields(log.Fields{ + "key": key, + "object": object.Ref, + }).Error("Unable to find unique key in object") + + // can't register the object + return nil + } + return attributes[0] +} + +// schema functions + +// validate a set of attributes against a schema +func (schema *RegTypeSchema) validate(attributes []*RegAttribute) []*RegAttribute { + + validated := make([]*RegAttribute, 0, len(attributes)) + for _, attribute := range attributes { + + // keys beginning with 'x-' are user defined, skip validation + if !strings.HasPrefix(attribute.Key, "x-") { + if schema.Attributes[attribute.Key] == nil { + // couldn't find a schema attribute + + log.WithFields(log.Fields{ + "key": attribute.Key, + "schema": schema.Ref, + }).Error("Schema validation failed") + + // don't add to the validated list + continue + } + } + + // all ok + validated = append(validated, attribute) + } + + return validated +} + +// add an attribute to the key map +func (schema *RegTypeSchema) addKeyIndex(object *RegObject, + attribute *RegAttribute) { + + keyix := schema.KeyIndex[attribute.Key] + // create a new object map if it didn't exist + if keyix == nil { + keyix = &RegKeyIndex{ + Ref: attribute.Key, + Objects: make(map[*RegObject][]*RegAttribute), + } + schema.KeyIndex[attribute.Key] = keyix + } + + // add the object/attribute reference + keyix.Objects[object] = append(keyix.Objects[object], attribute) +} + +// object functions + +// add a backlink to an object +func (object *RegObject) addBacklink(ref *RegObject) { + + // check if the backlink already exists, this could be the case + // if an object is referenced multiple times (e.g. admin-c & tech-c) + for _, blink := range object.Backlinks { + if blink == ref { + // already exists, just return as nothing to do + return + } + } + + // didn't find a match, add the backlink + object.Backlinks = append(object.Backlinks, ref) + +} + +////////////////////////////////////////////////////////////////////////// +// reload the registry + +func reloadRegistry(path string, commit string) { + + log.Debug("Reloading registry") + + // r will become the new registry data + registry := &Registry{ + Commit: commit, + Schema: make(map[string]*RegTypeSchema), + Types: make(map[string]*RegType), + } + + // bootstrap the schema registry type + registry.Types["schema"] = &RegType{ + Ref: "schema", + Objects: make(map[string]*RegObject), + } + registry.loadType("schema", path) + + // and parse the schema to get the remaining types + registry.parseSchema() + + // now load the remaining types + for _, rType := range registry.Types { + registry.loadType(rType.Ref, path) + } + + // mark relationships + registry.decorate() + + // trigger updates in any other modules + EventBus.Fire("RegistryUpdate", registry, path) + + // swap in the new registry data + RegistryData = registry +} + +////////////////////////////////////////////////////////////////////////// +// create and load the raw data for a registry type + +func (registry *Registry) loadType(typeName string, path string) { + + // the type will already have been created + rType := registry.Types[typeName] + + // as will the schema (unless attempting to load the schema itself) + schema := registry.Schema[typeName] + + // special case for DNS as the directory + // doesn't match the type name + if typeName == "domain" { + path += "/dns" + } else { + path += "/" + typeName + } + + // and load all the objects in this type + rType.loadObjects(schema, path) + +} + +////////////////////////////////////////////////////////////////////////// +// load all the objects associated with a type + +func (rType *RegType) loadObjects(schema *RegTypeSchema, path string) { + + entries, err := ioutil.ReadDir(path) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "path": path, + "type": rType.Ref, + }).Error("Failed to read registry type directory") + return + } + + // for each entry in the directory + for _, entry := range entries { + + // each file maps to a registry object + if !entry.IsDir() { + + filename := entry.Name() + // ignore dotfiles + if !strings.HasPrefix(filename, ".") { + + // load the attributes from file + attributes := loadAttributes(path + "/" + filename) + + // basic validation of attributes against the schema + // schema may be nil if we are actually loading the schema itself + if schema != nil { + attributes = schema.validate(attributes) + } + + // make the object + object := &RegObject{ + Ref: RegistryMakePath(rType.Ref, filename), + Data: attributes, + Backlinks: make([]*RegObject, 0), + } + + // add to type + rType.Objects[filename] = object + } + } + } + + log.WithFields(log.Fields{ + "ref": rType.Ref, + "path": path, + "count": len(rType.Objects), + }).Debug("Loaded registry type") + +} + +////////////////////////////////////////////////////////////////////////// +// read attributes from a file + +func loadAttributes(path string) []*RegAttribute { + + attributes := make([]*RegAttribute, 0) + + // open the file to start reading it + file, err := os.Open(path) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "path": path, + }).Error("Failed to read attributes from file") + return attributes + } + defer file.Close() + + // read the file line by line using the bufio scanner + scanner := bufio.NewScanner(file) + for scanner.Scan() { + + line := strings.TrimRight(scanner.Text(), "\r\n") + + // lines starting with '+' denote an empty line + if line[0] == '+' { + + // concatenate a \n on to the previous attribute value + attributes[len(attributes)-1].RawValue += "\n" + + } else { + + // look for a : separator in the first 20 characters + ix := strings.IndexByte(line, ':') + if ix == -1 || ix >= 20 { + // couldn't find one + + if len(line) <= 20 { + // hmmm, the line was shorter than 20 characters + // something is amiss + + log.WithFields(log.Fields{ + "length": len(line), + "path": path, + "line": line, + }).Warn("Short line detected") + + } else { + + // line is a continuation of the previous line, so + // concatenate the value on to the previous attribute value + attributes[len(attributes)-1].RawValue += + "\n" + string(line[20:]) + + } + } else { + // found a key and : separator + + // is there actually a value ? + var value string + if len(line) <= 20 { + // blank value + value = "" + } else { + value = string(line[20:]) + } + + // create a new attribute + a := &RegAttribute{ + Key: string(line[:ix]), + RawValue: value, + } + attributes = append(attributes, a) + } + } + } + + return attributes +} + +////////////////////////////////////////////////////////////////////////// +// parse schema files to extract keys and for attribute relations + +func (registry *Registry) parseSchema() { + + // for each object in the schema type + for _, object := range registry.Types["schema"].Objects { + + // look up the ref attribute + ref := object.GetSingleKey("ref") + if ref == nil { + log.WithFields(log.Fields{ + "object": object.Ref, + }).Error("Schema record without ref") + + // can't process this object + continue + } + + // create the type schema object + typeName := strings.TrimPrefix(ref.RawValue, "dn42.") + typeSchema := &RegTypeSchema{ + Ref: typeName, + Attributes: make(map[string]*RegAttributeSchema), + KeyIndex: make(map[string]*RegKeyIndex), + } + + // ensure the type exists + rType := registry.Types[typeName] + if rType == nil { + rType := &RegType{ + Ref: typeName, + Objects: make(map[string]*RegObject), + } + registry.Types[typeName] = rType + } + + // for each key attribute in the schema + attributes := object.GetKey("key") + for _, attribute := range attributes { + + // split the value on whitespace + fields := strings.Fields(attribute.RawValue) + keyName := fields[0] + + typeSchema.Attributes[keyName] = &RegAttributeSchema{ + Fields: fields[1:], + } + } + + // register the type schema + registry.Schema[typeName] = typeSchema + + } + + // scan the fields of each schema attribute to determine relationships + // this needs to be second step to allow pre-creation of the types + for _, typeSchema := range registry.Schema { + for attribName, attribSchema := range typeSchema.Attributes { + for _, field := range attribSchema.Fields { + if strings.HasPrefix(field, "lookup=") { + + // the relationships may be a multivalue, separated by , + rels := strings.Split(strings. + TrimPrefix(field, "lookup="), ",") + + // map to a regtype + relations := make([]*RegType, 0, len(rels)) + for ix := range rels { + relName := strings.TrimPrefix(rels[ix], "dn42.") + relation := registry.Types[relName] + + // log if unable to look up the type + if relation == nil { + // log unless this is the schema def lookup=str '>' [spec]... + if typeSchema.Ref != "schema" { + log.WithFields(log.Fields{ + "relation": relName, + "attribute": attribName, + "type": typeSchema.Ref, + }).Error("Relation to type that does not exist") + } + + } else { + // store the relationship + relations = append(relations, relation) + } + } + + // register the relations + attribSchema.Relations = relations + + // assume only 1 lookup= per key + break + } + } + } + } + + log.Debug("Schema parsing complete") +} + +////////////////////////////////////////////////////////////////////////// +// parse all attributes and decorate them + +func (registry *Registry) decorate() { + + cattribs := 0 + cmatched := 0 + + // walk each attribute value + for _, rType := range registry.Types { + schema := registry.Schema[rType.Ref] + for _, object := range rType.Objects { + for _, attribute := range object.Data { + cattribs += 1 + + // add this attribute to the key map + schema.addKeyIndex(object, attribute) + + attribSchema := schema.Attributes[attribute.Key] + // are there relations defined for this attribute ? + // attribSchema may be null if this attribute is user defined (x-*) + if (attribSchema != nil) && + attribute.matchRelation(object, attribSchema.Relations) { + // matched + cmatched += 1 + } else { + // no match, just copy the attribute data + attribute.Value = attribute.RawValue + } + } + } + } + + log.WithFields(log.Fields{ + "attributes": cattribs, + "matched": cmatched, + }).Debug("Decoration complete") + +} + +////////////////////////////////////////////////////////////////////////// +// match an attribute against schema relations + +func (attribute *RegAttribute) matchRelation(parent *RegObject, + relations []*RegType) bool { + + // it's not going to match if relations is empty + if relations == nil { + return false + } + + // check each relation + for _, relation := range relations { + + object := relation.Objects[attribute.RawValue] + if object != nil { + // found a match ! + + // decorate the attribute value + attribute.Value = fmt.Sprintf("[%s](%s)", + attribute.RawValue, object.Ref) + + // and add a back reference to the related object + object.addBacklink(parent) + + return true + } + + } + + // didn't find anything + return false +} + +////////////////////////////////////////////////////////////////////////// +// fetch the current commit hash + +func getCommitHash(regDir string, gitPath string) string { + + // run git to get the latest commit hash + cmd := exec.Command(gitPath, "log", "-1", "--format=%H") + cmd.Dir = regDir + // execute + out, err := cmd.Output() + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "gitPath": gitPath, + "regDir": regDir, + }).Error("Failed to execute git log") + } + + return strings.TrimSpace(string(out)) +} + +////////////////////////////////////////////////////////////////////////// +// refresh the registry + +func refreshRegistry(regDir string, gitPath string, branch string) { + + // run git fetch to get the current commits from the master + cmd := exec.Command(gitPath, "fetch") + cmd.Dir = regDir + // execute + if out, err := cmd.Output(); err != nil { + log.WithFields(log.Fields{ + "error": err, + "gitPath": gitPath, + "regDir": regDir, + }).Error("Failed to execute git fetch") + } else { + fmt.Printf("Git Fetch: %s", string(out)) + } + + // then reset hard to match the master + cmd = exec.Command(gitPath, "reset", "--hard", "origin/"+branch) + cmd.Dir = regDir + // execute + if out, err := cmd.Output(); err != nil { + log.WithFields(log.Fields{ + "error": err, + "gitPath": gitPath, + "regDir": regDir, + "branch": branch, + }).Error("Failed to execute git reset") + } else { + fmt.Printf("Git Reset: %s", string(out)) + } +} + +////////////////////////////////////////////////////////////////////////// +// called from main to initialse the registry data and syncing + +func InitialiseRegistryData(regDir string, refresh time.Duration, + gitPath string, autoPull bool, branch string) { + + // validate that the regDir/data path exists + dataPath := regDir + "/data" + regStat, err := os.Stat(dataPath) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "path": dataPath, + }).Fatal("Unable to find registry directory") + } + + // and it is a directory + if !regStat.IsDir() { + log.WithFields(log.Fields{ + "error": err, + "path": dataPath, + }).Fatal("Registry path is not a directory") + } + + // check that git exists + _, err = os.Stat(gitPath) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "path": gitPath, + }).Fatal("Unable to find git executable") + } + + // enforce a minimum update time + minTime := 10 * time.Minute + if refresh < minTime { + log.WithFields(log.Fields{ + "interval": refresh, + }).Error("Enforcing minimum update time of 10 minutes") + + refresh = minTime + } + + // initialise the previous commit hash + // and do initial load from registry + previousCommit = getCommitHash(regDir, gitPath) + reloadRegistry(dataPath, previousCommit) + + go func() { + + // every refresh interval + for range time.Tick(refresh) { + log.Debug("Refresh Timer") + + // automatically try to refresh the registry ? + if autoPull { + refreshRegistry(regDir, gitPath, branch) + } + + // get the latest hash + currentCommit := getCommitHash(regDir, gitPath) + + // has the registry been updated ? + if currentCommit != previousCommit { + log.WithFields(log.Fields{ + "current": currentCommit, + "previous": previousCommit, + }).Info("Registry has changed, refresh started") + + // refresh + reloadRegistry(dataPath, currentCommit) + + // update commit + previousCommit = currentCommit + } + + } + }() + +} + +////////////////////////////////////////////////////////////////////////// +// end of code diff --git a/roaapi.go b/roaapi.go new file mode 100644 index 0000000..6d92e67 --- /dev/null +++ b/roaapi.go @@ -0,0 +1,471 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry API Server +////////////////////////////////////////////////////////////////////////// + +package main + +////////////////////////////////////////////////////////////////////////// + +import ( + "fmt" + "github.com/gorilla/mux" + log "github.com/sirupsen/logrus" + // "math/big" + "bufio" + "net" + "net/http" + "os" + "sort" + "strconv" + "strings" + "time" +) + +////////////////////////////////////////////////////////////////////////// +// register the api + +func init() { + EventBus.Listen("APIEndpoint", InitROAAPI) + EventBus.Listen("RegistryUpdate", ROAUpdate) +} + +////////////////////////////////////////////////////////////////////////// +// data model + +type PrefixROA struct { + Prefix string `json:"prefix"` + MaxLen uint8 `json:"maxLength"` + ASN string `json:"asn"` +} + +type ROAFilter struct { + Number uint `json:"nr"` + Action string `json:"action"` + Prefix string `json:"prefix"` + MinLen uint8 `json:"minlen"` + MaxLen uint8 `json:"maxlen"` + Network *net.IPNet `json:"-"` + IPType uint8 `json:"-"` +} + +type ROA struct { + CTime time.Time + Commit string + Filters []*ROAFilter + IPv4 []*PrefixROA + IPv6 []*PrefixROA +} + +var ROAData *ROA + +// set validity period for one week +// this might appear to be a long time, but is intended to provide +// enough time to prevent expiry of the data between real registry +// updates (which may only happen infrequently) +const ROA_JSON_VALIDITY_PERIOD = (7 * 24) + +type ROAMetaData struct { + Counts uint `json:"counts"` + Generated uint32 `json:"generated"` + Valid uint32 `json:"valid"` + Signature string `json:"signature,omitempty"` + SignatureDate string `json:"signatureDate,omitempty"` +} + +type ROAJSON struct { + MetaData ROAMetaData `json:"metadata"` + Roas []*PrefixROA `json:"roas"` +} + +var ROAJSONResponse *ROAJSON + +////////////////////////////////////////////////////////////////////////// +// called from main to initialise the API routing + +func InitROAAPI(params ...interface{}) { + + router := params[0].(*mux.Router) + + s := router. + Methods("GET"). + PathPrefix("/roa"). + Subrouter() + + s.HandleFunc("/filter/{ipv}", roaFilterHandler) + s.HandleFunc("/json", roaJSONHandler) + s.HandleFunc("/bird/{birdv}/{ipv}", roaBirdHandler) + + log.Info("ROA API installed") +} + +////////////////////////////////////////////////////////////////////////// +// api handlers + +// return JSON formatted version of filter{,6}.txt +func roaFilterHandler(w http.ResponseWriter, r *http.Request) { + + vars := mux.Vars(r) + ipv := vars["ipv"] + + // pre-create an array to hold the result + filters := make([]*ROAFilter, 0, len(ROAData.Filters)) + + // helper closure to select from the filter array + fselect := func(a []*ROAFilter, t uint8) []*ROAFilter { + for _, f := range ROAData.Filters { + if f.IPType == t { + a = append(a, f) + } + } + return a + } + + // add ipv4 filters if required + if strings.ContainsRune(ipv, '4') { + filters = fselect(filters, 4) + } + + // add ipv6 filters if required + if strings.ContainsRune(ipv, '6') { + filters = fselect(filters, 6) + } + + ResponseJSON(w, filters) +} + +// return JSON formatted ROA data suitable for use with GoRTR +func roaJSONHandler(w http.ResponseWriter, r *http.Request) { + + // check validity period of returned data + tnow := uint32(time.Now().Unix()) + valid := ROAJSONResponse.MetaData.Valid + + // check if validity period is close to expiry + if (tnow > valid) || + ((valid - tnow) < (ROA_JSON_VALIDITY_PERIOD / 4)) { + // if so extend the validity period + ROAJSONResponse.MetaData.Valid += (ROA_JSON_VALIDITY_PERIOD * 3600) + } + + ResponseJSON(w, ROAJSONResponse) +} + +// return the roa in bird format +func roaBirdHandler(w http.ResponseWriter, r *http.Request) { + + vars := mux.Vars(r) + birdv := vars["birdv"] + ipv := vars["ipv"] + + // bird 1 or bird 2 format + birdf := "roa %s max %d as %s;\n" + if birdv == "2" { + birdf = "route %s max %d as %s;\n" + } + + var roa []*PrefixROA + if strings.ContainsRune(ipv, '4') { + roa = append(roa, ROAData.IPv4...) + } + if strings.ContainsRune(ipv, '6') { + roa = append(roa, ROAData.IPv6...) + } + + w.Header().Set("Content-Type", "text/plain") + w.Header().Set("Access-Control-Allow-Origin", "*") + + fmt.Fprintf(w, "#\n# dn42regsrv ROA Generator\n# Last Updated: %s\n"+ + "# Commit: %s\n#\n", ROAData.CTime.String(), ROAData.Commit) + + for _, r := range roa { + fmt.Fprintf(w, birdf, r.Prefix, r.MaxLen, r.ASN[2:]) + } + +} + +////////////////////////////////////////////////////////////////////////// +// called whenever the registry is updated + +func ROAUpdate(params ...interface{}) { + + registry := params[0].(*Registry) + path := params[1].(string) + + // initiate new ROA data + roa := &ROA{ + CTime: time.Now(), + Commit: registry.Commit, + } + + // load filter{,6}.txt files + if roa.loadFilter(path+"/filter.txt", 4) != nil { + // error loading IPv4 filter, don't update + return + } + + if roa.loadFilter(path+"/filter6.txt", 6) != nil { + // error loading IPv6 filter, don't update + return + } + + // compile ROA prefixes + roa.IPv4 = roa.CompileROA(registry, "route") + roa.IPv6 = roa.CompileROA(registry, "route6") + + // swap in the new data + ROAData = roa + + log.WithFields(log.Fields{ + "ipv4": len(roa.IPv4), + "ipv6": len(roa.IPv6), + }).Debug("ROA data updated") + + // pre-compute the JSON return struct + + utime := uint32(roa.CTime.Unix()) + + response := &ROAJSON{ + MetaData: ROAMetaData{ + Generated: utime, + Valid: utime + (ROA_JSON_VALIDITY_PERIOD * 3600), + }, + } + + response.Roas = append(roa.IPv4, roa.IPv6...) + response.MetaData.Counts = uint(len(response.Roas)) + + ROAJSONResponse = response +} + +////////////////////////////////////////////////////////////////////////// +// load network filter definitions from a filter file + +func (roa *ROA) loadFilter(path string, iptype uint8) error { + + // open the file for reading + file, err := os.Open(path) + if err != nil { + log.WithFields(log.Fields{ + "path": path, + "error": err, + }).Error("Unable to open filter file") + return err + } + defer file.Close() + + // helper closure to convert strings to numbers + var cerr error + convert := func(s string) int { + if cerr != nil { + return 0 + } + val, cerr := strconv.Atoi(s) + if cerr != nil { + log.WithFields(log.Fields{ + "number": s, + "error": err, + }).Error("Unable to parse number in filter file") + return 0 + } + return val + } + + filters := make([]*ROAFilter, 0) + + // read the file line by line + scanner := bufio.NewScanner(file) + for scanner.Scan() { + + line := strings.TrimSpace(scanner.Text()) + + // remove any comments + if ix := strings.IndexRune(line, '#'); ix != -1 { + line = line[:ix] + } + + fields := strings.Fields(line) + if len(fields) >= 5 { + + // parse the prefix in to a NetIP structure + prefix := fields[2] + _, network, err := net.ParseCIDR(prefix) + if err != nil { + log.WithFields(log.Fields{ + "path": path, + "prefix": prefix, + "error": err, + }).Error("Unable to parse CIDR in filter file") + + } else { + + // construct the filter object + roaf := &ROAFilter{ + Number: uint(convert(fields[0])), + Action: fields[1], + Prefix: prefix, + MinLen: uint8(convert(fields[3])), + MaxLen: uint8(convert(fields[4])), + Network: network, + IPType: iptype, + } + + // add to list if no strconv error + if cerr == nil { + filters = append(filters, roaf) + } + } + } + } + + // did something go wrong ? + if err := scanner.Err(); err != nil { + log.WithFields(log.Fields{ + "path": path, + "error": err, + }).Error("Scanner error reading filter file") + return err + } + + // filter.txt should be in order, + // but still sort by number just in case + sort.Slice(filters, func(i, j int) bool { + return filters[i].Number < filters[j].Number + }) + + // add to the roa object + roa.Filters = append(roa.Filters, filters...) + return nil +} + +////////////////////////////////////////////////////////////////////////// +// return the filter object that matches an IP address + +func (roa *ROA) MatchFilter(ip net.IP) *ROAFilter { + for _, filter := range roa.Filters { + if filter.Network.Contains(ip) { + return filter + } + } + + log.WithFields(log.Fields{ + "IP": ip, + }).Error("Couldn't match address to filter !") + + return nil +} + +////////////////////////////////////////////////////////////////////////// +// compile ROA data + +func (roa *ROA) CompileROA(registry *Registry, + tname string) []*PrefixROA { + + // prepare indices to the route object keys + stype := registry.Schema[tname] + routeIX := stype.KeyIndex[tname] + originIX := stype.KeyIndex["origin"] + mlenIX := stype.KeyIndex["max-length"] + + roalist := make([]*PrefixROA, 0, len(routeIX.Objects)) + + // for each object that has a route key + for object, rattribs := range routeIX.Objects { + + if len(rattribs) > 1 { + log.WithFields(log.Fields{ + "object": object.Ref, + }).Warn("Found object with multiple route attributes") + } + + // extract the prefix + prefix := rattribs[0].RawValue + _, pnet, err := net.ParseCIDR(prefix) + if err != nil { + log.WithFields(log.Fields{ + "object": object.Ref, + "prefix": prefix, + "error": err, + }).Error("Unable to parse CIDR in ROA") + continue + } + + // match the prefix to the prefix filters + filter := roa.MatchFilter(pnet.IP) + if filter == nil { + continue + } + + // don't allow routes that are denied in the filter rules + if filter.Action == "deny" { + log.WithFields(log.Fields{ + "object": object.Ref, + "prefix": prefix, + "filter": filter.Prefix, + }).Warn("Denied ROA through filter rule") + continue + } + + mlen := filter.MaxLen + + // if the prefix is greater than the filter.MaxLen + // then don't emit an ROA route (making the route invalid) + if ones, _ := pnet.Mask.Size(); ones > int(mlen) { + log.WithFields(log.Fields{ + "object": object.Ref, + "prefix": prefix, + "filter": filter.Prefix, + }).Debug("Defined ROA: Prefix > filter MaxLen") + continue + } + + // calculate the max-length for this object + + // check if the attribute has max-length defined + mattrib := mlenIX.Objects[object] + if mattrib != nil { + + // use the local max-length value + tmp, err := strconv.ParseUint(mattrib[0].RawValue, 10, 8) + if err != nil { + log.WithFields(log.Fields{ + "object": object.Ref, + "max-length": mattrib[0].RawValue, + "error": err, + }).Warn("Unable to convert max-length attribute") + } else { + + // filter rules still have precedence over local values + if (uint8(tmp) < mlen) && (uint8(tmp) > filter.MinLen) { + mlen = uint8(tmp) + } + + } + } + + // look up the origin key for this object + oattribs := originIX.Objects[object] + if oattribs == nil { + log.WithFields(log.Fields{ + "object": object.Ref, + }).Warn("Route Object without Origin") + } else { + + // then for origin that can announce this prefix + for _, oattrib := range oattribs { + + // add the ROA + roalist = append(roalist, &PrefixROA{ + Prefix: prefix, + MaxLen: mlen, + ASN: oattrib.RawValue, + }) + + } + } + } + + return roalist +} + +////////////////////////////////////////////////////////////////////////// +// end of code diff --git a/static.go b/static.go new file mode 100644 index 0000000..f535c76 --- /dev/null +++ b/static.go @@ -0,0 +1,55 @@ +////////////////////////////////////////////////////////////////////////// +// DN42 Registry API Server +////////////////////////////////////////////////////////////////////////// + +package main + +////////////////////////////////////////////////////////////////////////// + +import ( + "github.com/gorilla/mux" + log "github.com/sirupsen/logrus" + "net/http" + "os" +) + +////////////////////////////////////////////////////////////////////////// +// called from main to initialise the API routing + +func InstallStaticRoutes(router *mux.Router, staticPath string) { + + // an empty path disables static route serving + if staticPath == "" { + log.Info("Disabling static route serving") + return + } + + // validate that the staticPath exists + stat, err := os.Stat(staticPath) + if err != nil { + log.WithFields(log.Fields{ + "error": err, + "path": staticPath, + }).Fatal("Unable to find static page directory") + } + + // and it is a directory + if !stat.IsDir() { + log.WithFields(log.Fields{ + "error": err, + "path": staticPath, + }).Fatal("Static path is not a directory") + } + + // install a file server for the static route + router.PathPrefix("/").Handler(http.StripPrefix("/", + http.FileServer(http.Dir(staticPath)))).Methods("GET") + + log.WithFields(log.Fields{ + "path": staticPath, + }).Info("Static route installed") + +} + +////////////////////////////////////////////////////////////////////////// +// end of code