server implemented

This commit is contained in:
Sviatoslav Tsariov Yurievich 2023-03-23 11:36:11 +03:00
parent 6916a766b7
commit b3f0790146
15 changed files with 2339 additions and 0 deletions

41
.gitignore vendored Normal file
View File

@ -0,0 +1,41 @@
.DS_Store
__pycache__/
*.py[cod]
*.bak
# Packages
*.egg
*.egg-info
markdownserver.dat
env/
dist/
build/
docs/_build
eggs
parts
var
sdist
develop-eggs
.installed.cfg
lib
lib64
# Installer logs
pip-log.txt
# Unit test / coverage reports
.coverage
.tox
nosetests.xml
# Translations
*.mo
# Mr Developer
.mr.developer.cfg
.project
.pydevproject
.idea
.projectile
.ropeproject

12
Dockerfile Normal file
View File

@ -0,0 +1,12 @@
# syntax=docker/dockerfile:1
FROM python:3.8-slim-buster
WORKDIR /app
COPY requirements.txt requirements.txt
RUN pip3 install -r requirements.txt
COPY . .
CMD [ "python3", "server/server.py"]

View File

@ -1,3 +1,22 @@
# Athena
The documentation storage for Dipal
## Documentation server
Athena has HTTP server for reading markdown documentation.
If needed:
```
pip install venv
```
To prepare enviroment:
```
python -m venv env
source env/bin/activate
pip install -r requirements.txt
```
To start the server:
```
python server/server.py
```

9
docker-compose.yaml Normal file
View File

@ -0,0 +1,9 @@
version: '3.3'
services:
athena:
container_name: athena
image: athena
build: .
ports:
- "8009:8009"

13
requirements.txt Normal file
View File

@ -0,0 +1,13 @@
bottle==0.12.23
gevent==22.10.2
gevent-websocket==0.10.1
greenlet==2.0.2
importlib-metadata==6.0.0
Markdown==2.6.11
markdown-server==0.1.4
py-gfm==0.1.4
Pygments==2.14.0
zipp==3.13.0
zope.event==4.6
zope.interface==5.5.2

20
server/config/env.py Normal file
View File

@ -0,0 +1,20 @@
import os
port=8009
working_dir='.'
forbidden_dir_names=['env', '.git', '__pycache__', 'server']
allowed_file_endings=['.md']
encoding='utf-8'
root_path=os.getcwd()
server_path=os.path.join(root_path, 'server')
resource_dir=os.path.join(server_path, 'resources')
html_dir=os.path.join(resource_dir, 'html')
css_dir=os.path.join(resource_dir, 'css')
css_md_name='github.css'
css_index_name='github-markdown.css'
css_md_path=os.path.join(css_dir, css_md_name)
css_index_path=os.path.join(css_dir, css_index_name)
markdown_type='gfm'
test_cases_dir='test-cases'
test_suites_dir='test-suites'
test_reports_dir='test-reports'

View File

@ -0,0 +1,20 @@
from bottle import route, request, static_file
import config.env as env
import view.html as view
import model.markdown as markdown
import model.file_provider as file_provider
import controller.service as service
@route('/')
def index():
files = file_provider.collect_nested_files(env.working_dir)
return view.index(files)
@route('/view/<resource:path>')
def show_doc(resource):
if '.png' in resource:
return static_file(resource, root=env.working_dir)
content = file_provider.read_file(resource)
doc = markdown.read_doc(content)
return view.to_html(doc)

View File

@ -0,0 +1,17 @@
import json
import os
from datetime import datetime
import config.env as env
class ReportType:
Suite = 'suites'
Case = 'cases'
def read_request(request):
return json.loads(request.body.readlines()[0])
def make_report_path(resource, _type: ReportType):
report_resource = resource.replace('test-', '')
report_dir = os.path.splitext(report_resource)[0]
time = datetime.now().strftime('%Y_%m_%d-%H_%M_%S')
return env.test_reports_dir + '\\' + report_dir + '\\' + time + '.md'

View File

@ -0,0 +1,36 @@
def has_allowed_extension(name, allowed_file_endings=None):
if not allowed_file_endings:
return True
return any(name.endswith(file_ending) for file_ending in allowed_file_endings)
def collect_nested_files(directory_path, forbidden_dir_names=None, allowed_file_endings=None, collected_paths=None):
if forbidden_dir_names is None:
forbidden_dir_names = ['env', '.git', '__pycache__', 'server']
if allowed_file_endings is None:
allowed_file_endings = ['.md']
if collected_paths is None:
collected_paths = (directory_path, [])
if any(name in directory_path.split(os.path.sep) for name in forbidden_dir_names):
return collected_paths
subpaths = collected_paths[1]
nested_list = sorted(os.listdir(directory_path))
for name in nested_list:
nested_path = os.path.join(directory_path, name)
if os.path.isdir(nested_path):
nested_dir = collect_nested_files(nested_path, collected_paths=(nested_path, []))
if nested_dir[1]:
subpaths.append(nested_dir)
else:
if not has_allowed_extension(name, allowed_file_endings=allowed_file_endings):
continue
subpaths.append(nested_path)
return collected_paths

View File

@ -0,0 +1,54 @@
import os
import codecs
import config.env as env
def has_allowed_extension(name, allowed_file_endings=None):
if not allowed_file_endings: return True
for file_ending in allowed_file_endings:
if name.endswith(file_ending):
return True
return False
def collect_nested_files(directory_path, forbidden_dir_names=None, allowed_file_endings=None, collected_paths=None):
if not forbidden_dir_names: forbidden_dir_names = ['env', '.git', '__pycache__', 'server']
if not allowed_file_endings: allowed_file_endings = ['.md']
if not collected_paths: collected_paths = []
print(directory_path)
if any(name in directory_path.split('\\') + directory_path.split('/') for name in forbidden_dir_names): return
collected_paths = ((directory_path, []))
subpaths = collected_paths[1]
nested_list = sorted(os.listdir(directory_path))
for name in nested_list:
nested_path = os.path.join(directory_path, name)
if os.path.isdir(nested_path):
nested_dir = collect_nested_files(nested_path, collected_paths=subpaths)
if nested_dir:
subpaths.append(nested_dir)
else:
if not has_allowed_extension(name, allowed_file_endings=allowed_file_endings): continue
subpaths.append(nested_path)
return collected_paths
def read_file(file_path):
with codecs.open(
os.path.join(env.working_dir, file_path),
encoding=env.encoding,
mode='r'
) as file:
return file.read()
def write_file(file_path, content, create_dirs=True):
if create_dirs:
os.makedirs(os.path.dirname(file_path), exist_ok=True)
with codecs.open(
os.path.join(env.working_dir, file_path),
encoding=env.encoding,
mode='w+'
) as file:
return file.write(content)

7
server/model/markdown.py Normal file
View File

@ -0,0 +1,7 @@
import re
import json
import config.env as env
import markdown as md
def read_doc(content):
return md.markdown(content, extensions=[env.markdown_type, 'markdown.extensions.tables'])

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,863 @@
body {
width: 45em;
border: 1px solid #ddd;
outline: 1300px solid #fff;
margin: 16px auto;
}
body .markdown-body
{
padding: 30px;
}
@font-face {
font-family: octicons-anchor;
src: url(data:font/woff;charset=utf-8;base64,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) format('woff');
}
.markdown-body {
-ms-text-size-adjust: 100%;
-webkit-text-size-adjust: 100%;
color: #333;
overflow: hidden;
font-family: "Helvetica Neue", Helvetica, "Segoe UI", Arial, freesans, sans-serif;
font-size: 16px;
line-height: 1.6;
word-wrap: break-word;
}
.markdown-body a {
background: transparent;
}
.markdown-body a:active,
.markdown-body a:hover {
outline: 0;
}
.markdown-body strong {
font-weight: bold;
}
.markdown-body h1 {
font-size: 2em;
margin: 0.67em 0;
}
.markdown-body img {
border: 0;
}
.markdown-body hr {
-moz-box-sizing: content-box;
box-sizing: content-box;
height: 0;
}
.markdown-body pre {
overflow: auto;
}
.markdown-body code,
.markdown-body kbd,
.markdown-body pre {
font-family: monospace, monospace;
font-size: 1em;
}
.markdown-body input {
color: inherit;
font: inherit;
margin: 0;
}
.markdown-body html input[disabled] {
cursor: default;
}
.markdown-body input {
line-height: normal;
}
.markdown-body input[type="checkbox"] {
-moz-box-sizing: border-box;
box-sizing: border-box;
padding: 0;
}
.markdown-body table {
border-collapse: collapse;
border-spacing: 0;
}
.markdown-body td,
.markdown-body th {
padding: 0;
}
.markdown-body * {
-moz-box-sizing: border-box;
box-sizing: border-box;
}
.markdown-body input {
font: 13px/1.4 Helvetica, arial, freesans, clean, sans-serif, "Segoe UI Emoji", "Segoe UI Symbol";
}
.markdown-body a {
color: #4183c4;
text-decoration: none;
}
.markdown-body a:hover,
.markdown-body a:focus,
.markdown-body a:active {
text-decoration: underline;
}
.markdown-body hr {
height: 0;
margin: 15px 0;
overflow: hidden;
background: transparent;
border: 0;
border-bottom: 1px solid #ddd;
}
.markdown-body hr:before {
display: table;
content: "";
}
.markdown-body hr:after {
display: table;
clear: both;
content: "";
}
.markdown-body h1,
.markdown-body h2,
.markdown-body h3,
.markdown-body h4,
.markdown-body h5,
.markdown-body h6 {
margin-top: 15px;
margin-bottom: 15px;
line-height: 1.1;
}
.markdown-body h1 {
font-size: 30px;
}
.markdown-body h2 {
font-size: 21px;
}
.markdown-body h3 {
font-size: 16px;
}
.markdown-body h4 {
font-size: 14px;
}
.markdown-body h5 {
font-size: 12px;
}
.markdown-body h6 {
font-size: 11px;
}
.markdown-body blockquote {
margin: 0;
}
.markdown-body ul,
.markdown-body ol {
padding: 0;
margin-top: 0;
margin-bottom: 0;
}
.markdown-body ol ol,
.markdown-body ul ol {
list-style-type: lower-roman;
}
.markdown-body ul ul ol,
.markdown-body ul ol ol,
.markdown-body ol ul ol,
.markdown-body ol ol ol {
list-style-type: lower-alpha;
}
.markdown-body dd {
margin-left: 0;
}
.markdown-body code {
font: 12px Consolas, "Liberation Mono", Menlo, Courier, monospace;
}
.markdown-body pre {
margin-top: 0;
margin-bottom: 0;
font: 12px Consolas, "Liberation Mono", Menlo, Courier, monospace;
}
.markdown-body .octicon {
font: normal normal 16px octicons-anchor;
line-height: 1;
display: inline-block;
text-decoration: none;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
.markdown-body .octicon-link:before {
content: '\f05c';
}
.markdown-body>*:first-child {
margin-top: 0 !important;
}
.markdown-body>*:last-child {
margin-bottom: 0 !important;
}
.markdown-body .anchor {
position: absolute;
top: 0;
bottom: 0;
left: 0;
display: block;
padding-right: 6px;
padding-left: 30px;
margin-left: -30px;
}
.markdown-body .anchor:focus {
outline: none;
}
.markdown-body h1,
.markdown-body h2,
.markdown-body h3,
.markdown-body h4,
.markdown-body h5,
.markdown-body h6 {
position: relative;
margin-top: 1em;
margin-bottom: 16px;
font-weight: bold;
line-height: 1.4;
}
.markdown-body h1 .octicon-link,
.markdown-body h2 .octicon-link,
.markdown-body h3 .octicon-link,
.markdown-body h4 .octicon-link,
.markdown-body h5 .octicon-link,
.markdown-body h6 .octicon-link {
display: none;
color: #000;
vertical-align: middle;
}
.markdown-body h1:hover .anchor,
.markdown-body h2:hover .anchor,
.markdown-body h3:hover .anchor,
.markdown-body h4:hover .anchor,
.markdown-body h5:hover .anchor,
.markdown-body h6:hover .anchor {
padding-left: 8px;
margin-left: -30px;
line-height: 1;
text-decoration: none;
}
.markdown-body h1:hover .anchor .octicon-link,
.markdown-body h2:hover .anchor .octicon-link,
.markdown-body h3:hover .anchor .octicon-link,
.markdown-body h4:hover .anchor .octicon-link,
.markdown-body h5:hover .anchor .octicon-link,
.markdown-body h6:hover .anchor .octicon-link {
display: inline-block;
}
.markdown-body h1 {
padding-bottom: 0.3em;
font-size: 2.25em;
line-height: 1.2;
border-bottom: 1px solid #eee;
}
.markdown-body h2 {
padding-bottom: 0.3em;
font-size: 1.75em;
line-height: 1.225;
border-bottom: 1px solid #eee;
}
.markdown-body h3 {
font-size: 1.5em;
line-height: 1.43;
}
.markdown-body h4 {
font-size: 1.25em;
}
.markdown-body h5 {
font-size: 1em;
}
.markdown-body h6 {
font-size: 1em;
color: #777;
}
.markdown-body p,
.markdown-body blockquote,
.markdown-body ul,
.markdown-body ol,
.markdown-body dl,
.markdown-body table,
.markdown-body pre {
margin-top: 0;
margin-bottom: 16px;
}
.markdown-body hr {
height: 4px;
padding: 0;
margin: 16px 0;
background-color: #e7e7e7;
border: 0 none;
}
.markdown-body ul,
.markdown-body ol {
padding-left: 2em;
}
.markdown-body ul ul,
.markdown-body ul ol,
.markdown-body ol ol,
.markdown-body ol ul {
margin-top: 0;
margin-bottom: 0;
}
.markdown-body li>p {
margin-top: 16px;
}
.markdown-body dl {
padding: 0;
}
.markdown-body dl dt {
padding: 0;
margin-top: 16px;
font-size: 1em;
font-style: italic;
font-weight: bold;
}
.markdown-body dl dd {
padding: 0 16px;
margin-bottom: 16px;
}
.markdown-body blockquote {
padding: 0 15px;
color: #777;
border-left: 4px solid #ddd;
}
.markdown-body blockquote>:first-child {
margin-top: 0;
}
.markdown-body blockquote>:last-child {
margin-bottom: 0;
}
.markdown-body table {
display: block;
width: 100%;
overflow: auto;
word-break: normal;
word-break: keep-all;
}
.markdown-body table th {
font-weight: bold;
}
.markdown-body table th,
.markdown-body table td {
padding: 6px 13px;
border: 1px solid #ddd;
}
.markdown-body table tr {
background-color: #fff;
border-top: 1px solid #ccc;
}
.markdown-body table tr:nth-child(2n) {
background-color: #f8f8f8;
}
.markdown-body img {
max-width: 100%;
-moz-box-sizing: border-box;
box-sizing: border-box;
}
.markdown-body code {
padding: 0;
padding-top: 0.2em;
padding-bottom: 0.2em;
margin: 0;
font-size: 85%;
background-color: rgba(0,0,0,0.04);
border-radius: 3px;
}
.markdown-body code:before,
.markdown-body code:after {
letter-spacing: -0.2em;
content: "\00a0";
}
.markdown-body pre>code {
padding: 0;
margin: 0;
font-size: 100%;
word-break: normal;
white-space: pre;
background: transparent;
border: 0;
}
.markdown-body .highlight {
margin-bottom: 16px;
}
.markdown-body .highlight pre,
.markdown-body pre {
padding: 16px;
overflow: auto;
font-size: 85%;
line-height: 1.45;
background-color: #f7f7f7;
border-radius: 3px;
}
.markdown-body .highlight pre {
margin-bottom: 0;
word-break: normal;
}
.markdown-body pre {
word-wrap: normal;
}
.markdown-body pre code {
display: inline;
max-width: initial;
padding: 0;
margin: 0;
overflow: initial;
line-height: inherit;
word-wrap: normal;
background-color: transparent;
border: 0;
}
.markdown-body pre code:before,
.markdown-body pre code:after {
content: normal;
}
.markdown-body .highlight {
background: #fff;
}
.markdown-body .highlight .h {
color: #333;
font-style: normal;
font-weight: normal;
}
.markdown-body .highlight .mf,
.markdown-body .highlight .mh,
.markdown-body .highlight .mi,
.markdown-body .highlight .mo,
.markdown-body .highlight .il,
.markdown-body .highlight .m {
color: #945277;
}
.markdown-body .highlight .s,
.markdown-body .highlight .sb,
.markdown-body .highlight .sc,
.markdown-body .highlight .sd,
.markdown-body .highlight .s2,
.markdown-body .highlight .se,
.markdown-body .highlight .sh,
.markdown-body .highlight .si,
.markdown-body .highlight .sx,
.markdown-body .highlight .s1 {
color: #df5000;
}
.markdown-body .highlight .kc,
.markdown-body .highlight .kd,
.markdown-body .highlight .kn,
.markdown-body .highlight .kp,
.markdown-body .highlight .kr,
.markdown-body .highlight .kt,
.markdown-body .highlight .k,
.markdown-body .highlight .o {
font-weight: bold;
}
.markdown-body .highlight .kt {
color: #458;
}
.markdown-body .highlight .c,
.markdown-body .highlight .cm,
.markdown-body .highlight .c1 {
color: #998;
font-style: italic;
}
.markdown-body .highlight .cp,
.markdown-body .highlight .cs,
.markdown-body .highlight .cp .h {
color: #999;
font-weight: bold;
}
.markdown-body .highlight .cs {
font-style: italic;
}
.markdown-body .highlight .n {
color: #333;
}
.markdown-body .highlight .na,
.markdown-body .highlight .nv,
.markdown-body .highlight .vc,
.markdown-body .highlight .vg,
.markdown-body .highlight .vi {
color: #008080;
}
.markdown-body .highlight .nb {
color: #0086B3;
}
.markdown-body .highlight .nc {
color: #458;
font-weight: bold;
}
.markdown-body .highlight .no {
color: #094e99;
}
.markdown-body .highlight .ni {
color: #800080;
}
.markdown-body .highlight .ne {
color: #990000;
font-weight: bold;
}
.markdown-body .highlight .nf {
color: #945277;
font-weight: bold;
}
.markdown-body .highlight .nn {
color: #555;
}
.markdown-body .highlight .nt {
color: #000080;
}
.markdown-body .highlight .err {
color: #a61717;
background-color: #e3d2d2;
}
.markdown-body .highlight .gd {
color: #000;
background-color: #fdd;
}
.markdown-body .highlight .gd .x {
color: #000;
background-color: #faa;
}
.markdown-body .highlight .ge {
font-style: italic;
}
.markdown-body .highlight .gr {
color: #aa0000;
}
.markdown-body .highlight .gh {
color: #999;
}
.markdown-body .highlight .gi {
color: #000;
background-color: #dfd;
}
.markdown-body .highlight .gi .x {
color: #000;
background-color: #afa;
}
.markdown-body .highlight .go {
color: #888;
}
.markdown-body .highlight .gp {
color: #555;
}
.markdown-body .highlight .gs {
font-weight: bold;
}
.markdown-body .highlight .gu {
color: #800080;
font-weight: bold;
}
.markdown-body .highlight .gt {
color: #aa0000;
}
.markdown-body .highlight .ow {
font-weight: bold;
}
.markdown-body .highlight .w {
color: #bbb;
}
.markdown-body .highlight .sr {
color: #017936;
}
.markdown-body .highlight .ss {
color: #8b467f;
}
.markdown-body .highlight .bp {
color: #999;
}
.markdown-body .highlight .gc {
color: #999;
background-color: #EAF2F5;
}
.markdown-body kbd {
display: inline-block;
padding: 3px 5px;
font-size: 11px;
line-height: 10px;
color: #555;
vertical-align: middle;
background-color: #fcfcfc;
border: solid 1px #ccc;
border-bottom-color: #bbb;
border-radius: 3px;
box-shadow: inset 0 -1px 0 #bbb;
}
.markdown-body .highlight .pl-coc,
.markdown-body .highlight .pl-entm,
.markdown-body .highlight .pl-eoa,
.markdown-body .highlight .pl-mai .pl-sf,
.markdown-body .highlight .pl-pdv,
.markdown-body .highlight .pl-sc,
.markdown-body .highlight .pl-sr,
.markdown-body .highlight .pl-v,
.markdown-body .highlight .pl-vpf {
color: #0086b3;
}
.markdown-body .highlight .pl-eoac,
.markdown-body .highlight .pl-mdht,
.markdown-body .highlight .pl-mi1,
.markdown-body .highlight .pl-mri,
.markdown-body .highlight .pl-va,
.markdown-body .highlight .pl-vpu {
color: #008080;
}
.markdown-body .highlight .pl-c,
.markdown-body .highlight .pl-pdc {
color: #b4b7b4;
font-style: italic;
}
.markdown-body .highlight .pl-k,
.markdown-body .highlight .pl-ko,
.markdown-body .highlight .pl-kolp,
.markdown-body .highlight .pl-mc,
.markdown-body .highlight .pl-mr,
.markdown-body .highlight .pl-ms,
.markdown-body .highlight .pl-s,
.markdown-body .highlight .pl-sok,
.markdown-body .highlight .pl-st {
color: #6e5494;
}
.markdown-body .highlight .pl-ef,
.markdown-body .highlight .pl-enf,
.markdown-body .highlight .pl-enm,
.markdown-body .highlight .pl-entc,
.markdown-body .highlight .pl-eoi,
.markdown-body .highlight .pl-sf,
.markdown-body .highlight .pl-smc {
color: #d12089;
}
.markdown-body .highlight .pl-ens,
.markdown-body .highlight .pl-eoai,
.markdown-body .highlight .pl-kos,
.markdown-body .highlight .pl-mh .pl-pdh,
.markdown-body .highlight .pl-mp,
.markdown-body .highlight .pl-pde,
.markdown-body .highlight .pl-stp {
color: #458;
}
.markdown-body .highlight .pl-enti {
color: #d12089;
font-weight: bold;
}
.markdown-body .highlight .pl-cce,
.markdown-body .highlight .pl-enc,
.markdown-body .highlight .pl-kou,
.markdown-body .highlight .pl-mq {
color: #f93;
}
.markdown-body .highlight .pl-mp1 .pl-sf {
color: #458;
font-weight: bold;
}
.markdown-body .highlight .pl-cos,
.markdown-body .highlight .pl-ent,
.markdown-body .highlight .pl-md,
.markdown-body .highlight .pl-mdhf,
.markdown-body .highlight .pl-ml,
.markdown-body .highlight .pl-pdc1,
.markdown-body .highlight .pl-pds,
.markdown-body .highlight .pl-s1,
.markdown-body .highlight .pl-scp,
.markdown-body .highlight .pl-sol {
color: #df5000;
}
.markdown-body .highlight .pl-c1,
.markdown-body .highlight .pl-cn,
.markdown-body .highlight .pl-pse,
.markdown-body .highlight .pl-pse .pl-s2,
.markdown-body .highlight .pl-vi {
color: #a31515;
}
.markdown-body .highlight .pl-mb,
.markdown-body .highlight .pl-pdb {
color: #df5000;
font-weight: bold;
}
.markdown-body .highlight .pl-mi,
.markdown-body .highlight .pl-pdi {
color: #6e5494;
font-style: italic;
}
.markdown-body .highlight .pl-ms1 {
background-color: #f5f5f5;
}
.markdown-body .highlight .pl-mdh,
.markdown-body .highlight .pl-mdi {
font-weight: bold;
}
.markdown-body .highlight .pl-mdr {
color: #0086b3;
font-weight: bold;
}
.markdown-body .highlight .pl-s2 {
color: #333;
}
.markdown-body .highlight .pl-ii {
background-color: #df5000;
color: #fff;
}
.markdown-body .highlight .pl-ib {
background-color: #f93;
}
.markdown-body .highlight .pl-id {
background-color: #a31515;
color: #fff;
}
.markdown-body .highlight .pl-iu {
background-color: #b4b7b4;
}
.markdown-body .highlight .pl-mo {
color: #969896;
}
.markdown-body .task-list-item {
list-style-type: none;
}
.markdown-body .task-list-item+.task-list-item {
margin-top: 3px;
}
.markdown-body .task-list-item input {
float: left;
margin: 0.3em 0 0.25em -1.6em;
vertical-align: middle;
}

22
server/server.py Normal file
View File

@ -0,0 +1,22 @@
from gevent import monkey, signal_handler as sig
monkey.patch_all()
import bottle
import signal
from gevent.pywsgi import WSGIServer
from geventwebsocket.handler import WebSocketHandler
import config.env as env
import controller.controller as controller
if __name__ == '__main__':
app = bottle.app()
server = WSGIServer(("0.0.0.0", env.port), app, handler_class=WebSocketHandler)
def shutdown():
print('Shutting down ...')
server.stop(timeout=60)
exit(signal.SIGTERM)
sig(signal.SIGTERM, shutdown)
sig(signal.SIGINT, shutdown)
server.serve_forever()

170
server/view/html.py Normal file
View File

@ -0,0 +1,170 @@
import codecs
import os
import config.env as env
css_md_content = ""
with codecs.open(env.css_md_path, encoding=env.encoding, mode="r") as css:
css_md_content = css.read()
css_index_content = ""
with codecs.open(env.css_index_path, encoding=env.encoding, mode="r") as css:
css_index_content = css.read()
html_header = (
"""
<html>
<head>
<style type='text/css'>
<!--
"""
+ css_md_content
+ """
//-->
</style>
</head>
<body>
<div class='markdown-body'>
"""
)
html_footer = """
</div>
</body>
</html>
"""
style_basic = """
@media (prefers-color-scheme: dark) {
/* this is the github markdown dark background */
html { background-color: #0d1117; }
}
body {
width: 92%; max-width: 40em; margin: auto;
font-size: 1.2rem; line-height: 1.5; padding-bottom: 3em;}
code { font-family: "DejaVu Sans Mono", Consolas, monospace; }
pre { overflow: auto; }
math, .LaTeX { font-size: 1.1em; }
div.figure .caption { padding: 0em 1em 1em; font-size: 0.8em; }
ul li { list-style-type: disc; }
li p { margin: 0.5em 0; }
#TOC { border: 1px solid lightgray; padding: 0.5em; margin-bottom: 1em;}
#TOC li { list-style: circle; }
#TOC > ul { margin-bottom: 0px }
@media (min-width: 102em) {
body { position: relative; }
#TOC { display: block; position: absolute; left: -50%; top: 5em; max-width: 46%; width: 45%; }
}
"""
style_document_add = """
@media (max-width: 101em) {
#TOC { display: none; }
}
.topmenu { font-size: 1em; color: lightgrey; }
.topmenu a { color: lightgrey; }
"""
style_index = """
ul.file-listing a { text-decoration: none; }
ul.file-listing ul.file-listing { padding-left: 1.5em; }
ul li.dir-entry {
list-style-image: url('');
}
ul li.file-entry{
list-style-image: url('');
}
@media (prefers-color-scheme: dark) {
ul li.dir-entry {
list-style-image: url('');
}
ul li.file-entry{
list-style-image: url('');
}
}
"""
#markdown_css_link= """<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/github-markdown-css/5.1.0/github-markdown.css">"""
def show_nested_files(files):
subdirectory_files = files[1]
text = ''
for nested in subdirectory_files:
if type(nested) is str:
text += f'<li class="file-entry"><a href="/view/{nested}">{os.path.basename(nested)}</a>'
reportable_dirs = [item + '\\' for item in [env.test_cases_dir, env.test_suites_dir]]
if any([directory in nested for directory in reportable_dirs]):
text += f' <small><a href="/report/{nested}">[report]</a></small>'
text += '</li>'
elif type(nested) is tuple:
path = nested[0]
text += '<li class="dir-entry">' + os.path.basename(path) + '</li>'
text += show_nested_files(nested)
text = '<ul class="file-listing">' + text + '</ul>'
return text
def index(files):
text = ""
text += "<html><head><title>Server Index</title>"
text += '<style type="text/css">{}</style>'.format(style_basic + style_index)
text += """<style type='text/css'>
<!--
""" + css_index_content + """
//-->
</style>""".format(style_basic + style_index)
text += "</head><body class='markdown-body' style='max-width: 600px; margin: auto; margin-top: 80px'>"
text += '<h1>Testing documentation for Dipal:</h1>'
text += "<h3>Directory contents</h3>"
text += show_nested_files(files)
text += '</body></html>'
return text
def render_file_entry(nested):
filename = os.path.basename(nested)
link = f'<a href="/view/{nested}">{filename}</a>'
report_link = f' <small><a href="/report/{nested}">[report]</a></small>' if 'compiled\\' in nested else ''
return f'<li class="file-entry">{link}{report_link}</li>'
def render_directory_entry(nested):
dirname = os.path.basename(nested[0])
nested_files = nested[1]
return f'<li class="dir-entry">{dirname}{show_nested_files(nested_files[0])}</li>'
def show_nested_files2(files):
print(1, files)
nested_items = files[1]
nested_items_rendered = ''
for nested in nested_items:
if isinstance(nested, str):
nested_items_rendered += render_file_entry(nested)
elif isinstance(nested, tuple):
nested_items_rendered += render_directory_entry(nested)
return f'<ul class="file-listing">{nested_items_rendered}</ul>'
def index2(files):
style = style_basic + style_index
markdown_css = markdown_css_link
content = show_nested_files(files)
return f"""
<html>
<head>
<title>Server Index</title>
<style type="text/css">{style}</style>
{markdown_css}
</head>
<body class="markdown-body" style="max-width: 600px; margin: auto; margin-top: 80px">
<h1>Testing documentation for Dipal:</h1>
<h3>Directory contents</h3>
{content}
</body>
</html>
"""
def to_html(body):
return html_header + body + html_footer