mirror of
https://gitlab.com/crafty-controller/crafty-4.git
synced 2025-01-19 09:45:28 +01:00
Merge branch 'dev' into 'master'
v4.2.0 Closes #267 and #269 See merge request crafty-controller/crafty-4!650
This commit is contained in:
commit
de97cb799e
@ -11,6 +11,7 @@ docker-compose.yml.example
|
|||||||
.gitlab/
|
.gitlab/
|
||||||
.gitignore
|
.gitignore
|
||||||
.gitlab-ci.yml
|
.gitlab-ci.yml
|
||||||
|
lang_sort_log.txt
|
||||||
|
|
||||||
# root
|
# root
|
||||||
.editorconfig
|
.editorconfig
|
||||||
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -18,6 +18,7 @@ env.bak/
|
|||||||
venv.bak/
|
venv.bak/
|
||||||
|
|
||||||
.idea/
|
.idea/
|
||||||
|
/import/
|
||||||
/imports/
|
/imports/
|
||||||
/servers/
|
/servers/
|
||||||
/app/frontend/static/assets/images/auth/custom/
|
/app/frontend/static/assets/images/auth/custom/
|
||||||
@ -35,3 +36,4 @@ default.json
|
|||||||
app/config/
|
app/config/
|
||||||
docker/*
|
docker/*
|
||||||
!docker/docker-compose.yml
|
!docker/docker-compose.yml
|
||||||
|
lang_sort_log.txt
|
||||||
|
@ -81,3 +81,26 @@ sonarcloud-check:
|
|||||||
- .sonar/cache
|
- .sonar/cache
|
||||||
script:
|
script:
|
||||||
- sonar-scanner
|
- sonar-scanner
|
||||||
|
|
||||||
|
# Lang file checking
|
||||||
|
lang-check:
|
||||||
|
stage: lint
|
||||||
|
image: alpine:latest
|
||||||
|
tags:
|
||||||
|
- docker
|
||||||
|
rules:
|
||||||
|
- if: "$CODE_QUALITY_DISABLED"
|
||||||
|
when: never
|
||||||
|
- if: "$CI_COMMIT_TAG || $CI_COMMIT_BRANCH"
|
||||||
|
allow_failure: true
|
||||||
|
before_script:
|
||||||
|
- apk add --no-cache jq bash
|
||||||
|
script:
|
||||||
|
- chmod +x .gitlab/scripts/lang_sort.sh
|
||||||
|
- bash .gitlab/scripts/lang_sort.sh ./app/translations/
|
||||||
|
after_script:
|
||||||
|
- if [ -f .gitlab/scripts/lang_sort_log.txt ]; then cat .gitlab/scripts/lang_sort_log.txt; fi
|
||||||
|
artifacts:
|
||||||
|
paths:
|
||||||
|
- .gitlab/scripts/lang_sort_log.txt
|
||||||
|
expire_in: 1 week
|
||||||
|
95
.gitlab/scripts/lang_sort.sh
Normal file
95
.gitlab/scripts/lang_sort.sh
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# Ensure locale is set to C for predictable sorting
|
||||||
|
export LC_ALL=C
|
||||||
|
export LC_COLLATE=C
|
||||||
|
|
||||||
|
# Get the script's own path
|
||||||
|
SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
# Directory containing the JSON files to sort
|
||||||
|
DIR="$1"
|
||||||
|
found_missing_keys=false
|
||||||
|
|
||||||
|
|
||||||
|
##### Log Setup #####
|
||||||
|
# Log file path
|
||||||
|
LOGFILE="${SCRIPT_DIR}/lang_sort_log.txt"
|
||||||
|
|
||||||
|
# Redirect stdout and stderr to the logfile
|
||||||
|
exec > "${LOGFILE}" 2>&1
|
||||||
|
#####################
|
||||||
|
|
||||||
|
|
||||||
|
##### Exit Gates #####
|
||||||
|
# Check if jq is installed
|
||||||
|
if ! command -v jq &> /dev/null
|
||||||
|
then
|
||||||
|
echo "jq could not be found, please install jq first."
|
||||||
|
exit
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check for directory argument
|
||||||
|
if [ "$#" -ne 1 ]; then
|
||||||
|
echo "Usage: $0 /path/to/translations"
|
||||||
|
exit
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Check if en_EN.json exists in the directory
|
||||||
|
if [[ ! -f "${DIR}/en_EN.json" ]]; then
|
||||||
|
echo "The file en_EN.json does not exist in ${DIR}.Ensure you have the right directory, Exiting."
|
||||||
|
exit
|
||||||
|
fi
|
||||||
|
######################
|
||||||
|
|
||||||
|
|
||||||
|
# Sort keys of the en_EN.json file with 4-space indentation and overwrite it
|
||||||
|
jq -S --indent 4 '.' "${DIR}/en_EN.json" > "${DIR}/en_EN.json.tmp" && mv "${DIR}/en_EN.json.tmp" "${DIR}/en_EN.json"
|
||||||
|
|
||||||
|
# Function to recursively find all keys in a JSON object
|
||||||
|
function get_keys {
|
||||||
|
jq -r 'paths(scalars) | join("/")' "$1"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Get keys and subkeys from en_EN.json
|
||||||
|
ref_keys=$(mktemp)
|
||||||
|
get_keys "${DIR}/en_EN.json" | sort > "${ref_keys}"
|
||||||
|
|
||||||
|
# Iterate over each .json file in the directory
|
||||||
|
for file in "${DIR}"/*.json; do
|
||||||
|
# Check if file is a regular file and not en_EN.json, and does not contain "_incomplete" in its name
|
||||||
|
if [[ -f "${file}" && "${file}" != "${DIR}/en_EN.json" && ! "${file}" =~ _incomplete ]]; then
|
||||||
|
|
||||||
|
# Get keys and subkeys from the current file
|
||||||
|
current_keys=$(mktemp)
|
||||||
|
get_keys "${file}" | sort > "${current_keys}"
|
||||||
|
|
||||||
|
# Display keys present in en_EN.json but not in the current file
|
||||||
|
missing_keys=$(comm -23 "${ref_keys}" "${current_keys}")
|
||||||
|
if [[ -n "${missing_keys}" ]]; then
|
||||||
|
found_missing_keys=true
|
||||||
|
echo -e "\nKeys/subkeys present in en_EN.json but missing in $(basename "${file}"): "
|
||||||
|
echo "${missing_keys}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Sort keys of the JSON file and overwrite the original file
|
||||||
|
jq -S --indent 4 '.' "${file}" > "${file}.tmp" && mv "${file}.tmp" "${file}"
|
||||||
|
|
||||||
|
# Remove the temporary file
|
||||||
|
rm -f "${current_keys}"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# Remove the temporary file
|
||||||
|
rm -f "${ref_keys}"
|
||||||
|
|
||||||
|
if ${found_missing_keys}; then
|
||||||
|
echo -e "\n\nSorting complete!"
|
||||||
|
echo "Comparison found missing keys, Please Review!"
|
||||||
|
echo "-------------------------------------------------------------------"
|
||||||
|
echo "If there are stale translations, you can exclude with '_incomplete'"
|
||||||
|
echo " e.g. lol_EN_incomplete.json"
|
||||||
|
echo "-------------------------------------------------------------------"
|
||||||
|
exit 1
|
||||||
|
else
|
||||||
|
echo -e "\n\nComparison and Sorting complete!"
|
||||||
|
fi
|
45
CHANGELOG.md
45
CHANGELOG.md
@ -1,4 +1,49 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
## --- [4.2.0] - 2023/10/18
|
||||||
|
### New features
|
||||||
|
- Finish and Activate Arcadia notification backend ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/621) | [Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/626) | [Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/632))
|
||||||
|
- Add initial Webhook Notification (Discord, Mattermost, Slack, Teams) ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/594))
|
||||||
|
- Implementation of OpenMetrics endpoints, for use with services such as Prometheus ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/624))
|
||||||
|
### Bug fixes
|
||||||
|
- PWA: Removed the custom offline page in favour of browser default ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/607))
|
||||||
|
- Fix hidden servers appearing visible on public mobile status page ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/612))
|
||||||
|
- Correctly handle if a server returns a string instead of json data on socket ping ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/614))
|
||||||
|
- Bump tornado to resolve #269 ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/623))
|
||||||
|
- Bump crypto to resolve #267 & #268 ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/622))
|
||||||
|
- Fix select installs failing to start, returning missing python package `packaging` ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/629))
|
||||||
|
- Fix public status page not updating #255 ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/615))
|
||||||
|
- Fix service worker vulrn and CQ raised by SonarQ ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/631))
|
||||||
|
- Fix Backup Restore/Schedules, Backup button function on `remote-comms2` ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/634))
|
||||||
|
- Add a wait to the call for the directory so we can make sure the wait dialogue has time to show up first ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/637))
|
||||||
|
- Fix bug where a reaction loop could be created, but would be cut short by an error when the loop occurred ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/636))
|
||||||
|
- Use controller on update user call ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/640))
|
||||||
|
- Move `imports` to `import/upload` in bind mount to better serve users on unraid with limited vdisk storage ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/642))
|
||||||
|
- Fix bug where everytime a page was loaded user settings would be reset #286 ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/643))
|
||||||
|
- Fix tooltip info icon on server config page ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/647))
|
||||||
|
- Fix quick disable toggle on schedules list ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/649))
|
||||||
|
### Refactor
|
||||||
|
- Consolidate remaining frontend functions into API V2, and remove ajax internal API ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/585))
|
||||||
|
- Replace bleach with nh3 ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/628))
|
||||||
|
- Add API route for historical server stats ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/615))
|
||||||
|
- Add API route for host stats ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/615))
|
||||||
|
### Tweaks
|
||||||
|
- Polish/Enhance display for InApp Documentation ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/613))
|
||||||
|
- Add `get_users` command to Crafty's console ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/620))
|
||||||
|
- Make files hover cursor pointer ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/627))
|
||||||
|
- Use `Jar` class naming for jar refresh to make room for steamCMD naming in the future ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/630))
|
||||||
|
- Improve ui visibility of Build Wizard selection tabs ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/633))
|
||||||
|
- Add additional logging for server bootstrap & moves unnecessary logging to `debug` for improved log clarity ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/635))
|
||||||
|
- Bump orjson to `3.9.7` for python `3.12` support ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/638))
|
||||||
|
- Bump all Crafty required python dependancies, maintaining minimum `3.9` support ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/639)) Revert peewee bump ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/651))
|
||||||
|
- Better optimize and refactor docker launcher sh ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/642))
|
||||||
|
- Improve pop-up notifications with Toasts ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/641))
|
||||||
|
- Move username and password settings to buttons on panel config ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/643))
|
||||||
|
- Remove external references from front end deps ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/648))
|
||||||
|
### Lang
|
||||||
|
- `fr_FR` Translation Updated to latest en_EN ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/646))
|
||||||
|
- `de_DE`, `fr_FR`, `lol_EN`, `lv_LV`, `nl_BE`, `pl_PL` Translations Updated to latest `en_EN` ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/645))
|
||||||
|
<br><br>
|
||||||
|
|
||||||
## --- [4.1.3] - 2023/07/18
|
## --- [4.1.3] - 2023/07/18
|
||||||
### Bug fixes
|
### Bug fixes
|
||||||
- Include tzdata in Docker image ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/604))
|
- Include tzdata in Docker image ([Merge Request](https://gitlab.com/crafty-controller/crafty-4/-/merge_requests/604))
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
[![Crafty Logo](app/frontend/static/assets/images/logo_long.svg)](https://craftycontrol.com)
|
[![Crafty Logo](app/frontend/static/assets/images/logo_long.svg)](https://craftycontrol.com)
|
||||||
# Crafty Controller 4.1.3
|
# Crafty Controller 4.2.0
|
||||||
> Python based Control Panel for your Minecraft Server
|
> Python based Control Panel for your Minecraft Server
|
||||||
|
|
||||||
## What is Crafty Controller?
|
## What is Crafty Controller?
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
import logging
|
import logging
|
||||||
import queue
|
import queue
|
||||||
|
|
||||||
from app.classes.models.management import HelpersManagement
|
from prometheus_client import CollectorRegistry, Gauge
|
||||||
|
|
||||||
|
from app.classes.models.management import HelpersManagement, HelpersWebhooks
|
||||||
from app.classes.models.servers import HelperServers
|
from app.classes.models.servers import HelperServers
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -11,6 +13,8 @@ class ManagementController:
|
|||||||
def __init__(self, management_helper):
|
def __init__(self, management_helper):
|
||||||
self.management_helper = management_helper
|
self.management_helper = management_helper
|
||||||
self.command_queue = queue.Queue()
|
self.command_queue = queue.Queue()
|
||||||
|
self.host_registry = CollectorRegistry()
|
||||||
|
self.init_host_registries()
|
||||||
|
|
||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
# Config Methods
|
# Config Methods
|
||||||
@ -54,6 +58,19 @@ class ManagementController:
|
|||||||
def add_crafty_row():
|
def add_crafty_row():
|
||||||
HelpersManagement.create_crafty_row()
|
HelpersManagement.create_crafty_row()
|
||||||
|
|
||||||
|
def init_host_registries(self):
|
||||||
|
# REGISTRY Entries for Server Stats functions
|
||||||
|
self.cpu_usage = Gauge(
|
||||||
|
name="CPU_Usage",
|
||||||
|
documentation="The CPU usage of the server",
|
||||||
|
registry=self.host_registry,
|
||||||
|
)
|
||||||
|
self.mem_usage_percent = Gauge(
|
||||||
|
name="Mem_Usage",
|
||||||
|
documentation="The Memory usage of the server",
|
||||||
|
registry=self.host_registry,
|
||||||
|
)
|
||||||
|
|
||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
# Commands Methods
|
# Commands Methods
|
||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
@ -79,8 +96,8 @@ class ManagementController:
|
|||||||
# Audit_Log Methods
|
# Audit_Log Methods
|
||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_actity_log():
|
def get_activity_log():
|
||||||
return HelpersManagement.get_actity_log()
|
return HelpersManagement.get_activity_log()
|
||||||
|
|
||||||
def add_to_audit_log(self, user_id, log_msg, server_id=None, source_ip=None):
|
def add_to_audit_log(self, user_id, log_msg, server_id=None, source_ip=None):
|
||||||
return self.management_helper.add_to_audit_log(
|
return self.management_helper.add_to_audit_log(
|
||||||
@ -206,3 +223,30 @@ class ManagementController:
|
|||||||
@staticmethod
|
@staticmethod
|
||||||
def set_master_server_dir(server_dir):
|
def set_master_server_dir(server_dir):
|
||||||
HelpersManagement.set_master_server_dir(server_dir)
|
HelpersManagement.set_master_server_dir(server_dir)
|
||||||
|
|
||||||
|
# **********************************************************************************
|
||||||
|
# Webhooks Methods
|
||||||
|
# **********************************************************************************
|
||||||
|
@staticmethod
|
||||||
|
def create_webhook(data):
|
||||||
|
return HelpersWebhooks.create_webhook(data)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def modify_webhook(webhook_id, data):
|
||||||
|
HelpersWebhooks.modify_webhook(webhook_id, data)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_webhook_by_id(webhook_id):
|
||||||
|
return HelpersWebhooks.get_webhook_by_id(webhook_id)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_webhooks_by_server(server_id, model=False):
|
||||||
|
return HelpersWebhooks.get_webhooks_by_server(server_id, model)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def delete_webhook(webhook_id):
|
||||||
|
HelpersWebhooks.delete_webhook(webhook_id)
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def delete_webhook_by_server(server_id):
|
||||||
|
HelpersWebhooks.delete_webhooks_by_server(server_id)
|
||||||
|
@ -105,9 +105,9 @@ class ServersController(metaclass=Singleton):
|
|||||||
|
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
def get_history_stats(self, server_id, days):
|
def get_history_stats(self, server_id, hours):
|
||||||
srv = ServersController().get_server_instance_by_id(server_id)
|
srv = ServersController().get_server_instance_by_id(server_id)
|
||||||
return srv.stats_helper.get_history_stats(server_id, days)
|
return srv.stats_helper.get_history_stats(server_id, hours)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def update_unloaded_server(server_obj):
|
def update_unloaded_server(server_obj):
|
||||||
|
@ -31,7 +31,7 @@ class UsersController:
|
|||||||
for permission in PermissionsCrafty.get_permissions_list()
|
for permission in PermissionsCrafty.get_permissions_list()
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
"quantity": {"type": "number", "minimum": 0},
|
"quantity": {"type": "number", "minimum": -1},
|
||||||
"enabled": {"type": "boolean"},
|
"enabled": {"type": "boolean"},
|
||||||
}
|
}
|
||||||
self.user_jsonschema_props: t.Final = {
|
self.user_jsonschema_props: t.Final = {
|
||||||
@ -46,7 +46,7 @@ class UsersController:
|
|||||||
"password": {
|
"password": {
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"maxLength": 20,
|
"maxLength": 20,
|
||||||
"minLength": 4,
|
"minLength": 6,
|
||||||
"examples": ["crafty"],
|
"examples": ["crafty"],
|
||||||
"title": "Password",
|
"title": "Password",
|
||||||
},
|
},
|
||||||
@ -73,6 +73,8 @@ class UsersController:
|
|||||||
"examples": [False],
|
"examples": [False],
|
||||||
"title": "Superuser",
|
"title": "Superuser",
|
||||||
},
|
},
|
||||||
|
"manager": {"type": ["integer", "null"]},
|
||||||
|
"theme": {"type": "string"},
|
||||||
"permissions": {
|
"permissions": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
@ -84,7 +86,7 @@ class UsersController:
|
|||||||
"roles": {
|
"roles": {
|
||||||
"type": "array",
|
"type": "array",
|
||||||
"items": {
|
"items": {
|
||||||
"type": "string",
|
"type": "integer",
|
||||||
"minLength": 1,
|
"minLength": 1,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -212,14 +214,14 @@ class UsersController:
|
|||||||
limit_server_creation = 0
|
limit_server_creation = 0
|
||||||
limit_user_creation = 0
|
limit_user_creation = 0
|
||||||
limit_role_creation = 0
|
limit_role_creation = 0
|
||||||
|
if user_crafty_data:
|
||||||
PermissionsCrafty.add_or_update_user(
|
PermissionsCrafty.add_or_update_user(
|
||||||
user_id,
|
user_id,
|
||||||
permissions_mask,
|
permissions_mask,
|
||||||
limit_server_creation,
|
limit_server_creation,
|
||||||
limit_user_creation,
|
limit_user_creation,
|
||||||
limit_role_creation,
|
limit_role_creation,
|
||||||
)
|
)
|
||||||
|
|
||||||
self.users_helper.delete_user_roles(user_id, removed_roles)
|
self.users_helper.delete_user_roles(user_id, removed_roles)
|
||||||
|
|
||||||
|
@ -16,6 +16,12 @@ logger = logging.getLogger(__name__)
|
|||||||
|
|
||||||
class Server:
|
class Server:
|
||||||
def __init__(self, data):
|
def __init__(self, data):
|
||||||
|
if isinstance(data, str):
|
||||||
|
logger.error(
|
||||||
|
"Failed to calculate stats. Expected object. "
|
||||||
|
f"Server returned string: {data}"
|
||||||
|
)
|
||||||
|
return
|
||||||
self.description = data.get("description")
|
self.description = data.get("description")
|
||||||
# print(self.description)
|
# print(self.description)
|
||||||
if isinstance(self.description, dict):
|
if isinstance(self.description, dict):
|
||||||
|
@ -8,6 +8,7 @@ import requests
|
|||||||
|
|
||||||
from app.classes.controllers.servers_controller import ServersController
|
from app.classes.controllers.servers_controller import ServersController
|
||||||
from app.classes.models.server_permissions import PermissionsServers
|
from app.classes.models.server_permissions import PermissionsServers
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -179,9 +180,7 @@ class ServerJars:
|
|||||||
try:
|
try:
|
||||||
ServersController.set_import(server_id)
|
ServersController.set_import(server_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
user, "send_start_reload", {}
|
|
||||||
)
|
|
||||||
|
|
||||||
break
|
break
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
@ -206,11 +205,9 @@ class ServerJars:
|
|||||||
server_users = PermissionsServers.get_server_user_list(server_id)
|
server_users = PermissionsServers.get_server_user_list(server_id)
|
||||||
|
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user, "notification", "Executable download finished"
|
user, "notification", "Executable download finished"
|
||||||
)
|
)
|
||||||
time.sleep(3)
|
time.sleep(3)
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
user, "send_start_reload", {}
|
|
||||||
)
|
|
||||||
return success
|
return success
|
||||||
|
@ -226,7 +226,7 @@ class Stats:
|
|||||||
def get_server_players(self, server_id):
|
def get_server_players(self, server_id):
|
||||||
server = HelperServers.get_server_data_by_id(server_id)
|
server = HelperServers.get_server_data_by_id(server_id)
|
||||||
|
|
||||||
logger.info(f"Getting players for server {server}")
|
logger.debug(f"Getting players for server {server['server_name']}")
|
||||||
|
|
||||||
internal_ip = server["server_ip"]
|
internal_ip = server["server_ip"]
|
||||||
server_port = server["server_port"]
|
server_port = server["server_port"]
|
||||||
|
@ -17,6 +17,7 @@ from app.classes.models.users import HelperUsers
|
|||||||
from app.classes.models.servers import Servers
|
from app.classes.models.servers import Servers
|
||||||
from app.classes.models.server_permissions import PermissionsServers
|
from app.classes.models.server_permissions import PermissionsServers
|
||||||
from app.classes.shared.main_models import DatabaseShortcuts
|
from app.classes.shared.main_models import DatabaseShortcuts
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -78,11 +79,15 @@ class HostStats(BaseModel):
|
|||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
class Webhooks(BaseModel):
|
class Webhooks(BaseModel):
|
||||||
id = AutoField()
|
id = AutoField()
|
||||||
name = CharField(max_length=64, unique=True, index=True)
|
server_id = IntegerField(null=True)
|
||||||
method = CharField(default="POST")
|
name = CharField(default="Custom Webhook", max_length=64)
|
||||||
url = CharField(unique=True)
|
url = CharField(default="")
|
||||||
event = CharField(default="")
|
webhook_type = CharField(default="Custom")
|
||||||
send_data = BooleanField(default=True)
|
bot_name = CharField(default="Crafty Controller")
|
||||||
|
trigger = CharField(default="server_start,server_stop")
|
||||||
|
body = CharField(default="")
|
||||||
|
color = CharField(default="#005cd1")
|
||||||
|
enabled = BooleanField(default=True)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
table_name = "webhooks"
|
table_name = "webhooks"
|
||||||
@ -145,7 +150,7 @@ class HelpersManagement:
|
|||||||
# Audit_Log Methods
|
# Audit_Log Methods
|
||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_actity_log():
|
def get_activity_log():
|
||||||
query = AuditLog.select()
|
query = AuditLog.select()
|
||||||
return DatabaseShortcuts.return_db_rows(query)
|
return DatabaseShortcuts.return_db_rows(query)
|
||||||
|
|
||||||
@ -158,9 +163,7 @@ class HelpersManagement:
|
|||||||
server_users = PermissionsServers.get_server_user_list(server_id)
|
server_users = PermissionsServers.get_server_user_list(server_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
try:
|
try:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(user, "notification", audit_msg)
|
||||||
user, "notification", audit_msg
|
|
||||||
)
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.error(f"Error broadcasting to user {user} - {e}")
|
logger.error(f"Error broadcasting to user {user} - {e}")
|
||||||
|
|
||||||
@ -502,3 +505,82 @@ class HelpersManagement:
|
|||||||
f"Not removing {dir_to_del} from excluded directories - "
|
f"Not removing {dir_to_del} from excluded directories - "
|
||||||
f"not in the excluded directory list for server ID {server_id}"
|
f"not in the excluded directory list for server ID {server_id}"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
# **********************************************************************************
|
||||||
|
# Webhooks Class
|
||||||
|
# **********************************************************************************
|
||||||
|
class HelpersWebhooks:
|
||||||
|
def __init__(self, database):
|
||||||
|
self.database = database
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def create_webhook(create_data) -> int:
|
||||||
|
"""Create a webhook in the database
|
||||||
|
|
||||||
|
Args:
|
||||||
|
server_id: ID of a server this webhook will be married to
|
||||||
|
name: The name of the webhook
|
||||||
|
url: URL to the webhook
|
||||||
|
webhook_type: The provider this webhook will be sent to
|
||||||
|
bot name: The name that will appear when the webhook is sent
|
||||||
|
triggers: Server actions that will trigger this webhook
|
||||||
|
body: The message body of the webhook
|
||||||
|
enabled: Should Crafty trigger the webhook
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
int: The new webhooks's id
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
PeeweeException: If the webhook already exists
|
||||||
|
"""
|
||||||
|
return Webhooks.insert(
|
||||||
|
{
|
||||||
|
Webhooks.server_id: create_data["server_id"],
|
||||||
|
Webhooks.name: create_data["name"],
|
||||||
|
Webhooks.webhook_type: create_data["webhook_type"],
|
||||||
|
Webhooks.url: create_data["url"],
|
||||||
|
Webhooks.bot_name: create_data["bot_name"],
|
||||||
|
Webhooks.body: create_data["body"],
|
||||||
|
Webhooks.color: create_data["color"],
|
||||||
|
Webhooks.trigger: create_data["trigger"],
|
||||||
|
Webhooks.enabled: create_data["enabled"],
|
||||||
|
}
|
||||||
|
).execute()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def modify_webhook(webhook_id, updata):
|
||||||
|
Webhooks.update(updata).where(Webhooks.id == webhook_id).execute()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_webhook_by_id(webhook_id):
|
||||||
|
return model_to_dict(Webhooks.get(Webhooks.id == webhook_id))
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_webhooks_by_server(server_id, model):
|
||||||
|
if not model:
|
||||||
|
data = {}
|
||||||
|
for webhook in (
|
||||||
|
Webhooks.select().where(Webhooks.server_id == server_id).execute()
|
||||||
|
):
|
||||||
|
data[str(webhook.id)] = {
|
||||||
|
"webhook_type": webhook.webhook_type,
|
||||||
|
"name": webhook.name,
|
||||||
|
"url": webhook.url,
|
||||||
|
"bot_name": webhook.bot_name,
|
||||||
|
"trigger": webhook.trigger,
|
||||||
|
"body": webhook.body,
|
||||||
|
"color": webhook.color,
|
||||||
|
"enabled": webhook.enabled,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
data = Webhooks.select().where(Webhooks.server_id == server_id).execute()
|
||||||
|
return data
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def delete_webhook(webhook_id):
|
||||||
|
Webhooks.delete().where(Webhooks.id == webhook_id).execute()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def delete_webhooks_by_server(server_id):
|
||||||
|
Webhooks.delete().where(Webhooks.server_id == server_id).execute()
|
||||||
|
@ -8,6 +8,7 @@ from app.classes.shared.helpers import Helpers
|
|||||||
from app.classes.shared.main_models import DatabaseShortcuts
|
from app.classes.shared.main_models import DatabaseShortcuts
|
||||||
from app.classes.shared.migration import MigrationManager
|
from app.classes.shared.migration import MigrationManager
|
||||||
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from peewee import (
|
from peewee import (
|
||||||
SqliteDatabase,
|
SqliteDatabase,
|
||||||
@ -50,6 +51,7 @@ class ServerStats(Model):
|
|||||||
max = IntegerField(default=0)
|
max = IntegerField(default=0)
|
||||||
players = CharField(default="")
|
players = CharField(default="")
|
||||||
desc = CharField(default="Unable to Connect")
|
desc = CharField(default="Unable to Connect")
|
||||||
|
icon = CharField(default="")
|
||||||
version = CharField(default="")
|
version = CharField(default="")
|
||||||
updating = BooleanField(default=False)
|
updating = BooleanField(default=False)
|
||||||
waiting_start = BooleanField(default=False)
|
waiting_start = BooleanField(default=False)
|
||||||
@ -141,16 +143,20 @@ class HelperServerStats:
|
|||||||
self.database.close()
|
self.database.close()
|
||||||
return server_data
|
return server_data
|
||||||
|
|
||||||
def get_history_stats(self, server_id, num_days):
|
def get_history_stats(self, server_id, num_hours):
|
||||||
self.database.connect(reuse_if_open=True)
|
self.database.connect(reuse_if_open=True)
|
||||||
max_age = datetime.datetime.now() - timedelta(days=num_days)
|
max_age = datetime.datetime.now() - timedelta(hours=num_hours)
|
||||||
server_stats = (
|
query_stats = (
|
||||||
ServerStats.select()
|
ServerStats.select()
|
||||||
.where(ServerStats.created > max_age)
|
.where(ServerStats.created > max_age)
|
||||||
.where(ServerStats.server_id == server_id)
|
.where(ServerStats.server_id == server_id)
|
||||||
|
# .order_by(ServerStats.created.desc())
|
||||||
.execute(self.database)
|
.execute(self.database)
|
||||||
)
|
)
|
||||||
self.database.connect(reuse_if_open=True)
|
server_stats = []
|
||||||
|
for stat in query_stats:
|
||||||
|
server_stats.append(DatabaseShortcuts.get_data_obj(stat))
|
||||||
|
self.database.close()
|
||||||
return server_stats
|
return server_stats
|
||||||
|
|
||||||
def insert_server_stats(self, server_stats):
|
def insert_server_stats(self, server_stats):
|
||||||
@ -179,6 +185,7 @@ class HelperServerStats:
|
|||||||
ServerStats.max: server_stats.get("max", False),
|
ServerStats.max: server_stats.get("max", False),
|
||||||
ServerStats.players: server_stats.get("players", False),
|
ServerStats.players: server_stats.get("players", False),
|
||||||
ServerStats.desc: server_stats.get("desc", False),
|
ServerStats.desc: server_stats.get("desc", False),
|
||||||
|
ServerStats.icon: server_stats.get("icon", None),
|
||||||
ServerStats.version: server_stats.get("version", False),
|
ServerStats.version: server_stats.get("version", False),
|
||||||
}
|
}
|
||||||
).execute(self.database)
|
).execute(self.database)
|
||||||
|
@ -45,6 +45,7 @@ class Users(BaseModel):
|
|||||||
manager = IntegerField(default=None, null=True)
|
manager = IntegerField(default=None, null=True)
|
||||||
pfp = CharField(default="/static/assets/images/faces-clipart/pic-3.png")
|
pfp = CharField(default="/static/assets/images/faces-clipart/pic-3.png")
|
||||||
theme = CharField(default="default")
|
theme = CharField(default="default")
|
||||||
|
cleared_notifs = CharField(default="default")
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
table_name = "users"
|
table_name = "users"
|
||||||
@ -171,6 +172,7 @@ class HelperUsers:
|
|||||||
"roles": [],
|
"roles": [],
|
||||||
"servers": [],
|
"servers": [],
|
||||||
"support_logs": "",
|
"support_logs": "",
|
||||||
|
"cleared_notifs": "",
|
||||||
}
|
}
|
||||||
user = model_to_dict(Users.get(Users.user_id == user_id))
|
user = model_to_dict(Users.get(Users.user_id == user_id))
|
||||||
|
|
||||||
|
@ -11,6 +11,7 @@ from app.classes.shared.helpers import Helpers
|
|||||||
from app.classes.shared.tasks import TasksManager
|
from app.classes.shared.tasks import TasksManager
|
||||||
from app.classes.shared.migration import MigrationManager
|
from app.classes.shared.migration import MigrationManager
|
||||||
from app.classes.shared.main_controller import Controller
|
from app.classes.shared.main_controller import Controller
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -92,6 +93,9 @@ class MainPrompt(cmd.Cmd):
|
|||||||
|
|
||||||
self.controller.users.update_user(user_id, {"password": new_pass})
|
self.controller.users.update_user(user_id, {"password": new_pass})
|
||||||
|
|
||||||
|
def do_get_users(self, _line):
|
||||||
|
Console.info(self.controller.users.get_all_usernames())
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def do_threads(_line):
|
def do_threads(_line):
|
||||||
for thread in threading.enumerate():
|
for thread in threading.enumerate():
|
||||||
@ -115,7 +119,7 @@ class MainPrompt(cmd.Cmd):
|
|||||||
Console.info(
|
Console.info(
|
||||||
"Stopping all server daemons / threads - This may take a few seconds"
|
"Stopping all server daemons / threads - This may take a few seconds"
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.disconnect_all()
|
WebSocketManager().disconnect_all()
|
||||||
Console.info("Waiting for main thread to stop")
|
Console.info("Waiting for main thread to stop")
|
||||||
while True:
|
while True:
|
||||||
if self.tasks_manager.get_main_thread_run_status():
|
if self.tasks_manager.get_main_thread_run_status():
|
||||||
|
@ -8,6 +8,7 @@ from zipfile import ZipFile, ZIP_DEFLATED
|
|||||||
|
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.console import Console
|
from app.classes.shared.console import Console
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -149,7 +150,7 @@ class FileHelpers:
|
|||||||
"percent": 0,
|
"percent": 0,
|
||||||
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
||||||
}
|
}
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(server_id)},
|
{"id": str(server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -194,7 +195,7 @@ class FileHelpers:
|
|||||||
"percent": percent,
|
"percent": percent,
|
||||||
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
||||||
}
|
}
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(server_id)},
|
{"id": str(server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -215,7 +216,7 @@ class FileHelpers:
|
|||||||
"percent": 0,
|
"percent": 0,
|
||||||
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
||||||
}
|
}
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(server_id)},
|
{"id": str(server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -274,7 +275,7 @@ class FileHelpers:
|
|||||||
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
"total_files": self.helper.human_readable_file_size(dir_bytes),
|
||||||
}
|
}
|
||||||
# send status results to page.
|
# send status results to page.
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(server_id)},
|
{"id": str(server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -325,3 +326,12 @@ class FileHelpers:
|
|||||||
else:
|
else:
|
||||||
return "false"
|
return "false"
|
||||||
return
|
return
|
||||||
|
|
||||||
|
def unzip_server(self, zip_path, user_id):
|
||||||
|
if Helpers.check_file_perms(zip_path):
|
||||||
|
temp_dir = tempfile.mkdtemp()
|
||||||
|
with zipfile.ZipFile(zip_path, "r") as zip_ref:
|
||||||
|
# extracts archive to temp directory
|
||||||
|
zip_ref.extractall(temp_dir)
|
||||||
|
if user_id:
|
||||||
|
return temp_dir
|
||||||
|
@ -29,7 +29,6 @@ from app.classes.shared.null_writer import NullWriter
|
|||||||
from app.classes.shared.console import Console
|
from app.classes.shared.console import Console
|
||||||
from app.classes.shared.installer import installer
|
from app.classes.shared.installer import installer
|
||||||
from app.classes.shared.translation import Translation
|
from app.classes.shared.translation import Translation
|
||||||
from app.classes.web.websocket_helper import WebSocketHelper
|
|
||||||
|
|
||||||
with redirect_stderr(NullWriter()):
|
with redirect_stderr(NullWriter()):
|
||||||
import psutil
|
import psutil
|
||||||
@ -78,7 +77,6 @@ class Helpers:
|
|||||||
self.passhasher = PasswordHasher()
|
self.passhasher = PasswordHasher()
|
||||||
self.exiting = False
|
self.exiting = False
|
||||||
|
|
||||||
self.websocket_helper = WebSocketHelper(self)
|
|
||||||
self.translation = Translation(self)
|
self.translation = Translation(self)
|
||||||
self.update_available = False
|
self.update_available = False
|
||||||
self.ignored_names = ["crafty_managed.txt", "db_stats"]
|
self.ignored_names = ["crafty_managed.txt", "db_stats"]
|
||||||
@ -579,20 +577,16 @@ class Helpers:
|
|||||||
|
|
||||||
return version_data
|
return version_data
|
||||||
|
|
||||||
@staticmethod
|
def get_announcements(self):
|
||||||
def get_announcements():
|
data = []
|
||||||
data = (
|
|
||||||
'[{"id":"1","date":"Unknown",'
|
|
||||||
'"title":"Error getting Announcements",'
|
|
||||||
'"desc":"Error getting Announcements","link":""}]'
|
|
||||||
)
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
response = requests.get("https://craftycontrol.com/notify.json", timeout=2)
|
response = requests.get("https://craftycontrol.com/notify", timeout=2)
|
||||||
data = json.loads(response.content)
|
data = json.loads(response.content)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.error(f"Failed to fetch notifications with error: {e}")
|
logger.error(f"Failed to fetch notifications with error: {e}")
|
||||||
|
|
||||||
|
if self.update_available:
|
||||||
|
data.append(self.update_available)
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def get_version_string(self):
|
def get_version_string(self):
|
||||||
@ -1092,87 +1086,6 @@ class Helpers:
|
|||||||
|
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def generate_tree(self, folder, output=""):
|
|
||||||
dir_list = []
|
|
||||||
unsorted_files = []
|
|
||||||
file_list = os.listdir(folder)
|
|
||||||
for item in file_list:
|
|
||||||
if os.path.isdir(os.path.join(folder, item)):
|
|
||||||
dir_list.append(item)
|
|
||||||
elif str(item) != self.ignored_names:
|
|
||||||
unsorted_files.append(item)
|
|
||||||
file_list = sorted(dir_list, key=str.casefold) + sorted(
|
|
||||||
unsorted_files, key=str.casefold
|
|
||||||
)
|
|
||||||
for raw_filename in file_list:
|
|
||||||
filename = html.escape(raw_filename)
|
|
||||||
rel = os.path.join(folder, raw_filename)
|
|
||||||
dpath = os.path.join(folder, filename)
|
|
||||||
if os.path.isdir(rel):
|
|
||||||
if filename not in self.ignored_names:
|
|
||||||
output += f"""<li id="{dpath}li" class="tree-item"
|
|
||||||
data-path="{dpath}">
|
|
||||||
\n<div id="{dpath}" data-path="{dpath}" data-name="{filename}"
|
|
||||||
class="tree-caret tree-ctx-item tree-folder">
|
|
||||||
<span id="{dpath}span" class="files-tree-title" data-path="{dpath}"
|
|
||||||
data-name="{filename}" onclick="getDirView(event)">
|
|
||||||
<i style="color: var(--info);" class="far fa-folder"></i>
|
|
||||||
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
|
||||||
{filename}
|
|
||||||
</span>
|
|
||||||
</div><li>
|
|
||||||
\n"""
|
|
||||||
else:
|
|
||||||
if filename not in self.ignored_names:
|
|
||||||
output += f"""<li id="{dpath}li"
|
|
||||||
class="d-block tree-ctx-item tree-file tree-item"
|
|
||||||
data-path="{dpath}"
|
|
||||||
data-name="{filename}"
|
|
||||||
onclick="clickOnFile(event)"><span style="margin-right: 6px;">
|
|
||||||
<i class="far fa-file"></i></span>{filename}</li>"""
|
|
||||||
return output
|
|
||||||
|
|
||||||
def generate_dir(self, folder, output=""):
|
|
||||||
dir_list = []
|
|
||||||
unsorted_files = []
|
|
||||||
file_list = os.listdir(folder)
|
|
||||||
for item in file_list:
|
|
||||||
if os.path.isdir(os.path.join(folder, item)):
|
|
||||||
dir_list.append(item)
|
|
||||||
elif str(item) != self.ignored_names:
|
|
||||||
unsorted_files.append(item)
|
|
||||||
file_list = sorted(dir_list, key=str.casefold) + sorted(
|
|
||||||
unsorted_files, key=str.casefold
|
|
||||||
)
|
|
||||||
output += f"""<ul class="tree-nested d-block" id="{folder}ul">"""
|
|
||||||
for raw_filename in file_list:
|
|
||||||
filename = html.escape(raw_filename)
|
|
||||||
dpath = os.path.join(folder, filename)
|
|
||||||
rel = os.path.join(folder, raw_filename)
|
|
||||||
if os.path.isdir(rel):
|
|
||||||
if filename not in self.ignored_names:
|
|
||||||
output += f"""<li id="{dpath}li" class="tree-item"
|
|
||||||
data-path="{dpath}">
|
|
||||||
\n<div id="{dpath}" data-path="{dpath}" data-name="{filename}"
|
|
||||||
class="tree-caret tree-ctx-item tree-folder">
|
|
||||||
<span id="{dpath}span" class="files-tree-title" data-path="{dpath}"
|
|
||||||
data-name="{filename}" onclick="getDirView(event)">
|
|
||||||
<i style="color: var(--info);" class="far fa-folder"></i>
|
|
||||||
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
|
||||||
{filename}
|
|
||||||
</span>
|
|
||||||
</div><li>"""
|
|
||||||
else:
|
|
||||||
if filename not in self.ignored_names:
|
|
||||||
output += f"""<li id="{dpath}li"
|
|
||||||
class="d-block tree-ctx-item tree-file tree-item"
|
|
||||||
data-path="{dpath}"
|
|
||||||
data-name="{filename}"
|
|
||||||
onclick="clickOnFile(event)"><span style="margin-right: 6px;">
|
|
||||||
<i class="far fa-file"></i></span>{filename}</li>"""
|
|
||||||
output += "</ul>\n"
|
|
||||||
return output
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def generate_zip_tree(folder, output=""):
|
def generate_zip_tree(folder, output=""):
|
||||||
file_list = os.listdir(folder)
|
file_list = os.listdir(folder)
|
||||||
@ -1216,23 +1129,6 @@ class Helpers:
|
|||||||
</input></div><li>"""
|
</input></div><li>"""
|
||||||
return output
|
return output
|
||||||
|
|
||||||
def unzip_server(self, zip_path, user_id):
|
|
||||||
if Helpers.check_file_perms(zip_path):
|
|
||||||
temp_dir = tempfile.mkdtemp()
|
|
||||||
with zipfile.ZipFile(zip_path, "r") as zip_ref:
|
|
||||||
# extracts archive to temp directory
|
|
||||||
zip_ref.extractall(temp_dir)
|
|
||||||
if user_id:
|
|
||||||
self.websocket_helper.broadcast_user(
|
|
||||||
user_id, "send_temp_path", {"path": temp_dir}
|
|
||||||
)
|
|
||||||
|
|
||||||
def backup_select(self, path, user_id):
|
|
||||||
if user_id:
|
|
||||||
self.websocket_helper.broadcast_user(
|
|
||||||
user_id, "send_temp_path", {"path": path}
|
|
||||||
)
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def unzip_backup_archive(backup_path, zip_name):
|
def unzip_backup_archive(backup_path, zip_name):
|
||||||
zip_path = os.path.join(backup_path, zip_name)
|
zip_path = os.path.join(backup_path, zip_name)
|
||||||
|
@ -9,6 +9,7 @@ from app.classes.controllers.server_perms_controller import PermissionsServers
|
|||||||
from app.classes.controllers.servers_controller import ServersController
|
from app.classes.controllers.servers_controller import ServersController
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.file_helpers import FileHelpers
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -64,7 +65,7 @@ class ImportHelpers:
|
|||||||
ServersController.finish_import(new_id)
|
ServersController.finish_import(new_id)
|
||||||
server_users = PermissionsServers.get_server_user_list(new_id)
|
server_users = PermissionsServers.get_server_user_list(new_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "send_start_reload", {})
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
|
|
||||||
def import_java_zip_server(self, temp_dir, new_server_dir, port, new_id):
|
def import_java_zip_server(self, temp_dir, new_server_dir, port, new_id):
|
||||||
import_thread = threading.Thread(
|
import_thread = threading.Thread(
|
||||||
@ -108,7 +109,7 @@ class ImportHelpers:
|
|||||||
server_users = PermissionsServers.get_server_user_list(new_id)
|
server_users = PermissionsServers.get_server_user_list(new_id)
|
||||||
ServersController.finish_import(new_id)
|
ServersController.finish_import(new_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "send_start_reload", {})
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
# deletes temp dir
|
# deletes temp dir
|
||||||
FileHelpers.del_dirs(temp_dir)
|
FileHelpers.del_dirs(temp_dir)
|
||||||
|
|
||||||
@ -162,7 +163,7 @@ class ImportHelpers:
|
|||||||
ServersController.finish_import(new_id)
|
ServersController.finish_import(new_id)
|
||||||
server_users = PermissionsServers.get_server_user_list(new_id)
|
server_users = PermissionsServers.get_server_user_list(new_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "send_start_reload", {})
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
|
|
||||||
def import_bedrock_zip_server(
|
def import_bedrock_zip_server(
|
||||||
self, temp_dir, new_server_dir, full_jar_path, port, new_id
|
self, temp_dir, new_server_dir, full_jar_path, port, new_id
|
||||||
@ -209,7 +210,7 @@ class ImportHelpers:
|
|||||||
ServersController.finish_import(new_id)
|
ServersController.finish_import(new_id)
|
||||||
server_users = PermissionsServers.get_server_user_list(new_id)
|
server_users = PermissionsServers.get_server_user_list(new_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "send_start_reload", {})
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
if os.name != "nt":
|
if os.name != "nt":
|
||||||
if Helpers.check_file_exists(full_jar_path):
|
if Helpers.check_file_exists(full_jar_path):
|
||||||
os.chmod(full_jar_path, 0o2760)
|
os.chmod(full_jar_path, 0o2760)
|
||||||
@ -253,4 +254,4 @@ class ImportHelpers:
|
|||||||
ServersController.finish_import(new_id)
|
ServersController.finish_import(new_id)
|
||||||
server_users = PermissionsServers.get_server_user_list(new_id)
|
server_users = PermissionsServers.get_server_user_list(new_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "send_start_reload", {})
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
|
@ -5,13 +5,14 @@ from datetime import datetime
|
|||||||
import platform
|
import platform
|
||||||
import shutil
|
import shutil
|
||||||
import time
|
import time
|
||||||
|
import json
|
||||||
import logging
|
import logging
|
||||||
import threading
|
import threading
|
||||||
|
from zoneinfo import ZoneInfoNotFoundError
|
||||||
from peewee import DoesNotExist
|
from peewee import DoesNotExist
|
||||||
|
|
||||||
# TZLocal is set as a hidden import on win pipeline
|
# TZLocal is set as a hidden import on win pipeline
|
||||||
from tzlocal import get_localzone
|
from tzlocal import get_localzone
|
||||||
from tzlocal.utils import ZoneInfoNotFoundError
|
|
||||||
from apscheduler.schedulers.background import BackgroundScheduler
|
from apscheduler.schedulers.background import BackgroundScheduler
|
||||||
|
|
||||||
from app.classes.models.server_permissions import EnumPermissionsServer
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
@ -32,6 +33,7 @@ from app.classes.shared.helpers import Helpers
|
|||||||
from app.classes.shared.file_helpers import FileHelpers
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
from app.classes.shared.import_helper import ImportHelpers
|
from app.classes.shared.import_helper import ImportHelpers
|
||||||
from app.classes.minecraft.serverjars import ServerJars
|
from app.classes.minecraft.serverjars import ServerJars
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -84,6 +86,17 @@ class Controller:
|
|||||||
def set_project_root(self, root_dir):
|
def set_project_root(self, root_dir):
|
||||||
self.project_root = root_dir
|
self.project_root = root_dir
|
||||||
|
|
||||||
|
def set_config_json(self, data):
|
||||||
|
current_config = self.helper.get_all_settings()
|
||||||
|
for key in current_config:
|
||||||
|
if key in data:
|
||||||
|
current_config[key] = data[key]
|
||||||
|
keys = list(current_config.keys())
|
||||||
|
keys.sort()
|
||||||
|
sorted_data = {i: current_config[i] for i in keys}
|
||||||
|
with open(self.helper.settings_file, "w", encoding="utf-8") as f:
|
||||||
|
json.dump(sorted_data, f, indent=4)
|
||||||
|
|
||||||
def package_support_logs(self, exec_user):
|
def package_support_logs(self, exec_user):
|
||||||
if exec_user["preparing"]:
|
if exec_user["preparing"]:
|
||||||
return
|
return
|
||||||
@ -101,7 +114,7 @@ class Controller:
|
|||||||
self.del_support_file(exec_user["support_logs"])
|
self.del_support_file(exec_user["support_logs"])
|
||||||
# pausing so on screen notifications can run for user
|
# pausing so on screen notifications can run for user
|
||||||
time.sleep(7)
|
time.sleep(7)
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
exec_user["user_id"], "notification", "Preparing your support logs"
|
exec_user["user_id"], "notification", "Preparing your support logs"
|
||||||
)
|
)
|
||||||
self.helper.ensure_dir_exists(
|
self.helper.ensure_dir_exists(
|
||||||
@ -197,17 +210,15 @@ class Controller:
|
|||||||
) as f:
|
) as f:
|
||||||
f.write(sys_info_string)
|
f.write(sys_info_string)
|
||||||
FileHelpers.make_compressed_archive(temp_zip_storage, temp_dir, sys_info_string)
|
FileHelpers.make_compressed_archive(temp_zip_storage, temp_dir, sys_info_string)
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
exec_user["user_id"],
|
exec_user["user_id"],
|
||||||
"support_status_update",
|
"support_status_update",
|
||||||
Helpers.calc_percent(temp_dir, temp_zip_storage + ".zip"),
|
Helpers.calc_percent(temp_dir, temp_zip_storage + ".zip"),
|
||||||
)
|
)
|
||||||
|
|
||||||
temp_zip_storage += ".zip"
|
temp_zip_storage += ".zip"
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(exec_user["user_id"], "send_logs_bootbox", {})
|
||||||
exec_user["user_id"], "send_logs_bootbox", {}
|
|
||||||
)
|
|
||||||
|
|
||||||
self.users.set_support_path(exec_user["user_id"], temp_zip_storage)
|
self.users.set_support_path(exec_user["user_id"], temp_zip_storage)
|
||||||
|
|
||||||
@ -240,8 +251,8 @@ class Controller:
|
|||||||
results = Helpers.calc_percent(source_path, dest_path)
|
results = Helpers.calc_percent(source_path, dest_path)
|
||||||
self.log_stats = results
|
self.log_stats = results
|
||||||
|
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
exec_user["user_id"], "support_status_update", results
|
exec_user["user_id"], "support_status_update", results
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -300,15 +311,6 @@ class Controller:
|
|||||||
Helpers.ensure_dir_exists(new_server_path)
|
Helpers.ensure_dir_exists(new_server_path)
|
||||||
Helpers.ensure_dir_exists(backup_path)
|
Helpers.ensure_dir_exists(backup_path)
|
||||||
|
|
||||||
def _copy_import_dir_files(existing_server_path):
|
|
||||||
existing_server_path = Helpers.get_os_understandable_path(
|
|
||||||
existing_server_path
|
|
||||||
)
|
|
||||||
try:
|
|
||||||
FileHelpers.copy_dir(existing_server_path, new_server_path, True)
|
|
||||||
except shutil.Error as ex:
|
|
||||||
logger.error(f"Server import failed with error: {ex}")
|
|
||||||
|
|
||||||
def _create_server_properties_if_needed(port, empty=False):
|
def _create_server_properties_if_needed(port, empty=False):
|
||||||
properties_file = os.path.join(new_server_path, "server.properties")
|
properties_file = os.path.join(new_server_path, "server.properties")
|
||||||
has_properties = os.path.exists(properties_file)
|
has_properties = os.path.exists(properties_file)
|
||||||
@ -336,22 +338,25 @@ class Controller:
|
|||||||
server_file = f"{create_data['type']}-{create_data['version']}.jar"
|
server_file = f"{create_data['type']}-{create_data['version']}.jar"
|
||||||
|
|
||||||
# Create an EULA file
|
# Create an EULA file
|
||||||
with open(
|
if "agree_to_eula" in create_data:
|
||||||
os.path.join(new_server_path, "eula.txt"), "w", encoding="utf-8"
|
with open(
|
||||||
) as file:
|
os.path.join(new_server_path, "eula.txt"), "w", encoding="utf-8"
|
||||||
file.write(
|
) as file:
|
||||||
"eula=" + ("true" if create_data["agree_to_eula"] else "false")
|
file.write(
|
||||||
)
|
"eula="
|
||||||
|
+ ("true" if create_data["agree_to_eula"] else "false")
|
||||||
|
)
|
||||||
elif root_create_data["create_type"] == "import_server":
|
elif root_create_data["create_type"] == "import_server":
|
||||||
_copy_import_dir_files(create_data["existing_server_path"])
|
|
||||||
server_file = create_data["jarfile"]
|
server_file = create_data["jarfile"]
|
||||||
elif root_create_data["create_type"] == "import_zip":
|
elif root_create_data["create_type"] == "import_zip":
|
||||||
# TODO: Copy files from the zip file to the new server directory
|
# TODO: Copy files from the zip file to the new server directory
|
||||||
server_file = create_data["jarfile"]
|
server_file = create_data["jarfile"]
|
||||||
raise NotImplementedError("Not yet implemented")
|
raise NotImplementedError("Not yet implemented")
|
||||||
_create_server_properties_if_needed(
|
# self.import_helper.import_java_zip_server()
|
||||||
create_data["server_properties_port"],
|
if data["create_type"] == "minecraft_java":
|
||||||
)
|
_create_server_properties_if_needed(
|
||||||
|
create_data["server_properties_port"],
|
||||||
|
)
|
||||||
|
|
||||||
min_mem = create_data["mem_min"]
|
min_mem = create_data["mem_min"]
|
||||||
max_mem = create_data["mem_max"]
|
max_mem = create_data["mem_max"]
|
||||||
@ -364,30 +369,72 @@ class Controller:
|
|||||||
def _wrap_jar_if_windows():
|
def _wrap_jar_if_windows():
|
||||||
return f'"{server_file}"' if Helpers.is_os_windows() else server_file
|
return f'"{server_file}"' if Helpers.is_os_windows() else server_file
|
||||||
|
|
||||||
server_command = (
|
if root_create_data["create_type"] == "download_jar":
|
||||||
f"java -Xms{_gibs_to_mibs(min_mem)}M "
|
if Helpers.is_os_windows():
|
||||||
f"-Xmx{_gibs_to_mibs(max_mem)}M "
|
# Let's check for and setup for install server commands
|
||||||
f"-jar {_wrap_jar_if_windows()} nogui"
|
if create_data["type"] == "forge":
|
||||||
)
|
server_command = (
|
||||||
|
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
||||||
|
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
||||||
|
f'-jar "{server_file}" --installServer'
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
server_command = (
|
||||||
|
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
||||||
|
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
||||||
|
f'-jar "{server_file}" nogui'
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if create_data["type"] == "forge":
|
||||||
|
server_command = (
|
||||||
|
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
||||||
|
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
||||||
|
f"-jar {server_file} --installServer"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
server_command = (
|
||||||
|
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
||||||
|
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
||||||
|
f"-jar {server_file} nogui"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
server_command = (
|
||||||
|
f"java -Xms{_gibs_to_mibs(min_mem)}M "
|
||||||
|
f"-Xmx{_gibs_to_mibs(max_mem)}M "
|
||||||
|
f"-jar {_wrap_jar_if_windows()} nogui"
|
||||||
|
)
|
||||||
|
|
||||||
elif data["create_type"] == "minecraft_bedrock":
|
elif data["create_type"] == "minecraft_bedrock":
|
||||||
if root_create_data["create_type"] == "import_server":
|
if root_create_data["create_type"] == "import_server":
|
||||||
existing_server_path = Helpers.get_os_understandable_path(
|
existing_server_path = Helpers.get_os_understandable_path(
|
||||||
create_data["existing_server_path"]
|
create_data["existing_server_path"]
|
||||||
)
|
)
|
||||||
try:
|
if Helpers.is_os_windows():
|
||||||
FileHelpers.copy_dir(existing_server_path, new_server_path, True)
|
server_command = (
|
||||||
except shutil.Error as ex:
|
f'"{os.path.join(new_server_path, create_data["executable"])}"'
|
||||||
logger.error(f"Server import failed with error: {ex}")
|
)
|
||||||
|
else:
|
||||||
|
server_command = f"./{create_data['executable']}"
|
||||||
|
logger.debug("command: " + server_command)
|
||||||
|
server_file = create_data["executable"]
|
||||||
elif root_create_data["create_type"] == "import_zip":
|
elif root_create_data["create_type"] == "import_zip":
|
||||||
# TODO: Copy files from the zip file to the new server directory
|
# TODO: Copy files from the zip file to the new server directory
|
||||||
raise NotImplementedError("Not yet implemented")
|
raise NotImplementedError("Not yet implemented")
|
||||||
|
else:
|
||||||
|
server_file = "bedrock_server"
|
||||||
|
if Helpers.is_os_windows():
|
||||||
|
# if this is windows we will override the linux bedrock server name.
|
||||||
|
server_file = "bedrock_server.exe"
|
||||||
|
|
||||||
|
full_jar_path = os.path.join(new_server_path, server_file)
|
||||||
|
|
||||||
|
if self.helper.is_os_windows():
|
||||||
|
server_command = f'"{full_jar_path}"'
|
||||||
|
else:
|
||||||
|
server_command = f"./{server_file}"
|
||||||
_create_server_properties_if_needed(0, True)
|
_create_server_properties_if_needed(0, True)
|
||||||
|
|
||||||
server_command = create_data["command"]
|
server_command = create_data.get("command", server_command)
|
||||||
server_file = (
|
|
||||||
"./bedrock_server" # HACK: This is a hack to make the server start
|
|
||||||
)
|
|
||||||
elif data["create_type"] == "custom":
|
elif data["create_type"] == "custom":
|
||||||
# TODO: working_directory, executable_update
|
# TODO: working_directory, executable_update
|
||||||
if root_create_data["create_type"] == "raw_exec":
|
if root_create_data["create_type"] == "raw_exec":
|
||||||
@ -451,131 +498,85 @@ class Controller:
|
|||||||
server_host=monitoring_host,
|
server_host=monitoring_host,
|
||||||
server_type=monitoring_type,
|
server_type=monitoring_type,
|
||||||
)
|
)
|
||||||
|
if data["create_type"] == "minecraft_java":
|
||||||
if (
|
if root_create_data["create_type"] == "download_jar":
|
||||||
data["create_type"] == "minecraft_java"
|
# modded update urls from server jars will only update the installer
|
||||||
and root_create_data["create_type"] == "download_jar"
|
if create_data["category"] != "modded":
|
||||||
):
|
server_obj = self.servers.get_server_obj(new_server_id)
|
||||||
# modded update urls from server jars will only update the installer
|
url = (
|
||||||
if create_data["category"] != "modded":
|
f"https://serverjars.com/api/fetchJar/{create_data['category']}"
|
||||||
server_obj = self.servers.get_server_obj(new_server_id)
|
f"/{create_data['type']}/{create_data['version']}"
|
||||||
url = (
|
)
|
||||||
f"https://serverjars.com/api/fetchJar/{create_data['category']}"
|
server_obj.executable_update_url = url
|
||||||
f"/{create_data['type']}/{create_data['version']}"
|
self.servers.update_server(server_obj)
|
||||||
|
self.server_jars.download_jar(
|
||||||
|
create_data["category"],
|
||||||
|
create_data["type"],
|
||||||
|
create_data["version"],
|
||||||
|
full_jar_path,
|
||||||
|
new_server_id,
|
||||||
)
|
)
|
||||||
server_obj.executable_update_url = url
|
elif root_create_data["create_type"] == "import_server":
|
||||||
self.servers.update_server(server_obj)
|
ServersController.set_import(new_server_id)
|
||||||
self.server_jars.download_jar(
|
self.import_helper.import_jar_server(
|
||||||
create_data["category"],
|
create_data["existing_server_path"],
|
||||||
create_data["type"],
|
new_server_path,
|
||||||
create_data["version"],
|
monitoring_port,
|
||||||
full_jar_path,
|
new_server_id,
|
||||||
new_server_id,
|
)
|
||||||
)
|
elif root_create_data["create_type"] == "import_zip":
|
||||||
|
ServersController.set_import(new_server_id)
|
||||||
|
|
||||||
|
elif data["create_type"] == "minecraft_bedrock":
|
||||||
|
if root_create_data["create_type"] == "download_exe":
|
||||||
|
ServersController.set_import(new_server_id)
|
||||||
|
self.import_helper.download_bedrock_server(
|
||||||
|
new_server_path, new_server_id
|
||||||
|
)
|
||||||
|
elif root_create_data["create_type"] == "import_server":
|
||||||
|
ServersController.set_import(new_server_id)
|
||||||
|
full_exe_path = os.path.join(new_server_path, create_data["executable"])
|
||||||
|
self.import_helper.import_bedrock_server(
|
||||||
|
create_data["existing_server_path"],
|
||||||
|
new_server_path,
|
||||||
|
monitoring_port,
|
||||||
|
full_exe_path,
|
||||||
|
new_server_id,
|
||||||
|
)
|
||||||
|
elif root_create_data["create_type"] == "import_zip":
|
||||||
|
ServersController.set_import(new_server_id)
|
||||||
|
full_exe_path = os.path.join(new_server_path, create_data["executable"])
|
||||||
|
self.import_helper.import_bedrock_zip_server(
|
||||||
|
create_data["zip_path"],
|
||||||
|
new_server_path,
|
||||||
|
os.path.join(create_data["zip_root"], create_data["executable"]),
|
||||||
|
monitoring_port,
|
||||||
|
new_server_id,
|
||||||
|
)
|
||||||
|
|
||||||
|
exec_user = self.users.get_user_by_id(int(user_id))
|
||||||
|
captured_roles = data.get("roles", [])
|
||||||
|
# These lines create a new Role for the Server with full permissions
|
||||||
|
# and add the user to it if he's not a superuser
|
||||||
|
if len(captured_roles) == 0:
|
||||||
|
if not exec_user["superuser"]:
|
||||||
|
new_server_uuid = self.servers.get_server_data_by_id(new_server_id).get(
|
||||||
|
"server_uuid"
|
||||||
|
)
|
||||||
|
role_id = self.roles.add_role(
|
||||||
|
f"Creator of Server with uuid={new_server_uuid}",
|
||||||
|
exec_user["user_id"],
|
||||||
|
)
|
||||||
|
self.server_perms.add_role_server(new_server_id, role_id, "11111111")
|
||||||
|
self.users.add_role_to_user(exec_user["user_id"], role_id)
|
||||||
|
|
||||||
|
else:
|
||||||
|
for role in captured_roles:
|
||||||
|
role_id = role
|
||||||
|
self.server_perms.add_role_server(new_server_id, role_id, "11111111")
|
||||||
|
|
||||||
return new_server_id, server_fs_uuid
|
return new_server_id, server_fs_uuid
|
||||||
|
|
||||||
def create_jar_server(
|
|
||||||
self,
|
|
||||||
jar: str,
|
|
||||||
server: str,
|
|
||||||
version: str,
|
|
||||||
name: str,
|
|
||||||
min_mem: int,
|
|
||||||
max_mem: int,
|
|
||||||
port: int,
|
|
||||||
user_id: int,
|
|
||||||
):
|
|
||||||
server_id = Helpers.create_uuid()
|
|
||||||
server_dir = os.path.join(self.helper.servers_dir, server_id)
|
|
||||||
backup_path = os.path.join(self.helper.backup_path, server_id)
|
|
||||||
if Helpers.is_os_windows():
|
|
||||||
server_dir = Helpers.wtol_path(server_dir)
|
|
||||||
backup_path = Helpers.wtol_path(backup_path)
|
|
||||||
server_dir.replace(" ", "^ ")
|
|
||||||
backup_path.replace(" ", "^ ")
|
|
||||||
|
|
||||||
server_file = f"{server}-{version}.jar"
|
|
||||||
|
|
||||||
# make the dir - perhaps a UUID?
|
|
||||||
Helpers.ensure_dir_exists(server_dir)
|
|
||||||
Helpers.ensure_dir_exists(backup_path)
|
|
||||||
|
|
||||||
try:
|
|
||||||
# do a eula.txt
|
|
||||||
with open(
|
|
||||||
os.path.join(server_dir, "eula.txt"), "w", encoding="utf-8"
|
|
||||||
) as file:
|
|
||||||
file.write("eula=false")
|
|
||||||
file.close()
|
|
||||||
|
|
||||||
# setup server.properties with the port
|
|
||||||
with open(
|
|
||||||
os.path.join(server_dir, "server.properties"), "w", encoding="utf-8"
|
|
||||||
) as file:
|
|
||||||
file.write(f"server-port={port}")
|
|
||||||
file.close()
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
logger.error(f"Unable to create required server files due to :{e}")
|
|
||||||
return False
|
|
||||||
|
|
||||||
if Helpers.is_os_windows():
|
|
||||||
# Let's check for and setup for install server commands
|
|
||||||
if server == "forge":
|
|
||||||
server_command = (
|
|
||||||
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
|
||||||
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
|
||||||
f'-jar "{server_file}" --installServer'
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
server_command = (
|
|
||||||
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
|
||||||
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
|
||||||
f'-jar "{server_file}" nogui'
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
if server == "forge":
|
|
||||||
server_command = (
|
|
||||||
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
|
||||||
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
|
||||||
f"-jar {server_file} --installServer"
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
server_command = (
|
|
||||||
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
|
||||||
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
|
||||||
f"-jar {server_file} nogui"
|
|
||||||
)
|
|
||||||
server_log_file = "./logs/latest.log"
|
|
||||||
server_stop = "stop"
|
|
||||||
|
|
||||||
new_id = self.register_server(
|
|
||||||
name,
|
|
||||||
server_id,
|
|
||||||
server_dir,
|
|
||||||
backup_path,
|
|
||||||
server_command,
|
|
||||||
server_file,
|
|
||||||
server_log_file,
|
|
||||||
server_stop,
|
|
||||||
port,
|
|
||||||
user_id,
|
|
||||||
server_type="minecraft-java",
|
|
||||||
)
|
|
||||||
# modded update urls from server jars will only update the installer
|
|
||||||
if jar != "modded":
|
|
||||||
server_obj = self.servers.get_server_obj(new_id)
|
|
||||||
url = f"https://serverjars.com/api/fetchJar/{jar}/{server}/{version}"
|
|
||||||
server_obj.executable_update_url = url
|
|
||||||
self.servers.update_server(server_obj)
|
|
||||||
# download the jar
|
|
||||||
self.server_jars.download_jar(
|
|
||||||
jar, server, version, os.path.join(server_dir, server_file), new_id
|
|
||||||
)
|
|
||||||
|
|
||||||
return new_id
|
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def verify_jar_server(server_path: str, server_jar: str):
|
def verify_jar_server(server_path: str, server_jar: str):
|
||||||
server_path = Helpers.get_os_understandable_path(server_path)
|
server_path = Helpers.get_os_understandable_path(server_path)
|
||||||
@ -593,64 +594,7 @@ class Controller:
|
|||||||
return False
|
return False
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def import_jar_server(
|
def restore_java_zip_server(
|
||||||
self,
|
|
||||||
server_name: str,
|
|
||||||
server_path: str,
|
|
||||||
server_jar: str,
|
|
||||||
min_mem: int,
|
|
||||||
max_mem: int,
|
|
||||||
port: int,
|
|
||||||
user_id: int,
|
|
||||||
):
|
|
||||||
server_id = Helpers.create_uuid()
|
|
||||||
new_server_dir = os.path.join(self.helper.servers_dir, server_id)
|
|
||||||
backup_path = os.path.join(self.helper.backup_path, server_id)
|
|
||||||
if Helpers.is_os_windows():
|
|
||||||
new_server_dir = Helpers.wtol_path(new_server_dir)
|
|
||||||
backup_path = Helpers.wtol_path(backup_path)
|
|
||||||
new_server_dir.replace(" ", "^ ")
|
|
||||||
backup_path.replace(" ", "^ ")
|
|
||||||
|
|
||||||
Helpers.ensure_dir_exists(new_server_dir)
|
|
||||||
Helpers.ensure_dir_exists(backup_path)
|
|
||||||
server_path = Helpers.get_os_understandable_path(server_path)
|
|
||||||
|
|
||||||
full_jar_path = os.path.join(new_server_dir, server_jar)
|
|
||||||
|
|
||||||
if Helpers.is_os_windows():
|
|
||||||
server_command = (
|
|
||||||
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
|
||||||
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
|
||||||
f'-jar "{full_jar_path}" nogui'
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
server_command = (
|
|
||||||
f"java -Xms{Helpers.float_to_string(min_mem)}M "
|
|
||||||
f"-Xmx{Helpers.float_to_string(max_mem)}M "
|
|
||||||
f"-jar {full_jar_path} nogui"
|
|
||||||
)
|
|
||||||
server_log_file = "./logs/latest.log"
|
|
||||||
server_stop = "stop"
|
|
||||||
|
|
||||||
new_id = self.register_server(
|
|
||||||
server_name,
|
|
||||||
server_id,
|
|
||||||
new_server_dir,
|
|
||||||
backup_path,
|
|
||||||
server_command,
|
|
||||||
server_jar,
|
|
||||||
server_log_file,
|
|
||||||
server_stop,
|
|
||||||
port,
|
|
||||||
user_id,
|
|
||||||
server_type="minecraft-java",
|
|
||||||
)
|
|
||||||
ServersController.set_import(new_id)
|
|
||||||
self.import_helper.import_jar_server(server_path, new_server_dir, port, new_id)
|
|
||||||
return new_id
|
|
||||||
|
|
||||||
def import_zip_server(
|
|
||||||
self,
|
self,
|
||||||
server_name: str,
|
server_name: str,
|
||||||
zip_path: str,
|
zip_path: str,
|
||||||
@ -807,7 +751,7 @@ class Controller:
|
|||||||
self.import_helper.download_bedrock_server(new_server_dir, new_id)
|
self.import_helper.download_bedrock_server(new_server_dir, new_id)
|
||||||
return new_id
|
return new_id
|
||||||
|
|
||||||
def import_bedrock_zip_server(
|
def restore_bedrock_zip_server(
|
||||||
self,
|
self,
|
||||||
server_name: str,
|
server_name: str,
|
||||||
zip_path: str,
|
zip_path: str,
|
||||||
@ -952,6 +896,7 @@ class Controller:
|
|||||||
|
|
||||||
srv_obj = server["server_obj"]
|
srv_obj = server["server_obj"]
|
||||||
srv_obj.server_scheduler.shutdown()
|
srv_obj.server_scheduler.shutdown()
|
||||||
|
srv_obj.dir_scheduler.shutdown()
|
||||||
running = srv_obj.check_running()
|
running = srv_obj.check_running()
|
||||||
|
|
||||||
if running:
|
if running:
|
||||||
@ -1025,7 +970,7 @@ class Controller:
|
|||||||
def t_update_master_server_dir(self, new_server_path, user_id):
|
def t_update_master_server_dir(self, new_server_path, user_id):
|
||||||
new_server_path = self.helper.wtol_path(new_server_path)
|
new_server_path = self.helper.wtol_path(new_server_path)
|
||||||
new_server_path = os.path.join(new_server_path, "servers")
|
new_server_path = os.path.join(new_server_path, "servers")
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/panel_config", "move_status", "Checking dir"
|
"/panel/panel_config", "move_status", "Checking dir"
|
||||||
)
|
)
|
||||||
current_master = self.helper.wtol_path(
|
current_master = self.helper.wtol_path(
|
||||||
@ -1035,7 +980,7 @@ class Controller:
|
|||||||
logger.info(
|
logger.info(
|
||||||
"Admin tried to change server dir to current server dir. Canceling..."
|
"Admin tried to change server dir to current server dir. Canceling..."
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/panel_config",
|
"/panel/panel_config",
|
||||||
"move_status",
|
"move_status",
|
||||||
"done",
|
"done",
|
||||||
@ -1046,18 +991,18 @@ class Controller:
|
|||||||
"Admin tried to change server dir to be inside a sub directory of the"
|
"Admin tried to change server dir to be inside a sub directory of the"
|
||||||
" current server dir. This will result in a copy loop."
|
" current server dir. This will result in a copy loop."
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/panel_config",
|
"/panel/panel_config",
|
||||||
"move_status",
|
"move_status",
|
||||||
"done",
|
"done",
|
||||||
)
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/panel_config", "move_status", "Checking permissions"
|
"/panel/panel_config", "move_status", "Checking permissions"
|
||||||
)
|
)
|
||||||
if not self.helper.ensure_dir_exists(new_server_path):
|
if not self.helper.ensure_dir_exists(new_server_path):
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -1066,6 +1011,8 @@ class Controller:
|
|||||||
"the new directory."
|
"the new directory."
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
self.helper.dir_migration = False
|
||||||
|
|
||||||
return
|
return
|
||||||
# set the cached serve dir
|
# set the cached serve dir
|
||||||
self.helper.servers_dir = new_server_path
|
self.helper.servers_dir = new_server_path
|
||||||
@ -1079,7 +1026,7 @@ class Controller:
|
|||||||
new_server_path, server.get("server_uuid")
|
new_server_path, server.get("server_uuid")
|
||||||
)
|
)
|
||||||
if os.path.isdir(server_path):
|
if os.path.isdir(server_path):
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/panel_config",
|
"/panel/panel_config",
|
||||||
"move_status",
|
"move_status",
|
||||||
f"Moving {server.get('server_name')}",
|
f"Moving {server.get('server_name')}",
|
||||||
@ -1120,7 +1067,7 @@ class Controller:
|
|||||||
self.servers.update_unloaded_server(server_obj)
|
self.servers.update_unloaded_server(server_obj)
|
||||||
self.servers.init_all_servers()
|
self.servers.init_all_servers()
|
||||||
self.helper.dir_migration = False
|
self.helper.dir_migration = False
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/panel_config",
|
"/panel/panel_config",
|
||||||
"move_status",
|
"move_status",
|
||||||
"done",
|
"done",
|
||||||
|
@ -16,22 +16,27 @@ import json
|
|||||||
from zoneinfo import ZoneInfo
|
from zoneinfo import ZoneInfo
|
||||||
|
|
||||||
# TZLocal is set as a hidden import on win pipeline
|
# TZLocal is set as a hidden import on win pipeline
|
||||||
|
from zoneinfo import ZoneInfoNotFoundError
|
||||||
from tzlocal import get_localzone
|
from tzlocal import get_localzone
|
||||||
from tzlocal.utils import ZoneInfoNotFoundError
|
|
||||||
from apscheduler.schedulers.background import BackgroundScheduler
|
from apscheduler.schedulers.background import BackgroundScheduler
|
||||||
from apscheduler.jobstores.base import JobLookupError
|
from apscheduler.jobstores.base import JobLookupError, ConflictingIdError
|
||||||
|
|
||||||
|
# OpenMetrics/Prometheus Imports
|
||||||
|
from prometheus_client import CollectorRegistry, Gauge, Info
|
||||||
|
|
||||||
from app.classes.minecraft.stats import Stats
|
from app.classes.minecraft.stats import Stats
|
||||||
from app.classes.minecraft.mc_ping import ping, ping_bedrock
|
from app.classes.minecraft.mc_ping import ping, ping_bedrock
|
||||||
from app.classes.models.servers import HelperServers, Servers
|
from app.classes.models.servers import HelperServers, Servers
|
||||||
from app.classes.models.server_stats import HelperServerStats
|
from app.classes.models.server_stats import HelperServerStats
|
||||||
from app.classes.models.management import HelpersManagement
|
from app.classes.models.management import HelpersManagement, HelpersWebhooks
|
||||||
from app.classes.models.users import HelperUsers
|
from app.classes.models.users import HelperUsers
|
||||||
from app.classes.models.server_permissions import PermissionsServers
|
from app.classes.models.server_permissions import PermissionsServers
|
||||||
from app.classes.shared.console import Console
|
from app.classes.shared.console import Console
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.file_helpers import FileHelpers
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
from app.classes.shared.null_writer import NullWriter
|
from app.classes.shared.null_writer import NullWriter
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
from app.classes.web.webhooks.webhook_factory import WebhookFactory
|
||||||
|
|
||||||
with redirect_stderr(NullWriter()):
|
with redirect_stderr(NullWriter()):
|
||||||
import psutil
|
import psutil
|
||||||
@ -40,6 +45,45 @@ with redirect_stderr(NullWriter()):
|
|||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
def callback(called_func):
|
||||||
|
# Usage of @callback on method
|
||||||
|
# definition to run a webhook check
|
||||||
|
# on method completion
|
||||||
|
def wrapper(*args, **kwargs):
|
||||||
|
res = None
|
||||||
|
logger.debug("Checking for callbacks")
|
||||||
|
try:
|
||||||
|
res = called_func(*args, **kwargs)
|
||||||
|
finally:
|
||||||
|
events = WebhookFactory.get_monitored_events()
|
||||||
|
if called_func.__name__ in events:
|
||||||
|
server_webhooks = HelpersWebhooks.get_webhooks_by_server(
|
||||||
|
args[0].server_id, True
|
||||||
|
)
|
||||||
|
for swebhook in server_webhooks:
|
||||||
|
if called_func.__name__ in str(swebhook.trigger).split(","):
|
||||||
|
logger.info(
|
||||||
|
f"Found callback for event {called_func.__name__}"
|
||||||
|
f" for server {args[0].server_id}"
|
||||||
|
)
|
||||||
|
webhook = HelpersWebhooks.get_webhook_by_id(swebhook.id)
|
||||||
|
webhook_provider = WebhookFactory.create_provider(
|
||||||
|
webhook["webhook_type"]
|
||||||
|
)
|
||||||
|
if res is not False and swebhook.enabled:
|
||||||
|
webhook_provider.send(
|
||||||
|
bot_name=webhook["bot_name"],
|
||||||
|
server_name=args[0].name,
|
||||||
|
title=webhook["name"],
|
||||||
|
url=webhook["url"],
|
||||||
|
message=webhook["body"],
|
||||||
|
color=webhook["color"],
|
||||||
|
)
|
||||||
|
return res
|
||||||
|
|
||||||
|
return wrapper
|
||||||
|
|
||||||
|
|
||||||
class ServerOutBuf:
|
class ServerOutBuf:
|
||||||
lines = {}
|
lines = {}
|
||||||
|
|
||||||
@ -92,12 +136,13 @@ class ServerOutBuf:
|
|||||||
|
|
||||||
# TODO: Do not send data to clients who do not have permission to view
|
# TODO: Do not send data to clients who do not have permission to view
|
||||||
# this server's console
|
# this server's console
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
if len(WebSocketManager().clients) > 0:
|
||||||
"/panel/server_detail",
|
WebSocketManager().broadcast_page_params(
|
||||||
{"id": self.server_id},
|
"/panel/server_detail",
|
||||||
"vterm_new_line",
|
{"id": self.server_id},
|
||||||
{"line": highlighted + "<br />"},
|
"vterm_new_line",
|
||||||
)
|
{"line": highlighted + "<br />"},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
# **********************************************************************************
|
# **********************************************************************************
|
||||||
@ -133,6 +178,8 @@ class ServerInstance:
|
|||||||
self.server_object = HelperServers.get_server_obj(self.server_id)
|
self.server_object = HelperServers.get_server_obj(self.server_id)
|
||||||
self.stats_helper = HelperServerStats(self.server_id)
|
self.stats_helper = HelperServerStats(self.server_id)
|
||||||
self.last_backup_failed = False
|
self.last_backup_failed = False
|
||||||
|
self.server_registry = CollectorRegistry()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
with open(
|
with open(
|
||||||
os.path.join(self.server_object.path, "db_stats", "players_cache.json"),
|
os.path.join(self.server_object.path, "db_stats", "players_cache.json"),
|
||||||
@ -152,6 +199,7 @@ class ServerInstance:
|
|||||||
self.tz = ZoneInfo("Europe/London")
|
self.tz = ZoneInfo("Europe/London")
|
||||||
self.server_scheduler = BackgroundScheduler(timezone=str(self.tz))
|
self.server_scheduler = BackgroundScheduler(timezone=str(self.tz))
|
||||||
self.dir_scheduler = BackgroundScheduler(timezone=str(self.tz))
|
self.dir_scheduler = BackgroundScheduler(timezone=str(self.tz))
|
||||||
|
self.init_registries()
|
||||||
self.server_scheduler.start()
|
self.server_scheduler.start()
|
||||||
self.dir_scheduler.start()
|
self.dir_scheduler.start()
|
||||||
self.start_dir_calc_task()
|
self.start_dir_calc_task()
|
||||||
@ -251,6 +299,23 @@ class ServerInstance:
|
|||||||
seconds=5,
|
seconds=5,
|
||||||
id="stats_" + str(self.server_id),
|
id="stats_" + str(self.server_id),
|
||||||
)
|
)
|
||||||
|
logger.info(f"Saving server statistics {self.name} every {30} seconds")
|
||||||
|
Console.info(f"Saving server statistics {self.name} every {30} seconds")
|
||||||
|
try:
|
||||||
|
self.server_scheduler.add_job(
|
||||||
|
self.record_server_stats,
|
||||||
|
"interval",
|
||||||
|
seconds=30,
|
||||||
|
id="save_stats_" + str(self.server_id),
|
||||||
|
)
|
||||||
|
except ConflictingIdError:
|
||||||
|
self.server_scheduler.remove_job("save_stats_" + str(self.server_id))
|
||||||
|
self.server_scheduler.add_job(
|
||||||
|
self.record_server_stats,
|
||||||
|
"interval",
|
||||||
|
seconds=30,
|
||||||
|
id="save_stats_" + str(self.server_id),
|
||||||
|
)
|
||||||
|
|
||||||
def setup_server_run_command(self):
|
def setup_server_run_command(self):
|
||||||
# configure the server
|
# configure the server
|
||||||
@ -313,6 +378,7 @@ class ServerInstance:
|
|||||||
logger.critical(f"Unable to write/access {self.server_path}")
|
logger.critical(f"Unable to write/access {self.server_path}")
|
||||||
Console.critical(f"Unable to write/access {self.server_path}")
|
Console.critical(f"Unable to write/access {self.server_path}")
|
||||||
|
|
||||||
|
@callback
|
||||||
def start_server(self, user_id, forge_install=False):
|
def start_server(self, user_id, forge_install=False):
|
||||||
if not user_id:
|
if not user_id:
|
||||||
user_lang = self.helper.get_setting("language")
|
user_lang = self.helper.get_setting("language")
|
||||||
@ -322,7 +388,7 @@ class ServerInstance:
|
|||||||
# Checks if user is currently attempting to move global server
|
# Checks if user is currently attempting to move global server
|
||||||
# dir
|
# dir
|
||||||
if self.helper.dir_migration:
|
if self.helper.dir_migration:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -337,7 +403,7 @@ class ServerInstance:
|
|||||||
|
|
||||||
if self.stats_helper.get_import_status() and not forge_install:
|
if self.stats_helper.get_import_status() and not forge_install:
|
||||||
if user_id:
|
if user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -383,7 +449,7 @@ class ServerInstance:
|
|||||||
e_flag = True
|
e_flag = True
|
||||||
if not e_flag and self.settings["type"] == "minecraft-java":
|
if not e_flag and self.settings["type"] == "minecraft-java":
|
||||||
if user_id:
|
if user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id, "send_eula_bootbox", {"id": self.server_id}
|
user_id, "send_eula_bootbox", {"id": self.server_id}
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
@ -416,7 +482,7 @@ class ServerInstance:
|
|||||||
|
|
||||||
except:
|
except:
|
||||||
if user_id:
|
if user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -452,7 +518,7 @@ class ServerInstance:
|
|||||||
f"Server {self.name} failed to start with error code: {ex}"
|
f"Server {self.name} failed to start with error code: {ex}"
|
||||||
)
|
)
|
||||||
if user_id:
|
if user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -479,7 +545,7 @@ class ServerInstance:
|
|||||||
# Checks for java on initial fail
|
# Checks for java on initial fail
|
||||||
if not self.helper.detect_java():
|
if not self.helper.detect_java():
|
||||||
if user_id:
|
if user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -493,7 +559,7 @@ class ServerInstance:
|
|||||||
f"Server {self.name} failed to start with error code: {ex}"
|
f"Server {self.name} failed to start with error code: {ex}"
|
||||||
)
|
)
|
||||||
if user_id:
|
if user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -540,7 +606,7 @@ class ServerInstance:
|
|||||||
self.stats_helper.set_first_run()
|
self.stats_helper.set_first_run()
|
||||||
loc_server_port = self.stats_helper.get_server_stats()["server_port"]
|
loc_server_port = self.stats_helper.get_server_stats()["server_port"]
|
||||||
# Sends port reminder message.
|
# Sends port reminder message.
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -552,15 +618,11 @@ class ServerInstance:
|
|||||||
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
if user != user_id:
|
if user != user_id:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
user, "send_start_reload", {}
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
user, "send_start_reload", {}
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
logger.warning(
|
logger.warning(
|
||||||
f"Server PID {self.process.pid} died right after starting "
|
f"Server PID {self.process.pid} died right after starting "
|
||||||
@ -592,7 +654,7 @@ class ServerInstance:
|
|||||||
def check_internet_thread(self, user_id, user_lang):
|
def check_internet_thread(self, user_id, user_lang):
|
||||||
if user_id:
|
if user_id:
|
||||||
if not Helpers.check_internet():
|
if not Helpers.check_internet():
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user_id,
|
user_id,
|
||||||
"send_start_error",
|
"send_start_error",
|
||||||
{
|
{
|
||||||
@ -719,9 +781,7 @@ class ServerInstance:
|
|||||||
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
||||||
|
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
user, "send_start_reload", {}
|
|
||||||
)
|
|
||||||
break
|
break
|
||||||
|
|
||||||
def stop_crash_detection(self):
|
def stop_crash_detection(self):
|
||||||
@ -762,6 +822,7 @@ class ServerInstance:
|
|||||||
if self.server_thread:
|
if self.server_thread:
|
||||||
self.server_thread.join()
|
self.server_thread.join()
|
||||||
|
|
||||||
|
@callback
|
||||||
def stop_server(self):
|
def stop_server(self):
|
||||||
running = self.check_running()
|
running = self.check_running()
|
||||||
if not running:
|
if not running:
|
||||||
@ -779,6 +840,7 @@ class ServerInstance:
|
|||||||
f"Assuming it was never started."
|
f"Assuming it was never started."
|
||||||
)
|
)
|
||||||
if self.settings["stop_command"]:
|
if self.settings["stop_command"]:
|
||||||
|
logger.info(f"Stop command requested for {self.settings['server_name']}.")
|
||||||
self.send_command(self.settings["stop_command"])
|
self.send_command(self.settings["stop_command"])
|
||||||
self.write_player_cache()
|
self.write_player_cache()
|
||||||
else:
|
else:
|
||||||
@ -834,7 +896,7 @@ class ServerInstance:
|
|||||||
self.record_server_stats()
|
self.record_server_stats()
|
||||||
|
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "send_start_reload", {})
|
WebSocketManager().broadcast_user(user, "send_start_reload", {})
|
||||||
|
|
||||||
def restart_threaded_server(self, user_id):
|
def restart_threaded_server(self, user_id):
|
||||||
bu_conf = HelpersManagement.get_backup_config(self.server_id)
|
bu_conf = HelpersManagement.get_backup_config(self.server_id)
|
||||||
@ -848,6 +910,9 @@ class ServerInstance:
|
|||||||
if not self.check_running():
|
if not self.check_running():
|
||||||
self.run_threaded_server(user_id)
|
self.run_threaded_server(user_id)
|
||||||
else:
|
else:
|
||||||
|
logger.info(
|
||||||
|
f"Restart command detected. Sending stop command to {self.server_id}."
|
||||||
|
)
|
||||||
self.stop_threaded_server()
|
self.stop_threaded_server()
|
||||||
time.sleep(2)
|
time.sleep(2)
|
||||||
self.run_threaded_server(user_id)
|
self.run_threaded_server(user_id)
|
||||||
@ -869,6 +934,7 @@ class ServerInstance:
|
|||||||
self.last_rc = poll
|
self.last_rc = poll
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
@callback
|
||||||
def send_command(self, command):
|
def send_command(self, command):
|
||||||
if not self.check_running() and command.lower() != "start":
|
if not self.check_running() and command.lower() != "start":
|
||||||
logger.warning(f'Server not running, unable to send command "{command}"')
|
logger.warning(f'Server not running, unable to send command "{command}"')
|
||||||
@ -881,6 +947,7 @@ class ServerInstance:
|
|||||||
self.process.stdin.flush()
|
self.process.stdin.flush()
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
@callback
|
||||||
def crash_detected(self, name):
|
def crash_detected(self, name):
|
||||||
# clear the old scheduled watcher task
|
# clear the old scheduled watcher task
|
||||||
self.server_scheduler.remove_job(f"c_{self.server_id}")
|
self.server_scheduler.remove_job(f"c_{self.server_id}")
|
||||||
@ -901,6 +968,7 @@ class ServerInstance:
|
|||||||
f"The server {name} has crashed and will be restarted. "
|
f"The server {name} has crashed and will be restarted. "
|
||||||
f"Restarting server"
|
f"Restarting server"
|
||||||
)
|
)
|
||||||
|
|
||||||
self.run_threaded_server(None)
|
self.run_threaded_server(None)
|
||||||
return True
|
return True
|
||||||
logger.critical(
|
logger.critical(
|
||||||
@ -913,6 +981,7 @@ class ServerInstance:
|
|||||||
)
|
)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
|
@callback
|
||||||
def kill(self):
|
def kill(self):
|
||||||
logger.info(f"Terminating server {self.server_id} and all child processes")
|
logger.info(f"Terminating server {self.server_id} and all child processes")
|
||||||
try:
|
try:
|
||||||
@ -1001,6 +1070,7 @@ class ServerInstance:
|
|||||||
f.write("eula=true")
|
f.write("eula=true")
|
||||||
self.run_threaded_server(user_id)
|
self.run_threaded_server(user_id)
|
||||||
|
|
||||||
|
@callback
|
||||||
def backup_server(self):
|
def backup_server(self):
|
||||||
if self.settings["backup_path"] == "":
|
if self.settings["backup_path"] == "":
|
||||||
logger.critical("Backup path is None. Canceling Backup!")
|
logger.critical("Backup path is None. Canceling Backup!")
|
||||||
@ -1034,18 +1104,11 @@ class ServerInstance:
|
|||||||
logger.info(f"Backup Thread started for server {self.settings['server_name']}.")
|
logger.info(f"Backup Thread started for server {self.settings['server_name']}.")
|
||||||
|
|
||||||
def a_backup_server(self):
|
def a_backup_server(self):
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
|
||||||
"/panel/server_detail",
|
|
||||||
{"id": str(self.server_id)},
|
|
||||||
"backup_reload",
|
|
||||||
{"percent": 0, "total_files": 0},
|
|
||||||
)
|
|
||||||
was_server_running = None
|
was_server_running = None
|
||||||
logger.info(f"Starting server {self.name} (ID {self.server_id}) backup")
|
logger.info(f"Starting server {self.name} (ID {self.server_id}) backup")
|
||||||
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user,
|
user,
|
||||||
"notification",
|
"notification",
|
||||||
self.helper.translation.translate(
|
self.helper.translation.translate(
|
||||||
@ -1120,8 +1183,8 @@ class ServerInstance:
|
|||||||
self.is_backingup = False
|
self.is_backingup = False
|
||||||
logger.info(f"Backup of server: {self.name} completed")
|
logger.info(f"Backup of server: {self.name} completed")
|
||||||
results = {"percent": 100, "total_files": 0, "current_file": 0}
|
results = {"percent": 100, "total_files": 0, "current_file": 0}
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(self.server_id)},
|
{"id": str(self.server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -1129,7 +1192,7 @@ class ServerInstance:
|
|||||||
)
|
)
|
||||||
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
server_users = PermissionsServers.get_server_user_list(self.server_id)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user,
|
user,
|
||||||
"notification",
|
"notification",
|
||||||
self.helper.translation.translate(
|
self.helper.translation.translate(
|
||||||
@ -1158,8 +1221,8 @@ class ServerInstance:
|
|||||||
f"Failed to create backup of server {self.name} (ID {self.server_id})"
|
f"Failed to create backup of server {self.name} (ID {self.server_id})"
|
||||||
)
|
)
|
||||||
results = {"percent": 100, "total_files": 0, "current_file": 0}
|
results = {"percent": 100, "total_files": 0, "current_file": 0}
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(self.server_id)},
|
{"id": str(self.server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -1176,8 +1239,8 @@ class ServerInstance:
|
|||||||
def backup_status(self, source_path, dest_path):
|
def backup_status(self, source_path, dest_path):
|
||||||
results = Helpers.calc_percent(source_path, dest_path)
|
results = Helpers.calc_percent(source_path, dest_path)
|
||||||
self.backup_stats = results
|
self.backup_stats = results
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(self.server_id)},
|
{"id": str(self.server_id)},
|
||||||
"backup_status",
|
"backup_status",
|
||||||
@ -1222,6 +1285,7 @@ class ServerInstance:
|
|||||||
if f["path"].endswith(".zip")
|
if f["path"].endswith(".zip")
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@callback
|
||||||
def jar_update(self):
|
def jar_update(self):
|
||||||
self.stats_helper.set_update(True)
|
self.stats_helper.set_update(True)
|
||||||
update_thread = threading.Thread(
|
update_thread = threading.Thread(
|
||||||
@ -1280,14 +1344,14 @@ class ServerInstance:
|
|||||||
self.stop_threaded_server()
|
self.stop_threaded_server()
|
||||||
else:
|
else:
|
||||||
was_started = False
|
was_started = False
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
# There are clients
|
# There are clients
|
||||||
self.check_update()
|
self.check_update()
|
||||||
message = (
|
message = (
|
||||||
'<a data-id="' + str(self.server_id) + '" class=""> UPDATING...</i></a>'
|
'<a data-id="' + str(self.server_id) + '" class=""> UPDATING...</i></a>'
|
||||||
)
|
)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user_page(
|
WebSocketManager().broadcast_user_page(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
user,
|
user,
|
||||||
"update_button_status",
|
"update_button_status",
|
||||||
@ -1340,7 +1404,7 @@ class ServerInstance:
|
|||||||
# check if backup was successful
|
# check if backup was successful
|
||||||
if self.last_backup_failed:
|
if self.last_backup_failed:
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user,
|
user,
|
||||||
"notification",
|
"notification",
|
||||||
"Backup failed for " + self.name + ". canceling update.",
|
"Backup failed for " + self.name + ". canceling update.",
|
||||||
@ -1386,11 +1450,11 @@ class ServerInstance:
|
|||||||
logger.info("Executable updated successfully. Starting Server")
|
logger.info("Executable updated successfully. Starting Server")
|
||||||
|
|
||||||
self.stats_helper.set_update(False)
|
self.stats_helper.set_update(False)
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
# There are clients
|
# There are clients
|
||||||
self.check_update()
|
self.check_update()
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user,
|
user,
|
||||||
"notification",
|
"notification",
|
||||||
"Executable update finished for " + self.name,
|
"Executable update finished for " + self.name,
|
||||||
@ -1398,7 +1462,7 @@ class ServerInstance:
|
|||||||
# sleep so first notif can completely run
|
# sleep so first notif can completely run
|
||||||
time.sleep(3)
|
time.sleep(3)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user_page(
|
WebSocketManager().broadcast_user_page(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
user,
|
user,
|
||||||
"update_button_status",
|
"update_button_status",
|
||||||
@ -1408,10 +1472,10 @@ class ServerInstance:
|
|||||||
"wasRunning": was_started,
|
"wasRunning": was_started,
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.broadcast_user_page(
|
WebSocketManager().broadcast_user_page(
|
||||||
user, "/panel/dashboard", "send_start_reload", {}
|
user, "/panel/dashboard", "send_start_reload", {}
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user,
|
user,
|
||||||
"notification",
|
"notification",
|
||||||
"Executable update finished for " + self.name,
|
"Executable update finished for " + self.name,
|
||||||
@ -1428,7 +1492,7 @@ class ServerInstance:
|
|||||||
self.run_threaded_server(HelperUsers.get_user_id_by_name("system"))
|
self.run_threaded_server(HelperUsers.get_user_id_by_name("system"))
|
||||||
else:
|
else:
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(
|
WebSocketManager().broadcast_user(
|
||||||
user,
|
user,
|
||||||
"notification",
|
"notification",
|
||||||
"Executable update failed for "
|
"Executable update failed for "
|
||||||
@ -1438,7 +1502,7 @@ class ServerInstance:
|
|||||||
logger.error("Executable download failed.")
|
logger.error("Executable download failed.")
|
||||||
self.stats_helper.set_update(False)
|
self.stats_helper.set_update(False)
|
||||||
for user in server_users:
|
for user in server_users:
|
||||||
self.helper.websocket_helper.broadcast_user(user, "remove_spinner", {})
|
WebSocketManager().broadcast_user(user, "remove_spinner", {})
|
||||||
|
|
||||||
def start_dir_calc_task(self):
|
def start_dir_calc_task(self):
|
||||||
server_dt = HelperServers.get_server_data_by_id(self.server_id)
|
server_dt = HelperServers.get_server_data_by_id(self.server_id)
|
||||||
@ -1467,7 +1531,7 @@ class ServerInstance:
|
|||||||
def realtime_stats(self):
|
def realtime_stats(self):
|
||||||
# only get stats if clients are connected.
|
# only get stats if clients are connected.
|
||||||
# no point in burning cpu
|
# no point in burning cpu
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
if len(WebSocketManager().clients) > 0:
|
||||||
total_players = 0
|
total_players = 0
|
||||||
max_players = 0
|
max_players = 0
|
||||||
servers_ping = []
|
servers_ping = []
|
||||||
@ -1498,50 +1562,43 @@ class ServerInstance:
|
|||||||
"crashed": self.is_crashed,
|
"crashed": self.is_crashed,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
|
||||||
self.helper.websocket_helper.broadcast_page_params(
|
WebSocketManager().broadcast_page_params(
|
||||||
"/panel/server_detail",
|
"/panel/server_detail",
|
||||||
{"id": str(self.server_id)},
|
{"id": str(self.server_id)},
|
||||||
"update_server_details",
|
"update_server_details",
|
||||||
{
|
{
|
||||||
"id": raw_ping_result.get("id"),
|
"id": raw_ping_result.get("id"),
|
||||||
"started": raw_ping_result.get("started"),
|
"started": raw_ping_result.get("started"),
|
||||||
"running": raw_ping_result.get("running"),
|
"running": raw_ping_result.get("running"),
|
||||||
"cpu": raw_ping_result.get("cpu"),
|
"cpu": raw_ping_result.get("cpu"),
|
||||||
"mem": raw_ping_result.get("mem"),
|
"mem": raw_ping_result.get("mem"),
|
||||||
"mem_percent": raw_ping_result.get("mem_percent"),
|
"mem_percent": raw_ping_result.get("mem_percent"),
|
||||||
"world_name": raw_ping_result.get("world_name"),
|
"world_name": raw_ping_result.get("world_name"),
|
||||||
"world_size": raw_ping_result.get("world_size"),
|
"world_size": raw_ping_result.get("world_size"),
|
||||||
"server_port": raw_ping_result.get("server_port"),
|
"server_port": raw_ping_result.get("server_port"),
|
||||||
"int_ping_results": raw_ping_result.get("int_ping_results"),
|
"int_ping_results": raw_ping_result.get("int_ping_results"),
|
||||||
"online": raw_ping_result.get("online"),
|
"online": raw_ping_result.get("online"),
|
||||||
"max": raw_ping_result.get("max"),
|
"max": raw_ping_result.get("max"),
|
||||||
"players": raw_ping_result.get("players"),
|
"players": raw_ping_result.get("players"),
|
||||||
"desc": raw_ping_result.get("desc"),
|
"desc": raw_ping_result.get("desc"),
|
||||||
"version": raw_ping_result.get("version"),
|
"version": raw_ping_result.get("version"),
|
||||||
"icon": raw_ping_result.get("icon"),
|
"icon": raw_ping_result.get("icon"),
|
||||||
"crashed": self.is_crashed,
|
"crashed": self.is_crashed,
|
||||||
"created": datetime.datetime.now().strftime(
|
"created": datetime.datetime.now().strftime("%Y/%m/%d, %H:%M:%S"),
|
||||||
"%Y/%m/%d, %H:%M:%S"
|
"players_cache": self.player_cache,
|
||||||
),
|
},
|
||||||
"players_cache": self.player_cache,
|
)
|
||||||
},
|
|
||||||
)
|
|
||||||
total_players += int(raw_ping_result.get("online"))
|
total_players += int(raw_ping_result.get("online"))
|
||||||
max_players += int(raw_ping_result.get("max"))
|
max_players += int(raw_ping_result.get("max"))
|
||||||
|
|
||||||
self.record_server_stats()
|
# self.record_server_stats()
|
||||||
|
|
||||||
if (len(servers_ping) > 0) & (
|
if len(servers_ping) > 0:
|
||||||
len(self.helper.websocket_helper.clients) > 0
|
|
||||||
):
|
|
||||||
try:
|
try:
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/dashboard", "update_server_status", servers_ping
|
"/panel/dashboard", "update_server_status", servers_ping
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.broadcast_page(
|
|
||||||
"/status", "update_server_status", servers_ping
|
|
||||||
)
|
|
||||||
except:
|
except:
|
||||||
Console.critical("Can't broadcast server status to websocket")
|
Console.critical("Can't broadcast server status to websocket")
|
||||||
|
|
||||||
@ -1560,7 +1617,6 @@ class ServerInstance:
|
|||||||
|
|
||||||
# process stats
|
# process stats
|
||||||
p_stats = Stats._try_get_process_stats(self.process, self.check_running())
|
p_stats = Stats._try_get_process_stats(self.process, self.check_running())
|
||||||
|
|
||||||
internal_ip = server["server_ip"]
|
internal_ip = server["server_ip"]
|
||||||
server_port = server["server_port"]
|
server_port = server["server_port"]
|
||||||
server_name = server.get("server_name", f"ID#{server_id}")
|
server_name = server.get("server_name", f"ID#{server_id}")
|
||||||
@ -1606,6 +1662,7 @@ class ServerInstance:
|
|||||||
"players": ping_data.get("players", False),
|
"players": ping_data.get("players", False),
|
||||||
"desc": ping_data.get("server_description", False),
|
"desc": ping_data.get("server_description", False),
|
||||||
"version": ping_data.get("server_version", False),
|
"version": ping_data.get("server_version", False),
|
||||||
|
"icon": ping_data.get("server_icon"),
|
||||||
}
|
}
|
||||||
else:
|
else:
|
||||||
server_stats = {
|
server_stats = {
|
||||||
@ -1624,6 +1681,7 @@ class ServerInstance:
|
|||||||
"players": False,
|
"players": False,
|
||||||
"desc": False,
|
"desc": False,
|
||||||
"version": False,
|
"version": False,
|
||||||
|
"icon": None,
|
||||||
}
|
}
|
||||||
|
|
||||||
return server_stats
|
return server_stats
|
||||||
@ -1631,7 +1689,7 @@ class ServerInstance:
|
|||||||
def get_server_players(self):
|
def get_server_players(self):
|
||||||
server = HelperServers.get_server_data_by_id(self.server_id)
|
server = HelperServers.get_server_data_by_id(self.server_id)
|
||||||
|
|
||||||
logger.info(f"Getting players for server {server}")
|
logger.debug(f"Getting players for server {server['server_name']}")
|
||||||
|
|
||||||
internal_ip = server["server_ip"]
|
internal_ip = server["server_ip"]
|
||||||
server_port = server["server_port"]
|
server_port = server["server_port"]
|
||||||
@ -1672,7 +1730,6 @@ class ServerInstance:
|
|||||||
}
|
}
|
||||||
|
|
||||||
server_stats = {}
|
server_stats = {}
|
||||||
server = HelperServers.get_server_obj(server_id)
|
|
||||||
if not server:
|
if not server:
|
||||||
return {}
|
return {}
|
||||||
server_dt = HelperServers.get_server_data_by_id(server_id)
|
server_dt = HelperServers.get_server_data_by_id(server_id)
|
||||||
@ -1799,9 +1856,50 @@ class ServerInstance:
|
|||||||
server_stats = self.get_servers_stats()
|
server_stats = self.get_servers_stats()
|
||||||
self.stats_helper.insert_server_stats(server_stats)
|
self.stats_helper.insert_server_stats(server_stats)
|
||||||
|
|
||||||
|
self.cpu_usage.labels(f"{self.server_id}").set(server_stats.get("cpu"))
|
||||||
|
self.mem_usage_percent.labels(f"{self.server_id}").set(
|
||||||
|
server_stats.get("mem_percent")
|
||||||
|
)
|
||||||
|
self.minecraft_version.labels(f"{self.server_id}").info(
|
||||||
|
{"version": f"{server_stats.get('version')}"}
|
||||||
|
)
|
||||||
|
self.online_players.labels(f"{self.server_id}").set(server_stats.get("online"))
|
||||||
|
|
||||||
# delete old data
|
# delete old data
|
||||||
max_age = self.helper.get_setting("history_max_age")
|
max_age = self.helper.get_setting("history_max_age")
|
||||||
now = datetime.datetime.now()
|
now = datetime.datetime.now()
|
||||||
minimum_to_exist = now - datetime.timedelta(days=max_age)
|
minimum_to_exist = now - datetime.timedelta(days=max_age)
|
||||||
|
|
||||||
self.stats_helper.remove_old_stats(minimum_to_exist)
|
self.stats_helper.remove_old_stats(minimum_to_exist)
|
||||||
|
|
||||||
|
def init_registries(self):
|
||||||
|
# REGISTRY Entries for Server Stats functions
|
||||||
|
self.cpu_usage = Gauge(
|
||||||
|
name="CPU_Usage",
|
||||||
|
documentation="The CPU usage of the server",
|
||||||
|
labelnames=["server_id"],
|
||||||
|
registry=self.server_registry,
|
||||||
|
)
|
||||||
|
self.mem_usage_percent = Gauge(
|
||||||
|
name="Mem_Usage",
|
||||||
|
documentation="The Memory usage of the server",
|
||||||
|
labelnames=["server_id"],
|
||||||
|
registry=self.server_registry,
|
||||||
|
)
|
||||||
|
self.minecraft_version = Info(
|
||||||
|
name="Minecraft_Version",
|
||||||
|
documentation="The version of the minecraft of this server",
|
||||||
|
labelnames=["server_id"],
|
||||||
|
registry=self.server_registry,
|
||||||
|
)
|
||||||
|
|
||||||
|
self.online_players = Gauge(
|
||||||
|
name="online_players",
|
||||||
|
documentation="The number of players online for a server",
|
||||||
|
labelnames=["server_id"],
|
||||||
|
registry=self.server_registry,
|
||||||
|
)
|
||||||
|
|
||||||
|
def get_server_history(self):
|
||||||
|
history = self.stats_helper.get_history_stats(self.server_id, 1)
|
||||||
|
return history
|
||||||
|
@ -5,10 +5,10 @@ import threading
|
|||||||
import asyncio
|
import asyncio
|
||||||
import datetime
|
import datetime
|
||||||
import json
|
import json
|
||||||
|
from zoneinfo import ZoneInfoNotFoundError
|
||||||
from tzlocal import get_localzone
|
from tzlocal import get_localzone
|
||||||
from tzlocal.utils import ZoneInfoNotFoundError
|
|
||||||
from apscheduler.events import EVENT_JOB_EXECUTED
|
from apscheduler.events import EVENT_JOB_EXECUTED
|
||||||
|
from apscheduler.jobstores.base import JobLookupError
|
||||||
from apscheduler.schedulers.background import BackgroundScheduler
|
from apscheduler.schedulers.background import BackgroundScheduler
|
||||||
from apscheduler.triggers.cron import CronTrigger
|
from apscheduler.triggers.cron import CronTrigger
|
||||||
|
|
||||||
@ -20,6 +20,7 @@ from app.classes.shared.file_helpers import FileHelpers
|
|||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.main_controller import Controller
|
from app.classes.shared.main_controller import Controller
|
||||||
from app.classes.web.tornado_handler import Webserver
|
from app.classes.web.tornado_handler import Webserver
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger("apscheduler")
|
logger = logging.getLogger("apscheduler")
|
||||||
scheduler_intervals = {
|
scheduler_intervals = {
|
||||||
@ -41,10 +42,10 @@ scheduler_intervals = {
|
|||||||
class TasksManager:
|
class TasksManager:
|
||||||
controller: Controller
|
controller: Controller
|
||||||
|
|
||||||
def __init__(self, helper, controller):
|
def __init__(self, helper, controller, file_helper):
|
||||||
self.helper: Helpers = helper
|
self.helper: Helpers = helper
|
||||||
self.controller: Controller = controller
|
self.controller: Controller = controller
|
||||||
self.tornado: Webserver = Webserver(helper, controller, self)
|
self.tornado: Webserver = Webserver(helper, controller, self, file_helper)
|
||||||
try:
|
try:
|
||||||
self.tz = get_localzone()
|
self.tz = get_localzone()
|
||||||
except ZoneInfoNotFoundError as e:
|
except ZoneInfoNotFoundError as e:
|
||||||
@ -101,7 +102,7 @@ class TasksManager:
|
|||||||
)
|
)
|
||||||
except:
|
except:
|
||||||
logger.error(
|
logger.error(
|
||||||
"Server value requested does not exist! "
|
f"Server value {cmd['server_id']} requested does not exist! "
|
||||||
"Purging item from waiting commands."
|
"Purging item from waiting commands."
|
||||||
)
|
)
|
||||||
continue
|
continue
|
||||||
@ -324,11 +325,16 @@ class TasksManager:
|
|||||||
|
|
||||||
# Checks to make sure some doofus didn't actually make the newly
|
# Checks to make sure some doofus didn't actually make the newly
|
||||||
# created task a child of itself.
|
# created task a child of itself.
|
||||||
if str(job_data["parent"]) == str(sch_id):
|
if (
|
||||||
|
str(job_data["parent"]) == str(sch_id)
|
||||||
|
or job_data["interval_type"] != "reaction"
|
||||||
|
):
|
||||||
HelpersManagement.update_scheduled_task(sch_id, {"parent": None})
|
HelpersManagement.update_scheduled_task(sch_id, {"parent": None})
|
||||||
|
|
||||||
# Check to see if it's enabled and is not a chain reaction.
|
# Check to see if it's enabled and is not a chain reaction.
|
||||||
if job_data["enabled"] and job_data["interval_type"] != "reaction":
|
if job_data["enabled"] and job_data["interval_type"] != "reaction":
|
||||||
|
# Lets make sure this can not be mistaken for a reaction
|
||||||
|
job_data["parent"] = None
|
||||||
new_job = "error"
|
new_job = "error"
|
||||||
if job_data["cron_string"] != "":
|
if job_data["cron_string"] != "":
|
||||||
try:
|
try:
|
||||||
@ -449,7 +455,8 @@ class TasksManager:
|
|||||||
def update_job(self, sch_id, job_data):
|
def update_job(self, sch_id, job_data):
|
||||||
# Checks to make sure some doofus didn't actually make the newly
|
# Checks to make sure some doofus didn't actually make the newly
|
||||||
# created task a child of itself.
|
# created task a child of itself.
|
||||||
if str(job_data.get("parent")) == str(sch_id):
|
interval_type = job_data.get("interval_type")
|
||||||
|
if str(job_data.get("parent")) == str(sch_id) or interval_type != "reaction":
|
||||||
job_data["parent"] = None
|
job_data["parent"] = None
|
||||||
HelpersManagement.update_scheduled_task(sch_id, job_data)
|
HelpersManagement.update_scheduled_task(sch_id, job_data)
|
||||||
|
|
||||||
@ -466,13 +473,15 @@ class TasksManager:
|
|||||||
job_data = HelpersManagement.get_scheduled_task(sch_id)
|
job_data = HelpersManagement.get_scheduled_task(sch_id)
|
||||||
job_data["server_id"] = job_data["server_id"]["server_id"]
|
job_data["server_id"] = job_data["server_id"]["server_id"]
|
||||||
else:
|
else:
|
||||||
self.scheduler.remove_job(str(sch_id))
|
job = HelpersManagement.get_scheduled_task(sch_id)
|
||||||
|
if job["interval_type"] != "reaction":
|
||||||
|
self.scheduler.remove_job(str(sch_id))
|
||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if job_data["interval"] != "reaction":
|
if job_data["interval"] != "reaction":
|
||||||
self.scheduler.remove_job(str(sch_id))
|
self.scheduler.remove_job(str(sch_id))
|
||||||
except:
|
except JobLookupError:
|
||||||
logger.info(
|
logger.info(
|
||||||
"No job found in update job. "
|
"No job found in update job. "
|
||||||
"Assuming it was previously disabled. Starting new job."
|
"Assuming it was previously disabled. Starting new job."
|
||||||
@ -608,7 +617,10 @@ class TasksManager:
|
|||||||
):
|
):
|
||||||
# event job ID's are strings so we need to look at
|
# event job ID's are strings so we need to look at
|
||||||
# this as the same data type.
|
# this as the same data type.
|
||||||
if str(schedule.parent) == str(event.job_id):
|
if (
|
||||||
|
str(schedule.parent) == str(event.job_id)
|
||||||
|
and schedule.interval_type == "reaction"
|
||||||
|
):
|
||||||
if schedule.enabled:
|
if schedule.enabled:
|
||||||
delaytime = datetime.datetime.now() + datetime.timedelta(
|
delaytime = datetime.datetime.now() + datetime.timedelta(
|
||||||
seconds=schedule.delay
|
seconds=schedule.delay
|
||||||
@ -688,10 +700,16 @@ class TasksManager:
|
|||||||
# Stats are different
|
# Stats are different
|
||||||
|
|
||||||
host_stats = HelpersManagement.get_latest_hosts_stats()
|
host_stats = HelpersManagement.get_latest_hosts_stats()
|
||||||
if len(self.helper.websocket_helper.clients) > 0:
|
|
||||||
|
self.controller.management.cpu_usage.set(host_stats.get("cpu_usage"))
|
||||||
|
self.controller.management.mem_usage_percent.set(
|
||||||
|
host_stats.get("mem_percent")
|
||||||
|
)
|
||||||
|
|
||||||
|
if len(WebSocketManager().clients) > 0:
|
||||||
# There are clients
|
# There are clients
|
||||||
try:
|
try:
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/dashboard",
|
"/panel/dashboard",
|
||||||
"update_host_stats",
|
"update_host_stats",
|
||||||
{
|
{
|
||||||
@ -708,7 +726,7 @@ class TasksManager:
|
|||||||
},
|
},
|
||||||
)
|
)
|
||||||
except:
|
except:
|
||||||
self.helper.websocket_helper.broadcast_page(
|
WebSocketManager().broadcast_page(
|
||||||
"/panel/dashboard",
|
"/panel/dashboard",
|
||||||
"update_host_stats",
|
"update_host_stats",
|
||||||
{
|
{
|
||||||
@ -726,12 +744,21 @@ class TasksManager:
|
|||||||
def check_for_updates(self):
|
def check_for_updates(self):
|
||||||
logger.info("Checking for Crafty updates...")
|
logger.info("Checking for Crafty updates...")
|
||||||
self.helper.update_available = self.helper.check_remote_version()
|
self.helper.update_available = self.helper.check_remote_version()
|
||||||
|
remote = self.helper.update_available
|
||||||
if self.helper.update_available:
|
if self.helper.update_available:
|
||||||
logger.info(f"Found new version {self.helper.update_available}")
|
logger.info(f"Found new version {self.helper.update_available}")
|
||||||
else:
|
else:
|
||||||
logger.info(
|
logger.info(
|
||||||
"No updates found! You are on the most up to date Crafty version."
|
"No updates found! You are on the most up to date Crafty version."
|
||||||
)
|
)
|
||||||
|
if self.helper.update_available:
|
||||||
|
self.helper.update_available = {
|
||||||
|
"id": str(remote),
|
||||||
|
"title": f"{remote} Update Available",
|
||||||
|
"date": "",
|
||||||
|
"desc": "Release notes are available by clicking this notification.",
|
||||||
|
"link": "https://gitlab.com/crafty-controller/crafty-4/-/releases",
|
||||||
|
}
|
||||||
logger.info("Refreshing Gravatar PFPs...")
|
logger.info("Refreshing Gravatar PFPs...")
|
||||||
for user in HelperUsers.get_all_users():
|
for user in HelperUsers.get_all_users():
|
||||||
if user.email:
|
if user.email:
|
||||||
@ -740,11 +767,13 @@ class TasksManager:
|
|||||||
)
|
)
|
||||||
# Search for old files in imports
|
# Search for old files in imports
|
||||||
self.helper.ensure_dir_exists(
|
self.helper.ensure_dir_exists(
|
||||||
os.path.join(self.controller.project_root, "imports")
|
os.path.join(self.controller.project_root, "import", "upload")
|
||||||
)
|
)
|
||||||
for file in os.listdir(os.path.join(self.controller.project_root, "imports")):
|
for file in os.listdir(
|
||||||
|
os.path.join(self.controller.project_root, "import", "upload")
|
||||||
|
):
|
||||||
if self.helper.is_file_older_than_x_days(
|
if self.helper.is_file_older_than_x_days(
|
||||||
os.path.join(self.controller.project_root, "imports", file)
|
os.path.join(self.controller.project_root, "import", "upload", file)
|
||||||
):
|
):
|
||||||
try:
|
try:
|
||||||
os.remove(os.path.join(file))
|
os.remove(os.path.join(file))
|
||||||
|
@ -1,26 +1,25 @@
|
|||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
|
from app.classes.shared.singleton import Singleton
|
||||||
from app.classes.shared.console import Console
|
from app.classes.shared.console import Console
|
||||||
|
from app.classes.models.users import HelperUsers
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class WebSocketHelper:
|
class WebSocketManager(metaclass=Singleton):
|
||||||
def __init__(self, helper):
|
def __init__(self):
|
||||||
self.helper = helper
|
|
||||||
self.clients = set()
|
self.clients = set()
|
||||||
|
|
||||||
def add_client(self, client):
|
def add_client(self, client):
|
||||||
self.clients.add(client)
|
self.clients.add(client)
|
||||||
|
|
||||||
def remove_client(self, client):
|
def remove_client(self, client):
|
||||||
self.clients.remove(client)
|
if client in self.clients:
|
||||||
|
self.clients.remove(client)
|
||||||
def send_message(self, client, event_type: str, data):
|
else:
|
||||||
if client.check_auth():
|
logger.exception("Error caught while removing unknown WebSocket client")
|
||||||
message = str(json.dumps({"event": event_type, "data": data}))
|
|
||||||
client.write_message_helper(message)
|
|
||||||
|
|
||||||
def broadcast(self, event_type: str, data):
|
def broadcast(self, event_type: str, data):
|
||||||
logger.debug(
|
logger.debug(
|
||||||
@ -29,13 +28,21 @@ class WebSocketHelper:
|
|||||||
)
|
)
|
||||||
for client in self.clients:
|
for client in self.clients:
|
||||||
try:
|
try:
|
||||||
self.send_message(client, event_type, data)
|
client.send_message(event_type, data)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.exception(
|
logger.exception(
|
||||||
f"Error caught while sending WebSocket message to "
|
f"Error caught while sending WebSocket message to "
|
||||||
f"{client.get_remote_ip()} {e}"
|
f"{client.get_remote_ip()} {e}"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def broadcast_to_admins(self, event_type: str, data):
|
||||||
|
def filter_fn(client):
|
||||||
|
if client.get_user_id in HelperUsers.get_super_user_list():
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
self.broadcast_with_fn(filter_fn, event_type, data)
|
||||||
|
|
||||||
def broadcast_page(self, page: str, event_type: str, data):
|
def broadcast_page(self, page: str, event_type: str, data):
|
||||||
def filter_fn(client):
|
def filter_fn(client):
|
||||||
return client.page == page
|
return client.page == page
|
||||||
@ -90,13 +97,14 @@ class WebSocketHelper:
|
|||||||
static_clients = self.clients
|
static_clients = self.clients
|
||||||
clients = list(filter(filter_fn, static_clients))
|
clients = list(filter(filter_fn, static_clients))
|
||||||
logger.debug(
|
logger.debug(
|
||||||
f"Sending to {len(clients)} out of {len(self.clients)} "
|
f"Sending to {len(clients)} \
|
||||||
|
out of {len(self.clients)} "
|
||||||
f"clients: {json.dumps({'event': event_type, 'data': data})}"
|
f"clients: {json.dumps({'event': event_type, 'data': data})}"
|
||||||
)
|
)
|
||||||
|
|
||||||
for client in clients[:]:
|
for client in clients[:]:
|
||||||
try:
|
try:
|
||||||
self.send_message(client, event_type, data)
|
client.send_message(event_type, data)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logger.exception(
|
logger.exception(
|
||||||
f"Error catched while sending WebSocket message to "
|
f"Error catched while sending WebSocket message to "
|
@ -1,698 +0,0 @@
|
|||||||
import os
|
|
||||||
import html
|
|
||||||
import pathlib
|
|
||||||
import re
|
|
||||||
import logging
|
|
||||||
import time
|
|
||||||
import urllib.parse
|
|
||||||
import bleach
|
|
||||||
import tornado.web
|
|
||||||
import tornado.escape
|
|
||||||
|
|
||||||
from app.classes.models.server_permissions import EnumPermissionsServer
|
|
||||||
from app.classes.shared.console import Console
|
|
||||||
from app.classes.shared.helpers import Helpers
|
|
||||||
from app.classes.shared.server import ServerOutBuf
|
|
||||||
from app.classes.web.base_handler import BaseHandler
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class AjaxHandler(BaseHandler):
|
|
||||||
def render_page(self, template, page_data):
|
|
||||||
self.render(
|
|
||||||
template,
|
|
||||||
data=page_data,
|
|
||||||
translate=self.translator.translate,
|
|
||||||
)
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def get(self, page):
|
|
||||||
_, _, exec_user = self.current_user
|
|
||||||
error = bleach.clean(self.get_argument("error", "WTF Error!"))
|
|
||||||
|
|
||||||
template = "panel/denied.html"
|
|
||||||
|
|
||||||
page_data = {"user_data": exec_user, "error": error}
|
|
||||||
|
|
||||||
if page == "error":
|
|
||||||
template = "public/error.html"
|
|
||||||
self.render_page(template, page_data)
|
|
||||||
|
|
||||||
elif page == "server_log":
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
full_log = self.get_argument("full", False)
|
|
||||||
|
|
||||||
if server_id is None:
|
|
||||||
logger.warning("Server ID not found in server_log ajax call")
|
|
||||||
self.redirect("/panel/error?error=Server ID Not Found")
|
|
||||||
return
|
|
||||||
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
server_data = self.controller.servers.get_server_data_by_id(server_id)
|
|
||||||
if not server_data:
|
|
||||||
logger.warning("Server Data not found in server_log ajax call")
|
|
||||||
self.redirect("/panel/error?error=Server ID Not Found")
|
|
||||||
return
|
|
||||||
|
|
||||||
if not server_data["log_path"]:
|
|
||||||
logger.warning(
|
|
||||||
f"Log path not found in server_log ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
|
|
||||||
if full_log:
|
|
||||||
log_lines = self.helper.get_setting("max_log_lines")
|
|
||||||
data = Helpers.tail_file(
|
|
||||||
# If the log path is absolute it returns it as is
|
|
||||||
# If it is relative it joins the paths below like normal
|
|
||||||
pathlib.Path(server_data["path"], server_data["log_path"]),
|
|
||||||
log_lines,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
data = ServerOutBuf.lines.get(server_id, [])
|
|
||||||
|
|
||||||
for line in data:
|
|
||||||
try:
|
|
||||||
line = re.sub("(\033\\[(0;)?[0-9]*[A-z]?(;[0-9])?m?)", "", line)
|
|
||||||
line = re.sub("[A-z]{2}\b\b", "", line)
|
|
||||||
line = self.helper.log_colors(html.escape(line))
|
|
||||||
self.write(f"<span class='box'>{line}<br /></span>")
|
|
||||||
# self.write(d.encode("utf-8"))
|
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
logger.warning(f"Skipping Log Line due to error: {e}")
|
|
||||||
|
|
||||||
elif page == "announcements":
|
|
||||||
data = Helpers.get_announcements()
|
|
||||||
page_data["notify_data"] = data
|
|
||||||
self.render_page("ajax/notify.html", page_data)
|
|
||||||
|
|
||||||
elif page == "get_zip_tree":
|
|
||||||
path = self.get_argument("path", None)
|
|
||||||
|
|
||||||
self.write(
|
|
||||||
Helpers.get_os_understandable_path(path)
|
|
||||||
+ "\n"
|
|
||||||
+ Helpers.generate_zip_tree(path)
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "get_zip_dir":
|
|
||||||
path = self.get_argument("path", None)
|
|
||||||
|
|
||||||
self.write(
|
|
||||||
Helpers.get_os_understandable_path(path)
|
|
||||||
+ "\n"
|
|
||||||
+ Helpers.generate_zip_dir(path)
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "get_backup_tree":
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
folder = self.get_argument("path", None)
|
|
||||||
|
|
||||||
output = ""
|
|
||||||
|
|
||||||
dir_list = []
|
|
||||||
unsorted_files = []
|
|
||||||
file_list = os.listdir(folder)
|
|
||||||
for item in file_list:
|
|
||||||
if os.path.isdir(os.path.join(folder, item)):
|
|
||||||
dir_list.append(item)
|
|
||||||
else:
|
|
||||||
unsorted_files.append(item)
|
|
||||||
file_list = sorted(dir_list, key=str.casefold) + sorted(
|
|
||||||
unsorted_files, key=str.casefold
|
|
||||||
)
|
|
||||||
output += f"""<ul class="tree-nested d-block" id="{folder}ul">"""
|
|
||||||
for raw_filename in file_list:
|
|
||||||
filename = html.escape(raw_filename)
|
|
||||||
rel = os.path.join(folder, raw_filename)
|
|
||||||
dpath = os.path.join(folder, filename)
|
|
||||||
if str(dpath) in self.controller.management.get_excluded_backup_dirs(
|
|
||||||
server_id
|
|
||||||
):
|
|
||||||
if os.path.isdir(rel):
|
|
||||||
output += f"""<li class="tree-item" data-path="{dpath}">
|
|
||||||
\n<div id="{dpath}" data-path="{dpath}" data-name="{filename}" class="tree-caret tree-ctx-item tree-folder">
|
|
||||||
<input type="checkbox" class="checkBoxClass" name="root_path" value="{dpath}" checked>
|
|
||||||
<span id="{dpath}span" class="files-tree-title" data-path="{dpath}" data-name="{filename}" onclick="getDirView(event)">
|
|
||||||
<i style="color: var(--info);" class="far fa-folder"></i>
|
|
||||||
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
|
||||||
<strong>{filename}</strong>
|
|
||||||
</span>
|
|
||||||
</input></div><li>
|
|
||||||
\n"""
|
|
||||||
else:
|
|
||||||
output += f"""<li
|
|
||||||
class="d-block tree-ctx-item tree-file"
|
|
||||||
data-path="{dpath}"
|
|
||||||
data-name="{filename}"
|
|
||||||
onclick=""><input type='checkbox' class="checkBoxClass" name='root_path' value="{dpath}" checked><span style="margin-right: 6px;">
|
|
||||||
<i class="far fa-file"></i></span></input>{filename}</li>"""
|
|
||||||
|
|
||||||
else:
|
|
||||||
if os.path.isdir(rel):
|
|
||||||
output += f"""<li class="tree-item" data-path="{dpath}">
|
|
||||||
\n<div id="{dpath}" data-path="{dpath}" data-name="{filename}" class="tree-caret tree-ctx-item tree-folder">
|
|
||||||
<input type="checkbox" class="checkBoxClass" name="root_path" value="{dpath}">
|
|
||||||
<span id="{dpath}span" class="files-tree-title" data-path="{dpath}" data-name="{filename}" onclick="getDirView(event)">
|
|
||||||
<i style="color: var(--info);" class="far fa-folder"></i>
|
|
||||||
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
|
||||||
<strong>{filename}</strong>
|
|
||||||
</span>
|
|
||||||
</input></div><li>
|
|
||||||
\n"""
|
|
||||||
else:
|
|
||||||
output += f"""<li
|
|
||||||
class="d-block tree-ctx-item tree-file"
|
|
||||||
data-path="{dpath}"
|
|
||||||
data-name="{filename}"
|
|
||||||
onclick=""><input type='checkbox' class="checkBoxClass" name='root_path' value="{dpath}">
|
|
||||||
<span style="margin-right: 6px;"><i class="far fa-file">
|
|
||||||
</i></span></input>{filename}</li>"""
|
|
||||||
self.write(Helpers.get_os_understandable_path(folder) + "\n" + output)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "get_backup_dir":
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
folder = self.get_argument("path", None)
|
|
||||||
output = ""
|
|
||||||
|
|
||||||
dir_list = []
|
|
||||||
unsorted_files = []
|
|
||||||
file_list = os.listdir(folder)
|
|
||||||
for item in file_list:
|
|
||||||
if os.path.isdir(os.path.join(folder, item)):
|
|
||||||
dir_list.append(item)
|
|
||||||
else:
|
|
||||||
unsorted_files.append(item)
|
|
||||||
file_list = sorted(dir_list, key=str.casefold) + sorted(
|
|
||||||
unsorted_files, key=str.casefold
|
|
||||||
)
|
|
||||||
output += f"""<ul class="tree-nested d-block" id="{folder}ul">"""
|
|
||||||
for raw_filename in file_list:
|
|
||||||
filename = html.escape(raw_filename)
|
|
||||||
rel = os.path.join(folder, raw_filename)
|
|
||||||
dpath = os.path.join(folder, filename)
|
|
||||||
if str(dpath) in self.controller.management.get_excluded_backup_dirs(
|
|
||||||
server_id
|
|
||||||
):
|
|
||||||
if os.path.isdir(rel):
|
|
||||||
output += f"""<li class="tree-item" data-path="{dpath}">
|
|
||||||
\n<div id="{dpath}" data-path="{dpath}" data-name="{filename}" class="tree-caret tree-ctx-item tree-folder">
|
|
||||||
<input type="checkbox" name="root_path" value="{dpath}" checked>
|
|
||||||
<span id="{dpath}span" class="files-tree-title" data-path="{dpath}" data-name="{filename}" onclick="getDirView(event)">
|
|
||||||
<i class="far fa-folder"></i>
|
|
||||||
<i class="far fa-folder-open"></i>
|
|
||||||
<strong>{filename}</strong>
|
|
||||||
</span>
|
|
||||||
</input></div><li>"""
|
|
||||||
else:
|
|
||||||
output += f"""<li
|
|
||||||
class="tree-item tree-nested d-block tree-ctx-item tree-file"
|
|
||||||
data-path="{dpath}"
|
|
||||||
data-name="{filename}"
|
|
||||||
onclick=""><input type='checkbox' name='root_path' value='{dpath}' checked><span style="margin-right: 6px;">
|
|
||||||
<i class="far fa-file"></i></span></input>{filename}</li>"""
|
|
||||||
|
|
||||||
else:
|
|
||||||
if os.path.isdir(rel):
|
|
||||||
output += f"""<li class="tree-item" data-path="{dpath}">
|
|
||||||
\n<div id="{dpath}" data-path="{dpath}" data-name="{filename}" class="tree-caret tree-ctx-item tree-folder">
|
|
||||||
<input type="checkbox" name="root_path" value="{dpath}">
|
|
||||||
<span id="{dpath}span" class="files-tree-title" data-path="{dpath}" data-name="{filename}" onclick="getDirView(event)">
|
|
||||||
<i class="far fa-folder"></i>
|
|
||||||
<i class="far fa-folder-open"></i>
|
|
||||||
<strong>{filename}</strong>
|
|
||||||
</span>
|
|
||||||
</input></div><li>"""
|
|
||||||
else:
|
|
||||||
output += f"""<li
|
|
||||||
class="tree-item tree-nested d-block tree-ctx-item tree-file"
|
|
||||||
data-path="{dpath}"
|
|
||||||
data-name="{filename}"
|
|
||||||
onclick=""><input type='checkbox' name='root_path' value='{dpath}'>
|
|
||||||
<span style="margin-right: 6px;"><i class="far fa-file">
|
|
||||||
</i></span></input>{filename}</li>"""
|
|
||||||
|
|
||||||
self.write(Helpers.get_os_understandable_path(folder) + "\n" + output)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "get_dir":
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
path = self.get_argument("path", None)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "get_tree"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if Helpers.validate_traversal(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"], path
|
|
||||||
):
|
|
||||||
self.write(
|
|
||||||
Helpers.get_os_understandable_path(path)
|
|
||||||
+ "\n"
|
|
||||||
+ Helpers.generate_dir(path)
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def post(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
|
|
||||||
if page == "send_command":
|
|
||||||
command = self.get_body_argument("command", default=None, strip=True)
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
if server_id is None:
|
|
||||||
logger.warning("Server ID not found in send_command ajax call")
|
|
||||||
Console.warning("Server ID not found in send_command ajax call")
|
|
||||||
|
|
||||||
svr_obj = self.controller.servers.get_server_instance_by_id(server_id)
|
|
||||||
|
|
||||||
if command == svr_obj.settings["stop_command"]:
|
|
||||||
logger.info(
|
|
||||||
"Stop command detected as terminal input - intercepting."
|
|
||||||
+ f"Starting Crafty's stop process for server with id: {server_id}"
|
|
||||||
)
|
|
||||||
self.controller.management.send_command(
|
|
||||||
exec_user["user_id"], server_id, self.get_remote_ip(), "stop_server"
|
|
||||||
)
|
|
||||||
command = None
|
|
||||||
elif command == "restart":
|
|
||||||
logger.info(
|
|
||||||
"Restart command detected as terminal input - intercepting."
|
|
||||||
+ f"Starting Crafty's stop process for server with id: {server_id}"
|
|
||||||
)
|
|
||||||
self.controller.management.send_command(
|
|
||||||
exec_user["user_id"],
|
|
||||||
server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
"restart_server",
|
|
||||||
)
|
|
||||||
command = None
|
|
||||||
if command:
|
|
||||||
if svr_obj.check_running():
|
|
||||||
svr_obj.send_command(command)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Sent command to "
|
|
||||||
f"{self.controller.servers.get_server_friendly_name(server_id)} "
|
|
||||||
f"terminal: {command}",
|
|
||||||
server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
|
|
||||||
elif page == "send_order":
|
|
||||||
self.controller.users.update_server_order(
|
|
||||||
exec_user["user_id"], bleach.clean(self.get_argument("order"))
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
elif page == "backup_now":
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
if server_id is None:
|
|
||||||
logger.error("Server ID is none. Canceling backup!")
|
|
||||||
return
|
|
||||||
|
|
||||||
server = self.controller.servers.get_server_instance_by_id(server_id)
|
|
||||||
self.controller.management.add_to_audit_log_raw(
|
|
||||||
self.controller.users.get_user_by_id(exec_user["user_id"])["username"],
|
|
||||||
exec_user["user_id"],
|
|
||||||
server_id,
|
|
||||||
f"Backup now executed for server {server_id} ",
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
|
|
||||||
server.backup_server()
|
|
||||||
|
|
||||||
elif page == "select_photo":
|
|
||||||
if exec_user["superuser"]:
|
|
||||||
photo = urllib.parse.unquote(self.get_argument("photo", ""))
|
|
||||||
opacity = self.get_argument("opacity", 100)
|
|
||||||
self.controller.management.set_login_opacity(int(opacity))
|
|
||||||
if photo == "login_1.jpg":
|
|
||||||
self.controller.management.set_login_image("login_1.jpg")
|
|
||||||
self.controller.cached_login = f"{photo}"
|
|
||||||
else:
|
|
||||||
self.controller.management.set_login_image(f"custom/{photo}")
|
|
||||||
self.controller.cached_login = f"custom/{photo}"
|
|
||||||
return
|
|
||||||
|
|
||||||
elif page == "delete_photo":
|
|
||||||
if exec_user["superuser"]:
|
|
||||||
photo = urllib.parse.unquote(self.get_argument("photo", None))
|
|
||||||
if photo and photo != "login_1.jpg":
|
|
||||||
os.remove(
|
|
||||||
os.path.join(
|
|
||||||
self.controller.project_root,
|
|
||||||
f"app/frontend/static/assets/images/auth/custom/{photo}",
|
|
||||||
)
|
|
||||||
)
|
|
||||||
current = self.controller.cached_login
|
|
||||||
split = current.split("/")
|
|
||||||
if len(split) == 1:
|
|
||||||
current_photo = current
|
|
||||||
else:
|
|
||||||
current_photo = split[1]
|
|
||||||
if current_photo == photo:
|
|
||||||
self.controller.management.set_login_image("login_1.jpg")
|
|
||||||
self.controller.cached_login = "login_1.jpg"
|
|
||||||
return
|
|
||||||
|
|
||||||
elif page == "eula":
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
svr = self.controller.servers.get_server_instance_by_id(server_id)
|
|
||||||
svr.agree_eula(exec_user["user_id"])
|
|
||||||
|
|
||||||
elif page == "restore_backup":
|
|
||||||
if not permissions["Backup"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Backups")
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(self.get_argument("id", None))
|
|
||||||
zip_name = bleach.clean(self.get_argument("zip_file", None))
|
|
||||||
svr_obj = self.controller.servers.get_server_obj(server_id)
|
|
||||||
server_data = self.controller.servers.get_server_data_by_id(server_id)
|
|
||||||
|
|
||||||
# import the server again based on zipfile
|
|
||||||
if server_data["type"] == "minecraft-java":
|
|
||||||
backup_path = svr_obj.backup_path
|
|
||||||
if Helpers.validate_traversal(backup_path, zip_name):
|
|
||||||
temp_dir = Helpers.unzip_backup_archive(backup_path, zip_name)
|
|
||||||
new_server = self.controller.import_zip_server(
|
|
||||||
svr_obj.server_name,
|
|
||||||
temp_dir,
|
|
||||||
server_data["executable"],
|
|
||||||
"1",
|
|
||||||
"2",
|
|
||||||
server_data["server_port"],
|
|
||||||
server_data["created_by"],
|
|
||||||
)
|
|
||||||
new_server_id = new_server
|
|
||||||
new_server = self.controller.servers.get_server_data(new_server)
|
|
||||||
self.controller.rename_backup_dir(
|
|
||||||
server_id, new_server_id, new_server["server_uuid"]
|
|
||||||
)
|
|
||||||
# preserve current schedules
|
|
||||||
for schedule in self.controller.management.get_schedules_by_server(
|
|
||||||
server_id
|
|
||||||
):
|
|
||||||
self.tasks_manager.update_job(
|
|
||||||
schedule.schedule_id, {"server_id": new_server_id}
|
|
||||||
)
|
|
||||||
# preserve execution command
|
|
||||||
new_server_obj = self.controller.servers.get_server_obj(
|
|
||||||
new_server_id
|
|
||||||
)
|
|
||||||
new_server_obj.execution_command = server_data["execution_command"]
|
|
||||||
# reset executable path
|
|
||||||
if svr_obj.path in svr_obj.executable:
|
|
||||||
new_server_obj.executable = str(svr_obj.executable).replace(
|
|
||||||
svr_obj.path, new_server_obj.path
|
|
||||||
)
|
|
||||||
# reset run command path
|
|
||||||
if svr_obj.path in svr_obj.execution_command:
|
|
||||||
new_server_obj.execution_command = str(
|
|
||||||
svr_obj.execution_command
|
|
||||||
).replace(svr_obj.path, new_server_obj.path)
|
|
||||||
# reset log path
|
|
||||||
if svr_obj.path in svr_obj.log_path:
|
|
||||||
new_server_obj.log_path = str(svr_obj.log_path).replace(
|
|
||||||
svr_obj.path, new_server_obj.path
|
|
||||||
)
|
|
||||||
self.controller.servers.update_server(new_server_obj)
|
|
||||||
|
|
||||||
# preserve backup config
|
|
||||||
backup_config = self.controller.management.get_backup_config(
|
|
||||||
server_id
|
|
||||||
)
|
|
||||||
excluded_dirs = []
|
|
||||||
server_obj = self.controller.servers.get_server_obj(server_id)
|
|
||||||
loop_backup_path = self.helper.wtol_path(server_obj.path)
|
|
||||||
for item in self.controller.management.get_excluded_backup_dirs(
|
|
||||||
server_id
|
|
||||||
):
|
|
||||||
item_path = self.helper.wtol_path(item)
|
|
||||||
bu_path = os.path.relpath(item_path, loop_backup_path)
|
|
||||||
bu_path = os.path.join(new_server_obj.path, bu_path)
|
|
||||||
excluded_dirs.append(bu_path)
|
|
||||||
self.controller.management.set_backup_config(
|
|
||||||
new_server_id,
|
|
||||||
new_server_obj.backup_path,
|
|
||||||
backup_config["max_backups"],
|
|
||||||
excluded_dirs,
|
|
||||||
backup_config["compress"],
|
|
||||||
backup_config["shutdown"],
|
|
||||||
)
|
|
||||||
# remove old server's tasks
|
|
||||||
try:
|
|
||||||
self.tasks_manager.remove_all_server_tasks(server_id)
|
|
||||||
except:
|
|
||||||
logger.info("No active tasks found for server")
|
|
||||||
self.controller.remove_server(server_id, True)
|
|
||||||
self.redirect("/panel/dashboard")
|
|
||||||
|
|
||||||
else:
|
|
||||||
backup_path = svr_obj.backup_path
|
|
||||||
if Helpers.validate_traversal(backup_path, zip_name):
|
|
||||||
temp_dir = Helpers.unzip_backup_archive(backup_path, zip_name)
|
|
||||||
new_server = self.controller.import_bedrock_zip_server(
|
|
||||||
svr_obj.server_name,
|
|
||||||
temp_dir,
|
|
||||||
server_data["executable"],
|
|
||||||
server_data["server_port"],
|
|
||||||
server_data["created_by"],
|
|
||||||
)
|
|
||||||
new_server_id = new_server
|
|
||||||
new_server = self.controller.servers.get_server_data(new_server)
|
|
||||||
self.controller.rename_backup_dir(
|
|
||||||
server_id, new_server_id, new_server["server_uuid"]
|
|
||||||
)
|
|
||||||
# preserve current schedules
|
|
||||||
for schedule in self.controller.management.get_schedules_by_server(
|
|
||||||
server_id
|
|
||||||
):
|
|
||||||
self.tasks_manager.update_job(
|
|
||||||
schedule.schedule_id, {"server_id": new_server_id}
|
|
||||||
)
|
|
||||||
# preserve execution command
|
|
||||||
new_server_obj = self.controller.servers.get_server_obj(
|
|
||||||
new_server_id
|
|
||||||
)
|
|
||||||
new_server_obj.execution_command = server_data["execution_command"]
|
|
||||||
# reset executable path
|
|
||||||
if server_obj.path in server_obj.executable:
|
|
||||||
new_server_obj.executable = str(server_obj.executable).replace(
|
|
||||||
server_obj.path, new_server_obj.path
|
|
||||||
)
|
|
||||||
# reset run command path
|
|
||||||
if server_obj.path in server_obj.execution_command:
|
|
||||||
new_server_obj.execution_command = str(
|
|
||||||
server_obj.execution_command
|
|
||||||
).replace(server_obj.path, new_server_obj.path)
|
|
||||||
# reset log path
|
|
||||||
if server_obj.path in server_obj.log_path:
|
|
||||||
new_server_obj.log_path = str(server_obj.log_path).replace(
|
|
||||||
server_obj.path, new_server_obj.path
|
|
||||||
)
|
|
||||||
self.controller.servers.update_server(new_server_obj)
|
|
||||||
|
|
||||||
# preserve backup config
|
|
||||||
backup_config = self.controller.management.get_backup_config(
|
|
||||||
server_id
|
|
||||||
)
|
|
||||||
excluded_dirs = []
|
|
||||||
server_obj = self.controller.servers.get_server_obj(server_id)
|
|
||||||
loop_backup_path = self.helper.wtol_path(server_obj.path)
|
|
||||||
for item in self.controller.management.get_excluded_backup_dirs(
|
|
||||||
server_id
|
|
||||||
):
|
|
||||||
item_path = self.helper.wtol_path(item)
|
|
||||||
bu_path = os.path.relpath(item_path, loop_backup_path)
|
|
||||||
bu_path = os.path.join(new_server_obj.path, bu_path)
|
|
||||||
excluded_dirs.append(bu_path)
|
|
||||||
self.controller.management.set_backup_config(
|
|
||||||
new_server_id,
|
|
||||||
new_server_obj.backup_path,
|
|
||||||
backup_config["max_backups"],
|
|
||||||
excluded_dirs,
|
|
||||||
backup_config["compress"],
|
|
||||||
backup_config["shutdown"],
|
|
||||||
)
|
|
||||||
try:
|
|
||||||
self.tasks_manager.remove_all_server_tasks(server_id)
|
|
||||||
except:
|
|
||||||
logger.info("No active tasks found for server")
|
|
||||||
self.controller.remove_server(server_id, True)
|
|
||||||
self.redirect("/panel/dashboard")
|
|
||||||
|
|
||||||
elif page == "unzip_server":
|
|
||||||
path = urllib.parse.unquote(self.get_argument("path", ""))
|
|
||||||
if not path:
|
|
||||||
path = os.path.join(
|
|
||||||
self.controller.project_root,
|
|
||||||
"imports",
|
|
||||||
urllib.parse.unquote(self.get_argument("file", "")),
|
|
||||||
)
|
|
||||||
if Helpers.check_file_exists(path):
|
|
||||||
self.helper.unzip_server(path, exec_user["user_id"])
|
|
||||||
else:
|
|
||||||
user_id = exec_user["user_id"]
|
|
||||||
if user_id:
|
|
||||||
time.sleep(5)
|
|
||||||
user_lang = self.controller.users.get_user_lang_by_id(user_id)
|
|
||||||
self.helper.websocket_helper.broadcast_user(
|
|
||||||
user_id,
|
|
||||||
"send_start_error",
|
|
||||||
{
|
|
||||||
"error": self.helper.translation.translate(
|
|
||||||
"error", "no-file", user_lang
|
|
||||||
)
|
|
||||||
},
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
elif page == "backup_select":
|
|
||||||
path = self.get_argument("path", None)
|
|
||||||
self.helper.backup_select(path, exec_user["user_id"])
|
|
||||||
return
|
|
||||||
|
|
||||||
elif page == "jar_cache":
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Not a super user")
|
|
||||||
return
|
|
||||||
|
|
||||||
self.controller.server_jars.manual_refresh_cache()
|
|
||||||
return
|
|
||||||
|
|
||||||
elif page == "update_server_dir":
|
|
||||||
if self.helper.dir_migration:
|
|
||||||
return
|
|
||||||
for server in self.controller.servers.get_all_servers_stats():
|
|
||||||
if server["stats"]["running"]:
|
|
||||||
self.helper.websocket_helper.broadcast_user(
|
|
||||||
exec_user["user_id"],
|
|
||||||
"send_start_error",
|
|
||||||
{
|
|
||||||
"error": "You must stop all servers before "
|
|
||||||
"starting a storage migration."
|
|
||||||
},
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Not a super user")
|
|
||||||
return
|
|
||||||
if self.helper.is_env_docker():
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=This feature is not"
|
|
||||||
" supported on docker environments"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
new_dir = urllib.parse.unquote(self.get_argument("server_dir"))
|
|
||||||
self.controller.update_master_server_dir(new_dir, exec_user["user_id"])
|
|
||||||
return
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def delete(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
|
|
||||||
if page == "del_backup":
|
|
||||||
if not permissions["Backup"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Backups")
|
|
||||||
return
|
|
||||||
file_path = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("file_path", default=None, strip=True)
|
|
||||||
)
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
Console.warning(f"Delete {file_path} for server {server_id}")
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "del_backup"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
server_info = self.controller.servers.get_server_data_by_id(server_id)
|
|
||||||
if not (
|
|
||||||
self.helper.is_subdir(
|
|
||||||
file_path, Helpers.get_os_understandable_path(server_info["path"])
|
|
||||||
)
|
|
||||||
or self.helper.is_subdir(
|
|
||||||
file_path,
|
|
||||||
Helpers.get_os_understandable_path(server_info["backup_path"]),
|
|
||||||
)
|
|
||||||
) or not Helpers.check_file_exists(os.path.abspath(file_path)):
|
|
||||||
logger.warning(f"Invalid path in del_backup ajax call ({file_path})")
|
|
||||||
Console.warning(f"Invalid path in del_backup ajax call ({file_path})")
|
|
||||||
return
|
|
||||||
|
|
||||||
# Delete the file
|
|
||||||
if Helpers.validate_traversal(
|
|
||||||
Helpers.get_os_understandable_path(server_info["backup_path"]),
|
|
||||||
file_path,
|
|
||||||
):
|
|
||||||
os.remove(file_path)
|
|
||||||
|
|
||||||
def check_server_id(self, server_id, page_name):
|
|
||||||
if server_id is None:
|
|
||||||
logger.warning(
|
|
||||||
f"Server ID not defined in {page_name} ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Server ID not defined in {page_name} ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
# does this server id exist?
|
|
||||||
if not self.controller.servers.server_id_exists(server_id):
|
|
||||||
logger.warning(
|
|
||||||
f"Server ID not found in {page_name} ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Server ID not found in {page_name} ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
return True
|
|
@ -2,15 +2,16 @@ import logging
|
|||||||
import re
|
import re
|
||||||
import typing as t
|
import typing as t
|
||||||
import orjson
|
import orjson
|
||||||
import bleach
|
import nh3
|
||||||
import tornado.web
|
import tornado.web
|
||||||
|
|
||||||
from app.classes.models.crafty_permissions import EnumPermissionsCrafty
|
from app.classes.models.crafty_permissions import EnumPermissionsCrafty
|
||||||
from app.classes.models.users import ApiKeys
|
from app.classes.models.users import ApiKeys
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
from app.classes.shared.main_controller import Controller
|
from app.classes.shared.main_controller import Controller
|
||||||
from app.classes.shared.translation import Translation
|
from app.classes.shared.translation import Translation
|
||||||
from app.classes.models.management import DatabaseShortcuts
|
from app.classes.shared.main_models import DatabaseShortcuts
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -24,15 +25,22 @@ class BaseHandler(tornado.web.RequestHandler):
|
|||||||
helper: Helpers
|
helper: Helpers
|
||||||
controller: Controller
|
controller: Controller
|
||||||
translator: Translation
|
translator: Translation
|
||||||
|
file_helper: FileHelpers
|
||||||
|
|
||||||
# noinspection PyAttributeOutsideInit
|
# noinspection PyAttributeOutsideInit
|
||||||
def initialize(
|
def initialize(
|
||||||
self, helper=None, controller=None, tasks_manager=None, translator=None
|
self,
|
||||||
|
helper=None,
|
||||||
|
controller=None,
|
||||||
|
tasks_manager=None,
|
||||||
|
translator=None,
|
||||||
|
file_helper=None,
|
||||||
):
|
):
|
||||||
self.helper = helper
|
self.helper = helper
|
||||||
self.controller = controller
|
self.controller = controller
|
||||||
self.tasks_manager = tasks_manager
|
self.tasks_manager = tasks_manager
|
||||||
self.translator = translator
|
self.translator = translator
|
||||||
|
self.file_helper = file_helper
|
||||||
|
|
||||||
def set_default_headers(self) -> None:
|
def set_default_headers(self) -> None:
|
||||||
"""
|
"""
|
||||||
@ -93,7 +101,7 @@ class BaseHandler(tornado.web.RequestHandler):
|
|||||||
if type(text) in self.nobleach:
|
if type(text) in self.nobleach:
|
||||||
logger.debug("Auto-bleaching - bypass type")
|
logger.debug("Auto-bleaching - bypass type")
|
||||||
return text
|
return text
|
||||||
return bleach.clean(text)
|
return nh3.clean(text)
|
||||||
|
|
||||||
def get_argument(
|
def get_argument(
|
||||||
self,
|
self,
|
||||||
|
@ -1,464 +0,0 @@
|
|||||||
import os
|
|
||||||
import logging
|
|
||||||
import bleach
|
|
||||||
import tornado.web
|
|
||||||
import tornado.escape
|
|
||||||
|
|
||||||
from app.classes.models.server_permissions import EnumPermissionsServer
|
|
||||||
from app.classes.shared.console import Console
|
|
||||||
from app.classes.shared.helpers import Helpers
|
|
||||||
from app.classes.shared.file_helpers import FileHelpers
|
|
||||||
from app.classes.web.base_handler import BaseHandler
|
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class FileHandler(BaseHandler):
|
|
||||||
def render_page(self, template, page_data):
|
|
||||||
self.render(
|
|
||||||
template,
|
|
||||||
data=page_data,
|
|
||||||
translate=self.translator.translate,
|
|
||||||
)
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def get(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
|
|
||||||
if page == "get_file":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
file_path = Helpers.get_os_understandable_path(
|
|
||||||
self.get_argument("file_path", None)
|
|
||||||
)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "get_file"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
file_path,
|
|
||||||
Helpers.get_os_understandable_path(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"]
|
|
||||||
),
|
|
||||||
) or not Helpers.check_file_exists(os.path.abspath(file_path)):
|
|
||||||
logger.warning(
|
|
||||||
f"Invalid path in get_file file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid path in get_file file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
error = None
|
|
||||||
|
|
||||||
try:
|
|
||||||
with open(file_path, encoding="utf-8") as file:
|
|
||||||
file_contents = file.read()
|
|
||||||
except UnicodeDecodeError:
|
|
||||||
file_contents = ""
|
|
||||||
error = "UnicodeDecodeError"
|
|
||||||
|
|
||||||
self.write({"content": file_contents, "error": error})
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "get_tree":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
path = self.get_argument("path", None)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "get_tree"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if Helpers.validate_traversal(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"], path
|
|
||||||
):
|
|
||||||
self.write(
|
|
||||||
Helpers.get_os_understandable_path(path)
|
|
||||||
+ "\n"
|
|
||||||
+ self.helper.generate_tree(path)
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "get_dir":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
path = self.get_argument("path", None)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "get_tree"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if Helpers.validate_traversal(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"], path
|
|
||||||
):
|
|
||||||
self.write(
|
|
||||||
Helpers.get_os_understandable_path(path)
|
|
||||||
+ "\n"
|
|
||||||
+ self.helper.generate_dir(path)
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def post(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
|
|
||||||
if page == "create_file":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
file_parent = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("file_parent", default=None, strip=True)
|
|
||||||
)
|
|
||||||
file_name = self.get_body_argument("file_name", default=None, strip=True)
|
|
||||||
file_path = os.path.join(file_parent, file_name)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "create_file"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
file_path,
|
|
||||||
Helpers.get_os_understandable_path(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"]
|
|
||||||
),
|
|
||||||
) or Helpers.check_file_exists(os.path.abspath(file_path)):
|
|
||||||
logger.warning(
|
|
||||||
f"Invalid path in create_file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid path in create_file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
# Create the file by opening it
|
|
||||||
with open(file_path, "w", encoding="utf-8") as file_object:
|
|
||||||
file_object.close()
|
|
||||||
|
|
||||||
elif page == "create_dir":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
dir_parent = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("dir_parent", default=None, strip=True)
|
|
||||||
)
|
|
||||||
dir_name = self.get_body_argument("dir_name", default=None, strip=True)
|
|
||||||
dir_path = os.path.join(dir_parent, dir_name)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "create_dir"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
dir_path,
|
|
||||||
Helpers.get_os_understandable_path(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"]
|
|
||||||
),
|
|
||||||
) or Helpers.check_path_exists(os.path.abspath(dir_path)):
|
|
||||||
logger.warning(
|
|
||||||
f"Invalid path in create_dir file ajax call ({dir_path})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid path in create_dir file ajax call ({dir_path})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
# Create the directory
|
|
||||||
os.mkdir(dir_path)
|
|
||||||
|
|
||||||
elif page == "unzip_file":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
path = Helpers.get_os_understandable_path(self.get_argument("path", None))
|
|
||||||
if Helpers.is_os_windows():
|
|
||||||
path = Helpers.wtol_path(path)
|
|
||||||
FileHelpers.unzip_file(path)
|
|
||||||
self.redirect(f"/panel/server_detail?id={server_id}&subpage=files")
|
|
||||||
return
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def delete(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
if page == "del_file":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
file_path = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("file_path", default=None, strip=True)
|
|
||||||
)
|
|
||||||
|
|
||||||
Console.warning(f"Delete {file_path} for server {server_id}")
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "del_file"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
server_info = self.controller.servers.get_server_data_by_id(server_id)
|
|
||||||
if not (
|
|
||||||
self.helper.is_subdir(
|
|
||||||
file_path, Helpers.get_os_understandable_path(server_info["path"])
|
|
||||||
)
|
|
||||||
or self.helper.is_subdir(
|
|
||||||
file_path,
|
|
||||||
Helpers.get_os_understandable_path(server_info["backup_path"]),
|
|
||||||
)
|
|
||||||
) or not Helpers.check_file_exists(os.path.abspath(file_path)):
|
|
||||||
logger.warning(f"Invalid path in del_file file ajax call ({file_path})")
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid path in del_file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
# Delete the file
|
|
||||||
FileHelpers.del_file(file_path)
|
|
||||||
|
|
||||||
elif page == "del_dir":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
dir_path = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("dir_path", default=None, strip=True)
|
|
||||||
)
|
|
||||||
|
|
||||||
Console.warning(f"Delete {dir_path} for server {server_id}")
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "del_dir"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
server_info = self.controller.servers.get_server_data_by_id(server_id)
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
dir_path, Helpers.get_os_understandable_path(server_info["path"])
|
|
||||||
) or not Helpers.check_path_exists(os.path.abspath(dir_path)):
|
|
||||||
logger.warning(f"Invalid path in del_file file ajax call ({dir_path})")
|
|
||||||
Console.warning(f"Invalid path in del_file file ajax call ({dir_path})")
|
|
||||||
return
|
|
||||||
|
|
||||||
# Delete the directory
|
|
||||||
# os.rmdir(dir_path) # Would only remove empty directories
|
|
||||||
if Helpers.validate_traversal(
|
|
||||||
Helpers.get_os_understandable_path(server_info["path"]), dir_path
|
|
||||||
):
|
|
||||||
# Removes also when there are contents
|
|
||||||
FileHelpers.del_dirs(dir_path)
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def put(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
if page == "save_file":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
file_contents = self.get_body_argument(
|
|
||||||
"file_contents", default=None, strip=True
|
|
||||||
)
|
|
||||||
file_path = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("file_path", default=None, strip=True)
|
|
||||||
)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "save_file"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
file_path,
|
|
||||||
Helpers.get_os_understandable_path(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"]
|
|
||||||
),
|
|
||||||
) or not Helpers.check_file_exists(os.path.abspath(file_path)):
|
|
||||||
logger.warning(
|
|
||||||
f"Invalid path in save_file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid path in save_file file ajax call ({file_path})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
# Open the file in write mode and store the content in file_object
|
|
||||||
with open(file_path, "w", encoding="utf-8") as file_object:
|
|
||||||
file_object.write(file_contents)
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def patch(self, page):
|
|
||||||
api_key, _, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
user_perms = self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
if page == "rename_file":
|
|
||||||
if not permissions["Files"] in user_perms:
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access to Files")
|
|
||||||
return
|
|
||||||
item_path = Helpers.get_os_understandable_path(
|
|
||||||
self.get_body_argument("item_path", default=None, strip=True)
|
|
||||||
)
|
|
||||||
new_item_name = self.get_body_argument(
|
|
||||||
"new_item_name", default=None, strip=True
|
|
||||||
)
|
|
||||||
|
|
||||||
if not self.check_server_id(server_id, "rename_file"):
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
if item_path is None or new_item_name is None:
|
|
||||||
logger.warning("Invalid path(s) in rename_file file ajax call")
|
|
||||||
Console.warning("Invalid path(s) in rename_file file ajax call")
|
|
||||||
return
|
|
||||||
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
item_path,
|
|
||||||
Helpers.get_os_understandable_path(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"]
|
|
||||||
),
|
|
||||||
) or not Helpers.check_path_exists(os.path.abspath(item_path)):
|
|
||||||
logger.warning(
|
|
||||||
f"Invalid old name path in rename_file file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid old name path in rename_file file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
new_item_path = os.path.join(os.path.split(item_path)[0], new_item_name)
|
|
||||||
|
|
||||||
if not self.helper.is_subdir(
|
|
||||||
new_item_path,
|
|
||||||
Helpers.get_os_understandable_path(
|
|
||||||
self.controller.servers.get_server_data_by_id(server_id)["path"]
|
|
||||||
),
|
|
||||||
) or Helpers.check_path_exists(os.path.abspath(new_item_path)):
|
|
||||||
logger.warning(
|
|
||||||
f"Invalid new name path in rename_file file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Invalid new name path in rename_file file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
# RENAME
|
|
||||||
os.rename(item_path, new_item_path)
|
|
||||||
|
|
||||||
def check_server_id(self, server_id, page_name):
|
|
||||||
if server_id is None:
|
|
||||||
logger.warning(
|
|
||||||
f"Server ID not defined in {page_name} file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Server ID not defined in {page_name} file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
server_id = bleach.clean(server_id)
|
|
||||||
|
|
||||||
# does this server id exist?
|
|
||||||
if not self.controller.servers.server_id_exists(server_id):
|
|
||||||
logger.warning(
|
|
||||||
f"Server ID not found in {page_name} file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
Console.warning(
|
|
||||||
f"Server ID not found in {page_name} file ajax call ({server_id})"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
return True
|
|
53
app/classes/web/metrics_handler.py
Normal file
53
app/classes/web/metrics_handler.py
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
import logging
|
||||||
|
import typing as t
|
||||||
|
|
||||||
|
from prometheus_client import REGISTRY, CollectorRegistry
|
||||||
|
from prometheus_client.exposition import _bake_output
|
||||||
|
from prometheus_client.exposition import parse_qs, urlparse
|
||||||
|
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class BaseMetricsHandler(BaseApiHandler):
|
||||||
|
"""HTTP handler that gives metrics from ``REGISTRY``."""
|
||||||
|
|
||||||
|
registry: CollectorRegistry = REGISTRY
|
||||||
|
# registry.unregister(GC_COLLECTOR)
|
||||||
|
# registry.unregister(PLATFORM_COLLECTOR)
|
||||||
|
# registry.unregister(PROCESS_COLLECTOR)
|
||||||
|
|
||||||
|
def get_registry(self) -> None:
|
||||||
|
# Prepare parameters
|
||||||
|
registry = self.registry
|
||||||
|
accept_header = self.request.headers.get("Accept")
|
||||||
|
accept_encoding_header = self.request.headers.get("Accept-Encoding")
|
||||||
|
params = parse_qs(urlparse(self.request.path).query)
|
||||||
|
# Bake output
|
||||||
|
status, headers, output = _bake_output(
|
||||||
|
registry, accept_header, accept_encoding_header, params, False
|
||||||
|
)
|
||||||
|
# Return output
|
||||||
|
self.finish_metrics(int(status.split(" ", maxsplit=1)[0]), headers, output)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def factory(cls, registry: CollectorRegistry) -> type:
|
||||||
|
"""Returns a dynamic MetricsHandler class tied
|
||||||
|
to the passed registry.
|
||||||
|
"""
|
||||||
|
# This implementation relies on MetricsHandler.registry
|
||||||
|
# (defined above and defaulted to REGISTRY).
|
||||||
|
|
||||||
|
# As we have unicode_literals, we need to create a str()
|
||||||
|
# object for type().
|
||||||
|
cls_name = str(cls.__name__)
|
||||||
|
MyMetricsHandler = type(cls_name, (cls, object), {"registry": registry})
|
||||||
|
return MyMetricsHandler
|
||||||
|
|
||||||
|
def finish_metrics(self, status: int, headers, data: t.Dict[str, t.Any]):
|
||||||
|
self.set_status(status)
|
||||||
|
self.set_header("Content-Type", "text/plain")
|
||||||
|
for header in headers:
|
||||||
|
self.set_header(*header)
|
||||||
|
self.finish(data)
|
@ -7,7 +7,8 @@ import json
|
|||||||
import logging
|
import logging
|
||||||
import threading
|
import threading
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
import bleach
|
from zoneinfo import ZoneInfoNotFoundError
|
||||||
|
import nh3
|
||||||
import requests
|
import requests
|
||||||
import tornado.web
|
import tornado.web
|
||||||
import tornado.escape
|
import tornado.escape
|
||||||
@ -15,7 +16,6 @@ from tornado import iostream
|
|||||||
|
|
||||||
# TZLocal is set as a hidden import on win pipeline
|
# TZLocal is set as a hidden import on win pipeline
|
||||||
from tzlocal import get_localzone
|
from tzlocal import get_localzone
|
||||||
from tzlocal.utils import ZoneInfoNotFoundError
|
|
||||||
|
|
||||||
from app.classes.models.servers import Servers
|
from app.classes.models.servers import Servers
|
||||||
from app.classes.models.server_permissions import EnumPermissionsServer
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
@ -25,6 +25,7 @@ from app.classes.controllers.roles_controller import RolesController
|
|||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.main_models import DatabaseShortcuts
|
from app.classes.shared.main_models import DatabaseShortcuts
|
||||||
from app.classes.web.base_handler import BaseHandler
|
from app.classes.web.base_handler import BaseHandler
|
||||||
|
from app.classes.web.webhooks.webhook_factory import WebhookFactory
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -67,9 +68,7 @@ class PanelHandler(BaseHandler):
|
|||||||
) in self.controller.crafty_perms.list_defined_crafty_permissions():
|
) in self.controller.crafty_perms.list_defined_crafty_permissions():
|
||||||
argument = int(
|
argument = int(
|
||||||
float(
|
float(
|
||||||
bleach.clean(
|
nh3.clean(self.get_argument(f"permission_{permission.name}", "0"))
|
||||||
self.get_argument(f"permission_{permission.name}", "0")
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
if argument:
|
if argument:
|
||||||
@ -78,9 +77,7 @@ class PanelHandler(BaseHandler):
|
|||||||
)
|
)
|
||||||
|
|
||||||
q_argument = int(
|
q_argument = int(
|
||||||
float(
|
float(nh3.clean(self.get_argument(f"quantity_{permission.name}", "0")))
|
||||||
bleach.clean(self.get_argument(f"quantity_{permission.name}", "0"))
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
if q_argument:
|
if q_argument:
|
||||||
server_quantity[permission.name] = q_argument
|
server_quantity[permission.name] = q_argument
|
||||||
@ -348,7 +345,9 @@ class PanelHandler(BaseHandler):
|
|||||||
) as credits_default_local:
|
) as credits_default_local:
|
||||||
try:
|
try:
|
||||||
remote = requests.get(
|
remote = requests.get(
|
||||||
"https://craftycontrol.com/credits-v2", allow_redirects=True
|
"https://craftycontrol.com/credits-v2",
|
||||||
|
allow_redirects=True,
|
||||||
|
timeout=10,
|
||||||
)
|
)
|
||||||
credits_dict: dict = remote.json()
|
credits_dict: dict = remote.json()
|
||||||
if not credits_dict["staff"]:
|
if not credits_dict["staff"]:
|
||||||
@ -479,7 +478,7 @@ class PanelHandler(BaseHandler):
|
|||||||
template = "panel/dashboard.html"
|
template = "panel/dashboard.html"
|
||||||
|
|
||||||
elif page == "server_detail":
|
elif page == "server_detail":
|
||||||
subpage = bleach.clean(self.get_argument("subpage", ""))
|
subpage = nh3.clean(self.get_argument("subpage", ""))
|
||||||
|
|
||||||
server_id = self.check_server_id()
|
server_id = self.check_server_id()
|
||||||
if server_id is None:
|
if server_id is None:
|
||||||
@ -747,8 +746,24 @@ class PanelHandler(BaseHandler):
|
|||||||
0, page_data["options"].pop(page_data["options"].index(days))
|
0, page_data["options"].pop(page_data["options"].index(days))
|
||||||
)
|
)
|
||||||
page_data["history_stats"] = self.controller.servers.get_history_stats(
|
page_data["history_stats"] = self.controller.servers.get_history_stats(
|
||||||
server_id, days
|
server_id, hours=(days * 24)
|
||||||
)
|
)
|
||||||
|
if subpage == "webhooks":
|
||||||
|
if (
|
||||||
|
not page_data["permissions"]["Config"]
|
||||||
|
in page_data["user_permissions"]
|
||||||
|
):
|
||||||
|
if not superuser:
|
||||||
|
self.redirect(
|
||||||
|
"/panel/error?error=Unauthorized access to Webhooks Config"
|
||||||
|
)
|
||||||
|
return
|
||||||
|
page_data[
|
||||||
|
"webhooks"
|
||||||
|
] = self.controller.management.get_webhooks_by_server(
|
||||||
|
server_id, model=True
|
||||||
|
)
|
||||||
|
page_data["triggers"] = WebhookFactory.get_monitored_events()
|
||||||
|
|
||||||
def get_banned_players_html():
|
def get_banned_players_html():
|
||||||
banned_players = self.controller.servers.get_banned_players(server_id)
|
banned_players = self.controller.servers.get_banned_players(server_id)
|
||||||
@ -1016,6 +1031,110 @@ class PanelHandler(BaseHandler):
|
|||||||
|
|
||||||
template = "panel/panel_edit_user.html"
|
template = "panel/panel_edit_user.html"
|
||||||
|
|
||||||
|
elif page == "add_webhook":
|
||||||
|
server_id = self.get_argument("id", None)
|
||||||
|
if server_id is None:
|
||||||
|
return self.redirect("/panel/error?error=Invalid Server ID")
|
||||||
|
server_obj = self.controller.servers.get_server_instance_by_id(server_id)
|
||||||
|
page_data["backup_failed"] = server_obj.last_backup_status()
|
||||||
|
server_obj = None
|
||||||
|
page_data["active_link"] = "webhooks"
|
||||||
|
page_data["server_data"] = self.controller.servers.get_server_data_by_id(
|
||||||
|
server_id
|
||||||
|
)
|
||||||
|
page_data[
|
||||||
|
"user_permissions"
|
||||||
|
] = self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
exec_user["user_id"], server_id
|
||||||
|
)
|
||||||
|
page_data["permissions"] = {
|
||||||
|
"Commands": EnumPermissionsServer.COMMANDS,
|
||||||
|
"Terminal": EnumPermissionsServer.TERMINAL,
|
||||||
|
"Logs": EnumPermissionsServer.LOGS,
|
||||||
|
"Schedule": EnumPermissionsServer.SCHEDULE,
|
||||||
|
"Backup": EnumPermissionsServer.BACKUP,
|
||||||
|
"Files": EnumPermissionsServer.FILES,
|
||||||
|
"Config": EnumPermissionsServer.CONFIG,
|
||||||
|
"Players": EnumPermissionsServer.PLAYERS,
|
||||||
|
}
|
||||||
|
page_data["server_stats"] = self.controller.servers.get_server_stats_by_id(
|
||||||
|
server_id
|
||||||
|
)
|
||||||
|
page_data["server_stats"][
|
||||||
|
"server_type"
|
||||||
|
] = self.controller.servers.get_server_type_by_id(server_id)
|
||||||
|
page_data["new_webhook"] = True
|
||||||
|
page_data["webhook"] = {}
|
||||||
|
page_data["webhook"]["webhook_type"] = "Custom"
|
||||||
|
page_data["webhook"]["name"] = ""
|
||||||
|
page_data["webhook"]["url"] = ""
|
||||||
|
page_data["webhook"]["bot_name"] = "Crafty Controller"
|
||||||
|
page_data["webhook"]["trigger"] = []
|
||||||
|
page_data["webhook"]["body"] = ""
|
||||||
|
page_data["webhook"]["color"] = "#005cd1"
|
||||||
|
page_data["webhook"]["enabled"] = True
|
||||||
|
|
||||||
|
page_data["providers"] = WebhookFactory.get_supported_providers()
|
||||||
|
page_data["triggers"] = WebhookFactory.get_monitored_events()
|
||||||
|
|
||||||
|
if not EnumPermissionsServer.CONFIG in page_data["user_permissions"]:
|
||||||
|
if not superuser:
|
||||||
|
self.redirect("/panel/error?error=Unauthorized access To Webhooks")
|
||||||
|
return
|
||||||
|
|
||||||
|
template = "panel/server_webhook_edit.html"
|
||||||
|
|
||||||
|
elif page == "webhook_edit":
|
||||||
|
server_id = self.get_argument("id", None)
|
||||||
|
webhook_id = self.get_argument("webhook_id", None)
|
||||||
|
if server_id is None:
|
||||||
|
return self.redirect("/panel/error?error=Invalid Server ID")
|
||||||
|
server_obj = self.controller.servers.get_server_instance_by_id(server_id)
|
||||||
|
page_data["backup_failed"] = server_obj.last_backup_status()
|
||||||
|
server_obj = None
|
||||||
|
page_data["active_link"] = "webhooks"
|
||||||
|
page_data["server_data"] = self.controller.servers.get_server_data_by_id(
|
||||||
|
server_id
|
||||||
|
)
|
||||||
|
page_data[
|
||||||
|
"user_permissions"
|
||||||
|
] = self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
exec_user["user_id"], server_id
|
||||||
|
)
|
||||||
|
page_data["permissions"] = {
|
||||||
|
"Commands": EnumPermissionsServer.COMMANDS,
|
||||||
|
"Terminal": EnumPermissionsServer.TERMINAL,
|
||||||
|
"Logs": EnumPermissionsServer.LOGS,
|
||||||
|
"Schedule": EnumPermissionsServer.SCHEDULE,
|
||||||
|
"Backup": EnumPermissionsServer.BACKUP,
|
||||||
|
"Files": EnumPermissionsServer.FILES,
|
||||||
|
"Config": EnumPermissionsServer.CONFIG,
|
||||||
|
"Players": EnumPermissionsServer.PLAYERS,
|
||||||
|
}
|
||||||
|
page_data["server_stats"] = self.controller.servers.get_server_stats_by_id(
|
||||||
|
server_id
|
||||||
|
)
|
||||||
|
page_data["server_stats"][
|
||||||
|
"server_type"
|
||||||
|
] = self.controller.servers.get_server_type_by_id(server_id)
|
||||||
|
page_data["new_webhook"] = False
|
||||||
|
page_data["webhook"] = self.controller.management.get_webhook_by_id(
|
||||||
|
webhook_id
|
||||||
|
)
|
||||||
|
page_data["webhook"]["trigger"] = str(
|
||||||
|
page_data["webhook"]["trigger"]
|
||||||
|
).split(",")
|
||||||
|
|
||||||
|
page_data["providers"] = WebhookFactory.get_supported_providers()
|
||||||
|
page_data["triggers"] = WebhookFactory.get_monitored_events()
|
||||||
|
|
||||||
|
if not EnumPermissionsServer.CONFIG in page_data["user_permissions"]:
|
||||||
|
if not superuser:
|
||||||
|
self.redirect("/panel/error?error=Unauthorized access To Webhooks")
|
||||||
|
return
|
||||||
|
|
||||||
|
template = "panel/server_webhook_edit.html"
|
||||||
|
|
||||||
elif page == "add_schedule":
|
elif page == "add_schedule":
|
||||||
server_id = self.get_argument("id", None)
|
server_id = self.get_argument("id", None)
|
||||||
if server_id is None:
|
if server_id is None:
|
||||||
@ -1284,7 +1403,7 @@ class PanelHandler(BaseHandler):
|
|||||||
template = "panel/panel_edit_user_apikeys.html"
|
template = "panel/panel_edit_user_apikeys.html"
|
||||||
|
|
||||||
elif page == "remove_user":
|
elif page == "remove_user":
|
||||||
user_id = bleach.clean(self.get_argument("id", None))
|
user_id = nh3.clean(self.get_argument("id", None))
|
||||||
|
|
||||||
if (
|
if (
|
||||||
not superuser
|
not superuser
|
||||||
@ -1415,40 +1534,8 @@ class PanelHandler(BaseHandler):
|
|||||||
|
|
||||||
template = "panel/panel_edit_role.html"
|
template = "panel/panel_edit_role.html"
|
||||||
|
|
||||||
elif page == "remove_role":
|
|
||||||
role_id = bleach.clean(self.get_argument("id", None))
|
|
||||||
|
|
||||||
if (
|
|
||||||
not superuser
|
|
||||||
and self.controller.roles.get_role(role_id)["manager"]
|
|
||||||
!= exec_user["user_id"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: not superuser not"
|
|
||||||
" role manager"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if role_id is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid Role ID")
|
|
||||||
return
|
|
||||||
# does this user id exist?
|
|
||||||
target_role = self.controller.roles.get_role(role_id)
|
|
||||||
if not target_role:
|
|
||||||
self.redirect("/panel/error?error=Invalid Role ID")
|
|
||||||
return
|
|
||||||
|
|
||||||
self.controller.roles.remove_role(role_id)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Removed role {target_role['role_name']} (RID:{role_id})",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect("/panel/panel_config")
|
|
||||||
|
|
||||||
elif page == "activity_logs":
|
elif page == "activity_logs":
|
||||||
page_data["audit_logs"] = self.controller.management.get_actity_log()
|
page_data["audit_logs"] = self.controller.management.get_activity_log()
|
||||||
|
|
||||||
template = "panel/activity_logs.html"
|
template = "panel/activity_logs.html"
|
||||||
|
|
||||||
@ -1530,606 +1617,3 @@ class PanelHandler(BaseHandler):
|
|||||||
utc_offset=(time.timezone * -1 / 60 / 60),
|
utc_offset=(time.timezone * -1 / 60 / 60),
|
||||||
translate=self.translator.translate,
|
translate=self.translator.translate,
|
||||||
)
|
)
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def post(self, page):
|
|
||||||
api_key, _token_data, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
server_id = self.get_argument("id", None)
|
|
||||||
permissions = {
|
|
||||||
"Commands": EnumPermissionsServer.COMMANDS,
|
|
||||||
"Terminal": EnumPermissionsServer.TERMINAL,
|
|
||||||
"Logs": EnumPermissionsServer.LOGS,
|
|
||||||
"Schedule": EnumPermissionsServer.SCHEDULE,
|
|
||||||
"Backup": EnumPermissionsServer.BACKUP,
|
|
||||||
"Files": EnumPermissionsServer.FILES,
|
|
||||||
"Config": EnumPermissionsServer.CONFIG,
|
|
||||||
"Players": EnumPermissionsServer.PLAYERS,
|
|
||||||
}
|
|
||||||
if superuser:
|
|
||||||
# defined_servers = self.controller.servers.list_defined_servers()
|
|
||||||
exec_user_role = {"Super User"}
|
|
||||||
exec_user_crafty_permissions = (
|
|
||||||
self.controller.crafty_perms.list_defined_crafty_permissions()
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
exec_user_crafty_permissions = (
|
|
||||||
self.controller.crafty_perms.get_crafty_permissions_list(
|
|
||||||
exec_user["user_id"]
|
|
||||||
)
|
|
||||||
)
|
|
||||||
# defined_servers =
|
|
||||||
# self.controller.servers.get_authorized_servers(exec_user["user_id"])
|
|
||||||
exec_user_role = set()
|
|
||||||
for r in exec_user["roles"]:
|
|
||||||
role = self.controller.roles.get_role(r)
|
|
||||||
exec_user_role.add(role["role_name"])
|
|
||||||
|
|
||||||
if page == "server_backup":
|
|
||||||
logger.debug(self.request.arguments)
|
|
||||||
|
|
||||||
server_id = self.check_server_id()
|
|
||||||
if not server_id:
|
|
||||||
return
|
|
||||||
|
|
||||||
if (
|
|
||||||
not permissions["Backup"]
|
|
||||||
in self.controller.server_perms.get_user_id_permissions_list(
|
|
||||||
exec_user["user_id"], server_id
|
|
||||||
)
|
|
||||||
and not superuser
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: User not authorized"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
server_obj = self.controller.servers.get_server_obj(server_id)
|
|
||||||
compress = self.get_argument("compress", False)
|
|
||||||
shutdown = self.get_argument("shutdown", False)
|
|
||||||
check_changed = self.get_argument("changed")
|
|
||||||
before = self.get_argument("backup_before", "")
|
|
||||||
after = self.get_argument("backup_after", "")
|
|
||||||
if str(check_changed) == str(1):
|
|
||||||
checked = self.get_body_arguments("root_path")
|
|
||||||
else:
|
|
||||||
checked = self.controller.management.get_excluded_backup_dirs(server_id)
|
|
||||||
if superuser:
|
|
||||||
backup_path = self.get_argument("backup_path", None)
|
|
||||||
if Helpers.is_os_windows():
|
|
||||||
backup_path.replace(" ", "^ ")
|
|
||||||
backup_path = Helpers.wtol_path(backup_path)
|
|
||||||
else:
|
|
||||||
backup_path = server_obj.backup_path
|
|
||||||
max_backups = bleach.clean(self.get_argument("max_backups", None))
|
|
||||||
|
|
||||||
server_obj = self.controller.servers.get_server_obj(server_id)
|
|
||||||
|
|
||||||
server_obj.backup_path = backup_path
|
|
||||||
self.controller.servers.update_server(server_obj)
|
|
||||||
self.controller.management.set_backup_config(
|
|
||||||
server_id,
|
|
||||||
max_backups=max_backups,
|
|
||||||
excluded_dirs=checked,
|
|
||||||
compress=bool(compress),
|
|
||||||
shutdown=bool(shutdown),
|
|
||||||
before=before,
|
|
||||||
after=after,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Edited server {server_id}: updated backups",
|
|
||||||
server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.tasks_manager.reload_schedule_from_db()
|
|
||||||
self.redirect(f"/panel/server_detail?id={server_id}&subpage=backup")
|
|
||||||
|
|
||||||
elif page == "config_json":
|
|
||||||
try:
|
|
||||||
data = {}
|
|
||||||
with open(self.helper.settings_file, "r", encoding="utf-8") as f:
|
|
||||||
keys = json.load(f).keys()
|
|
||||||
this_uuid = self.get_argument("uuid")
|
|
||||||
for key in keys:
|
|
||||||
arg_data = self.get_argument(key)
|
|
||||||
if arg_data.startswith(this_uuid):
|
|
||||||
arg_data = arg_data.split(",")
|
|
||||||
arg_data.pop(0)
|
|
||||||
data[key] = arg_data
|
|
||||||
else:
|
|
||||||
try:
|
|
||||||
data[key] = int(arg_data)
|
|
||||||
except:
|
|
||||||
if arg_data == "True":
|
|
||||||
data[key] = True
|
|
||||||
elif arg_data == "False":
|
|
||||||
data[key] = False
|
|
||||||
else:
|
|
||||||
data[key] = arg_data
|
|
||||||
keys = list(data.keys())
|
|
||||||
keys.sort()
|
|
||||||
sorted_data = {i: data[i] for i in keys}
|
|
||||||
with open(self.helper.settings_file, "w", encoding="utf-8") as f:
|
|
||||||
json.dump(sorted_data, f, indent=4)
|
|
||||||
except Exception as e:
|
|
||||||
logger.critical(
|
|
||||||
"Config File Error: Unable to read "
|
|
||||||
f"{self.helper.settings_file} due to {e}"
|
|
||||||
)
|
|
||||||
|
|
||||||
self.redirect("/panel/config_json")
|
|
||||||
|
|
||||||
elif page == "edit_user":
|
|
||||||
if bleach.clean(self.get_argument("username", None)).lower() == "system":
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: "
|
|
||||||
"system user is not editable"
|
|
||||||
)
|
|
||||||
user_id = bleach.clean(self.get_argument("id", None))
|
|
||||||
user = self.controller.users.get_user_by_id(user_id)
|
|
||||||
username = bleach.clean(self.get_argument("username", None).lower())
|
|
||||||
theme = bleach.clean(self.get_argument("theme", "default"))
|
|
||||||
if (
|
|
||||||
username != self.controller.users.get_user_by_id(user_id)["username"]
|
|
||||||
and username in self.controller.users.get_all_usernames()
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Duplicate User: Useranme already exists."
|
|
||||||
)
|
|
||||||
password0 = bleach.clean(self.get_argument("password0", None))
|
|
||||||
password1 = bleach.clean(self.get_argument("password1", None))
|
|
||||||
email = bleach.clean(self.get_argument("email", "default@example.com"))
|
|
||||||
enabled = int(float(self.get_argument("enabled", "0")))
|
|
||||||
try:
|
|
||||||
hints = int(bleach.clean(self.get_argument("hints")))
|
|
||||||
hints = True
|
|
||||||
except:
|
|
||||||
hints = False
|
|
||||||
lang = bleach.clean(
|
|
||||||
self.get_argument("language"), self.helper.get_setting("language")
|
|
||||||
)
|
|
||||||
|
|
||||||
if superuser:
|
|
||||||
# Checks if user is trying to change super user status of self.
|
|
||||||
# We don't want that. Automatically make them stay super user
|
|
||||||
# since we know they are.
|
|
||||||
if str(exec_user["user_id"]) != str(user_id):
|
|
||||||
superuser = int(bleach.clean(self.get_argument("superuser", "0")))
|
|
||||||
else:
|
|
||||||
superuser = 1
|
|
||||||
else:
|
|
||||||
superuser = 0
|
|
||||||
|
|
||||||
if exec_user["superuser"]:
|
|
||||||
manager = self.get_argument("manager")
|
|
||||||
if manager == "":
|
|
||||||
manager = None
|
|
||||||
else:
|
|
||||||
manager = int(manager)
|
|
||||||
else:
|
|
||||||
manager = user["manager"]
|
|
||||||
|
|
||||||
if (
|
|
||||||
not exec_user["superuser"]
|
|
||||||
and int(exec_user["user_id"]) != user["manager"]
|
|
||||||
):
|
|
||||||
if username is None or username == "":
|
|
||||||
self.redirect("/panel/error?error=Invalid username")
|
|
||||||
return
|
|
||||||
if user_id is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid User ID")
|
|
||||||
return
|
|
||||||
if (
|
|
||||||
EnumPermissionsCrafty.USER_CONFIG
|
|
||||||
not in exec_user_crafty_permissions
|
|
||||||
):
|
|
||||||
if str(user_id) != str(exec_user["user_id"]):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: not a user editor"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
user_data = {
|
|
||||||
"username": username,
|
|
||||||
"password": password0,
|
|
||||||
"email": email,
|
|
||||||
"lang": lang,
|
|
||||||
"hints": hints,
|
|
||||||
"theme": theme,
|
|
||||||
}
|
|
||||||
self.controller.users.update_user(user_id, user_data=user_data)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Edited user {username} (UID:{user_id}) password",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect("/panel/panel_config")
|
|
||||||
return
|
|
||||||
# does this user id exist?
|
|
||||||
if not self.controller.users.user_id_exists(user_id):
|
|
||||||
self.redirect("/panel/error?error=Invalid User ID")
|
|
||||||
return
|
|
||||||
else:
|
|
||||||
if password0 != password1:
|
|
||||||
self.redirect("/panel/error?error=Passwords must match")
|
|
||||||
return
|
|
||||||
|
|
||||||
roles = self.get_user_role_memberships()
|
|
||||||
permissions_mask, server_quantity = self.get_perms_quantity()
|
|
||||||
|
|
||||||
# if email is None or "":
|
|
||||||
# email = "default@example.com"
|
|
||||||
|
|
||||||
user_data = {
|
|
||||||
"username": username,
|
|
||||||
"manager": manager,
|
|
||||||
"password": password0,
|
|
||||||
"email": email,
|
|
||||||
"enabled": enabled,
|
|
||||||
"roles": roles,
|
|
||||||
"lang": lang,
|
|
||||||
"superuser": superuser,
|
|
||||||
"hints": hints,
|
|
||||||
"theme": theme,
|
|
||||||
}
|
|
||||||
user_crafty_data = {
|
|
||||||
"permissions_mask": permissions_mask,
|
|
||||||
"server_quantity": server_quantity,
|
|
||||||
}
|
|
||||||
self.controller.users.update_user(
|
|
||||||
user_id, user_data=user_data, user_crafty_data=user_crafty_data
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Edited user {username} (UID:{user_id}) with roles {roles} "
|
|
||||||
f"and permissions {permissions_mask}",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect("/panel/panel_config")
|
|
||||||
|
|
||||||
elif page == "edit_user_apikeys":
|
|
||||||
user_id = self.get_argument("id", None)
|
|
||||||
name = self.get_argument("name", None)
|
|
||||||
superuser = self.get_argument("superuser", None) == "1"
|
|
||||||
|
|
||||||
if name is None or name == "":
|
|
||||||
self.redirect("/panel/error?error=Invalid API key name")
|
|
||||||
return
|
|
||||||
if user_id is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid User ID")
|
|
||||||
return
|
|
||||||
# does this user id exist?
|
|
||||||
if not self.controller.users.user_id_exists(user_id):
|
|
||||||
self.redirect("/panel/error?error=Invalid User ID")
|
|
||||||
return
|
|
||||||
|
|
||||||
if str(user_id) != str(exec_user["user_id"]) and not exec_user["superuser"]:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=You do not have access to change"
|
|
||||||
+ "this user's api key."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
crafty_permissions_mask = self.get_perms()
|
|
||||||
server_permissions_mask = self.get_perms_server()
|
|
||||||
|
|
||||||
self.controller.users.add_user_api_key(
|
|
||||||
name,
|
|
||||||
user_id,
|
|
||||||
superuser,
|
|
||||||
server_permissions_mask,
|
|
||||||
crafty_permissions_mask,
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Added API key {name} with crafty permissions "
|
|
||||||
f"{crafty_permissions_mask}"
|
|
||||||
f" and {server_permissions_mask} for user with UID: {user_id}",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect(f"/panel/edit_user_apikeys?id={user_id}")
|
|
||||||
|
|
||||||
elif page == "get_token":
|
|
||||||
key_id = self.get_argument("id", None)
|
|
||||||
|
|
||||||
if key_id is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid Key ID")
|
|
||||||
return
|
|
||||||
key = self.controller.users.get_user_api_key(key_id)
|
|
||||||
# does this user id exist?
|
|
||||||
if key is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid Key ID")
|
|
||||||
return
|
|
||||||
|
|
||||||
if (
|
|
||||||
str(key.user_id) != str(exec_user["user_id"])
|
|
||||||
and not exec_user["superuser"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=You are not authorized to access this key."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Generated a new API token for the key {key.name} "
|
|
||||||
f"from user with UID: {key.user_id}",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
|
|
||||||
self.write(
|
|
||||||
self.controller.authentication.generate(
|
|
||||||
key.user_id_id, {"token_id": key.token_id}
|
|
||||||
)
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
|
|
||||||
elif page == "add_user":
|
|
||||||
username = bleach.clean(self.get_argument("username", None).lower())
|
|
||||||
if username.lower() == "system":
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: "
|
|
||||||
"username system is reserved for the Crafty system."
|
|
||||||
" Please choose a different username."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
password0 = bleach.clean(self.get_argument("password0", None))
|
|
||||||
password1 = bleach.clean(self.get_argument("password1", None))
|
|
||||||
email = bleach.clean(self.get_argument("email", "default@example.com"))
|
|
||||||
enabled = int(float(self.get_argument("enabled", "0")))
|
|
||||||
theme = bleach.clean(self.get_argument("theme"), "default")
|
|
||||||
hints = True
|
|
||||||
lang = bleach.clean(
|
|
||||||
self.get_argument("lang", self.helper.get_setting("language"))
|
|
||||||
)
|
|
||||||
# We don't want a non-super user to be able to create a super user.
|
|
||||||
if superuser:
|
|
||||||
new_superuser = int(bleach.clean(self.get_argument("superuser", "0")))
|
|
||||||
else:
|
|
||||||
new_superuser = 0
|
|
||||||
|
|
||||||
if EnumPermissionsCrafty.USER_CONFIG not in exec_user_crafty_permissions:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: not a user editor"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
if (
|
|
||||||
not self.controller.crafty_perms.can_add_user(exec_user["user_id"])
|
|
||||||
and not exec_user["superuser"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: quantity limit reached"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if username is None or username == "":
|
|
||||||
self.redirect("/panel/error?error=Invalid username")
|
|
||||||
return
|
|
||||||
|
|
||||||
if exec_user["superuser"]:
|
|
||||||
manager = self.get_argument("manager")
|
|
||||||
if manager == "":
|
|
||||||
manager = None
|
|
||||||
else:
|
|
||||||
manager = int(manager)
|
|
||||||
else:
|
|
||||||
manager = int(exec_user["user_id"])
|
|
||||||
# does this user id exist?
|
|
||||||
if self.controller.users.get_id_by_name(username) is not None:
|
|
||||||
self.redirect("/panel/error?error=User exists")
|
|
||||||
return
|
|
||||||
|
|
||||||
if password0 != password1:
|
|
||||||
self.redirect("/panel/error?error=Passwords must match")
|
|
||||||
return
|
|
||||||
|
|
||||||
roles = self.get_user_role_memberships()
|
|
||||||
permissions_mask, server_quantity = self.get_perms_quantity()
|
|
||||||
|
|
||||||
user_id = self.controller.users.add_user(
|
|
||||||
username,
|
|
||||||
manager=manager,
|
|
||||||
password=password0,
|
|
||||||
email=email,
|
|
||||||
enabled=enabled,
|
|
||||||
superuser=new_superuser,
|
|
||||||
theme=theme,
|
|
||||||
)
|
|
||||||
user_data = {"roles": roles, "lang": lang, "hints": True}
|
|
||||||
user_crafty_data = {
|
|
||||||
"permissions_mask": permissions_mask,
|
|
||||||
"server_quantity": server_quantity,
|
|
||||||
}
|
|
||||||
self.controller.users.update_user(
|
|
||||||
user_id, user_data=user_data, user_crafty_data=user_crafty_data
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Added user {username} (UID:{user_id})",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Edited user {username} (UID:{user_id}) with roles {roles}",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect("/panel/panel_config")
|
|
||||||
|
|
||||||
elif page == "edit_role":
|
|
||||||
role_id = bleach.clean(self.get_argument("id", None))
|
|
||||||
role_name = bleach.clean(self.get_argument("role_name", None))
|
|
||||||
|
|
||||||
role = self.controller.roles.get_role(role_id)
|
|
||||||
|
|
||||||
if (
|
|
||||||
EnumPermissionsCrafty.ROLES_CONFIG not in exec_user_crafty_permissions
|
|
||||||
and exec_user["user_id"] != role["manager"]
|
|
||||||
and not exec_user["superuser"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: not a role editor"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if role_name is None or role_name == "":
|
|
||||||
self.redirect("/panel/error?error=Invalid username")
|
|
||||||
return
|
|
||||||
if role_id is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid Role ID")
|
|
||||||
return
|
|
||||||
# does this user id exist?
|
|
||||||
if not self.controller.roles.role_id_exists(role_id):
|
|
||||||
self.redirect("/panel/error?error=Invalid Role ID")
|
|
||||||
return
|
|
||||||
|
|
||||||
if exec_user["superuser"]:
|
|
||||||
manager = self.get_argument("manager", None)
|
|
||||||
if manager == "":
|
|
||||||
manager = None
|
|
||||||
else:
|
|
||||||
manager = role["manager"]
|
|
||||||
|
|
||||||
servers = self.get_role_servers()
|
|
||||||
|
|
||||||
self.controller.roles.update_role_advanced(
|
|
||||||
role_id, role_name, servers, manager
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"edited role {role_name} (RID:{role_id}) with servers {servers}",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect("/panel/panel_config")
|
|
||||||
|
|
||||||
elif page == "add_role":
|
|
||||||
role_name = bleach.clean(self.get_argument("role_name", None))
|
|
||||||
if exec_user["superuser"]:
|
|
||||||
manager = self.get_argument("manager", None)
|
|
||||||
if manager == "":
|
|
||||||
manager = None
|
|
||||||
else:
|
|
||||||
manager = exec_user["user_id"]
|
|
||||||
|
|
||||||
if EnumPermissionsCrafty.ROLES_CONFIG not in exec_user_crafty_permissions:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: not a role editor"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if (
|
|
||||||
not self.controller.crafty_perms.can_add_role(exec_user["user_id"])
|
|
||||||
and not exec_user["superuser"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: quantity limit reached"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if role_name is None or role_name == "":
|
|
||||||
self.redirect("/panel/error?error=Invalid role name")
|
|
||||||
return
|
|
||||||
# does this user id exist?
|
|
||||||
if self.controller.roles.get_roleid_by_name(role_name) is not None:
|
|
||||||
self.redirect("/panel/error?error=Role exists")
|
|
||||||
return
|
|
||||||
|
|
||||||
servers = self.get_role_servers()
|
|
||||||
|
|
||||||
role_id = self.controller.roles.add_role_advanced(
|
|
||||||
role_name, servers, manager
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"created role {role_name} (RID:{role_id})",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.redirect("/panel/panel_config")
|
|
||||||
|
|
||||||
else:
|
|
||||||
self.set_status(404)
|
|
||||||
page_data = {
|
|
||||||
"lang": self.helper.get_setting("language"),
|
|
||||||
"lang_page": Helpers.get_lang_page(self.helper.get_setting("language")),
|
|
||||||
}
|
|
||||||
self.render(
|
|
||||||
"public/404.html", translate=self.translator.translate, data=page_data
|
|
||||||
)
|
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def delete(self, page):
|
|
||||||
api_key, _token_data, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
page_data = {
|
|
||||||
# todo: make this actually pull and compare version data
|
|
||||||
"update_available": False,
|
|
||||||
"version_data": self.helper.get_version_string(),
|
|
||||||
"user_data": exec_user,
|
|
||||||
"hosts_data": self.controller.management.get_latest_hosts_stats(),
|
|
||||||
"show_contribute": self.helper.get_setting("show_contribute_link", True),
|
|
||||||
"lang": self.controller.users.get_user_lang_by_id(exec_user["user_id"]),
|
|
||||||
"lang_page": Helpers.get_lang_page(
|
|
||||||
self.controller.users.get_user_lang_by_id(exec_user["user_id"])
|
|
||||||
),
|
|
||||||
}
|
|
||||||
|
|
||||||
if page == "remove_apikey":
|
|
||||||
key_id = bleach.clean(self.get_argument("id", None))
|
|
||||||
|
|
||||||
if not superuser:
|
|
||||||
self.redirect("/panel/error?error=Unauthorized access: not superuser")
|
|
||||||
return
|
|
||||||
if key_id is None or self.controller.users.get_user_api_key(key_id) is None:
|
|
||||||
self.redirect("/panel/error?error=Invalid Key ID")
|
|
||||||
return
|
|
||||||
# does this user id exist?
|
|
||||||
target_key = self.controller.users.get_user_api_key(key_id)
|
|
||||||
if not target_key:
|
|
||||||
self.redirect("/panel/error?error=Invalid Key ID")
|
|
||||||
return
|
|
||||||
|
|
||||||
key_obj = self.controller.users.get_user_api_key(key_id)
|
|
||||||
|
|
||||||
if key_obj.user_id != exec_user["user_id"] and not exec_user["superuser"]:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=You do not have access to change"
|
|
||||||
+ "this user's api key."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
self.controller.users.delete_user_api_key(key_id)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"Removed API key {target_key} "
|
|
||||||
f"(ID: {key_id}) from user {exec_user['user_id']}",
|
|
||||||
server_id=0,
|
|
||||||
source_ip=self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
self.finish()
|
|
||||||
self.redirect(f"/panel/edit_user_apikeys?id={key_obj.user_id}")
|
|
||||||
else:
|
|
||||||
self.set_status(404)
|
|
||||||
self.render(
|
|
||||||
"public/404.html",
|
|
||||||
data=page_data,
|
|
||||||
translate=self.translator.translate,
|
|
||||||
)
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import logging
|
import logging
|
||||||
import bleach
|
import nh3
|
||||||
|
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.models.users import HelperUsers
|
from app.classes.models.users import HelperUsers
|
||||||
@ -28,8 +28,8 @@ class PublicHandler(BaseHandler):
|
|||||||
# self.clear_cookie("user_data")
|
# self.clear_cookie("user_data")
|
||||||
|
|
||||||
def get(self, page=None):
|
def get(self, page=None):
|
||||||
error = bleach.clean(self.get_argument("error", "Invalid Login!"))
|
error = nh3.clean(self.get_argument("error", "Invalid Login!"))
|
||||||
error_msg = bleach.clean(self.get_argument("error_msg", ""))
|
error_msg = nh3.clean(self.get_argument("error_msg", ""))
|
||||||
|
|
||||||
page_data = {
|
page_data = {
|
||||||
"version": self.helper.get_version_string(),
|
"version": self.helper.get_version_string(),
|
||||||
@ -82,8 +82,8 @@ class PublicHandler(BaseHandler):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def post(self, page=None):
|
def post(self, page=None):
|
||||||
error = bleach.clean(self.get_argument("error", "Invalid Login!"))
|
error = nh3.clean(self.get_argument("error", "Invalid Login!"))
|
||||||
error_msg = bleach.clean(self.get_argument("error_msg", ""))
|
error_msg = nh3.clean(self.get_argument("error_msg", ""))
|
||||||
|
|
||||||
page_data = {
|
page_data = {
|
||||||
"version": self.helper.get_version_string(),
|
"version": self.helper.get_version_string(),
|
||||||
@ -100,8 +100,8 @@ class PublicHandler(BaseHandler):
|
|||||||
if self.request.query:
|
if self.request.query:
|
||||||
next_page = "/login?" + self.request.query
|
next_page = "/login?" + self.request.query
|
||||||
|
|
||||||
entered_username = bleach.clean(self.get_argument("username"))
|
entered_username = nh3.clean(self.get_argument("username"))
|
||||||
entered_password = bleach.clean(self.get_argument("password"))
|
entered_password = nh3.clean(self.get_argument("password"))
|
||||||
|
|
||||||
# pylint: disable=no-member
|
# pylint: disable=no-member
|
||||||
try:
|
try:
|
||||||
|
@ -12,6 +12,7 @@ from app.classes.web.routes.api.roles.index import ApiRolesIndexHandler
|
|||||||
from app.classes.web.routes.api.roles.role.index import ApiRolesRoleIndexHandler
|
from app.classes.web.routes.api.roles.role.index import ApiRolesRoleIndexHandler
|
||||||
from app.classes.web.routes.api.roles.role.servers import ApiRolesRoleServersHandler
|
from app.classes.web.routes.api.roles.role.servers import ApiRolesRoleServersHandler
|
||||||
from app.classes.web.routes.api.roles.role.users import ApiRolesRoleUsersHandler
|
from app.classes.web.routes.api.roles.role.users import ApiRolesRoleUsersHandler
|
||||||
|
|
||||||
from app.classes.web.routes.api.servers.index import ApiServersIndexHandler
|
from app.classes.web.routes.api.servers.index import ApiServersIndexHandler
|
||||||
from app.classes.web.routes.api.servers.server.action import (
|
from app.classes.web.routes.api.servers.server.action import (
|
||||||
ApiServersServerActionHandler,
|
ApiServersServerActionHandler,
|
||||||
@ -21,25 +22,63 @@ from app.classes.web.routes.api.servers.server.logs import ApiServersServerLogsH
|
|||||||
from app.classes.web.routes.api.servers.server.public import (
|
from app.classes.web.routes.api.servers.server.public import (
|
||||||
ApiServersServerPublicHandler,
|
ApiServersServerPublicHandler,
|
||||||
)
|
)
|
||||||
|
from app.classes.web.routes.api.servers.server.status import (
|
||||||
|
ApiServersServerStatusHandler,
|
||||||
|
)
|
||||||
from app.classes.web.routes.api.servers.server.stats import ApiServersServerStatsHandler
|
from app.classes.web.routes.api.servers.server.stats import ApiServersServerStatsHandler
|
||||||
|
from app.classes.web.routes.api.servers.server.history import (
|
||||||
|
ApiServersServerHistoryHandler,
|
||||||
|
)
|
||||||
from app.classes.web.routes.api.servers.server.stdin import ApiServersServerStdinHandler
|
from app.classes.web.routes.api.servers.server.stdin import ApiServersServerStdinHandler
|
||||||
from app.classes.web.routes.api.servers.server.tasks.index import (
|
from app.classes.web.routes.api.servers.server.tasks.index import (
|
||||||
ApiServersServerTasksIndexHandler,
|
ApiServersServerTasksIndexHandler,
|
||||||
)
|
)
|
||||||
|
from app.classes.web.routes.api.servers.server.backups.index import (
|
||||||
|
ApiServersServerBackupsIndexHandler,
|
||||||
|
)
|
||||||
|
from app.classes.web.routes.api.servers.server.backups.backup.index import (
|
||||||
|
ApiServersServerBackupsBackupIndexHandler,
|
||||||
|
)
|
||||||
|
from app.classes.web.routes.api.servers.server.files import (
|
||||||
|
ApiServersServerFilesIndexHandler,
|
||||||
|
ApiServersServerFilesCreateHandler,
|
||||||
|
ApiServersServerFilesZipHandler,
|
||||||
|
)
|
||||||
from app.classes.web.routes.api.servers.server.tasks.task.children import (
|
from app.classes.web.routes.api.servers.server.tasks.task.children import (
|
||||||
ApiServersServerTasksTaskChildrenHandler,
|
ApiServersServerTasksTaskChildrenHandler,
|
||||||
)
|
)
|
||||||
from app.classes.web.routes.api.servers.server.tasks.task.index import (
|
from app.classes.web.routes.api.servers.server.tasks.task.index import (
|
||||||
ApiServersServerTasksTaskIndexHandler,
|
ApiServersServerTasksTaskIndexHandler,
|
||||||
)
|
)
|
||||||
|
from app.classes.web.routes.api.servers.server.webhooks.index import (
|
||||||
|
ApiServersServerWebhooksIndexHandler,
|
||||||
|
)
|
||||||
|
from app.classes.web.routes.api.servers.server.webhooks.webhook.index import (
|
||||||
|
ApiServersServerWebhooksManagementIndexHandler,
|
||||||
|
)
|
||||||
from app.classes.web.routes.api.servers.server.users import ApiServersServerUsersHandler
|
from app.classes.web.routes.api.servers.server.users import ApiServersServerUsersHandler
|
||||||
from app.classes.web.routes.api.users.index import ApiUsersIndexHandler
|
from app.classes.web.routes.api.users.index import ApiUsersIndexHandler
|
||||||
from app.classes.web.routes.api.users.user.index import ApiUsersUserIndexHandler
|
from app.classes.web.routes.api.users.user.index import ApiUsersUserIndexHandler
|
||||||
from app.classes.web.routes.api.users.user.permissions import (
|
from app.classes.web.routes.api.users.user.permissions import (
|
||||||
ApiUsersUserPermissionsHandler,
|
ApiUsersUserPermissionsHandler,
|
||||||
)
|
)
|
||||||
|
from app.classes.web.routes.api.users.user.api import ApiUsersUserKeyHandler
|
||||||
from app.classes.web.routes.api.users.user.pfp import ApiUsersUserPfpHandler
|
from app.classes.web.routes.api.users.user.pfp import ApiUsersUserPfpHandler
|
||||||
from app.classes.web.routes.api.users.user.public import ApiUsersUserPublicHandler
|
from app.classes.web.routes.api.users.user.public import ApiUsersUserPublicHandler
|
||||||
|
from app.classes.web.routes.api.crafty.announcements.index import (
|
||||||
|
ApiAnnounceIndexHandler,
|
||||||
|
)
|
||||||
|
from app.classes.web.routes.api.crafty.config.index import (
|
||||||
|
ApiCraftyConfigIndexHandler,
|
||||||
|
ApiCraftyCustomizeIndexHandler,
|
||||||
|
)
|
||||||
|
from app.classes.web.routes.api.crafty.config.server_dir import (
|
||||||
|
ApiCraftyConfigServerDirHandler,
|
||||||
|
)
|
||||||
|
from app.classes.web.routes.api.crafty.stats.stats import ApiCraftyHostStatsHandler
|
||||||
|
from app.classes.web.routes.api.crafty.clogs.index import ApiCraftyLogIndexHandler
|
||||||
|
from app.classes.web.routes.api.crafty.imports.index import ApiImportFilesIndexHandler
|
||||||
|
from app.classes.web.routes.api.crafty.exe_cache import ApiCraftyJarCacheIndexHandler
|
||||||
|
|
||||||
|
|
||||||
def api_handlers(handler_args):
|
def api_handlers(handler_args):
|
||||||
@ -55,12 +94,62 @@ def api_handlers(handler_args):
|
|||||||
ApiAuthInvalidateTokensHandler,
|
ApiAuthInvalidateTokensHandler,
|
||||||
handler_args,
|
handler_args,
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/announcements/?",
|
||||||
|
ApiAnnounceIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/config/?",
|
||||||
|
ApiCraftyConfigIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/config/customize/?",
|
||||||
|
ApiCraftyCustomizeIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/config/servers_dir/?",
|
||||||
|
ApiCraftyConfigServerDirHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/stats/?",
|
||||||
|
ApiCraftyHostStatsHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/logs/([a-z0-9_]+)/?",
|
||||||
|
ApiCraftyLogIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/crafty/JarCache/?",
|
||||||
|
ApiCraftyJarCacheIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/import/file/unzip/?",
|
||||||
|
ApiImportFilesIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
# User routes
|
# User routes
|
||||||
(
|
(
|
||||||
r"/api/v2/users/?",
|
r"/api/v2/users/?",
|
||||||
ApiUsersIndexHandler,
|
ApiUsersIndexHandler,
|
||||||
handler_args,
|
handler_args,
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/users/([0-9]+)/key/?",
|
||||||
|
ApiUsersUserKeyHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/users/([0-9]+)/key/([0-9]+)/?",
|
||||||
|
ApiUsersUserKeyHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
(
|
(
|
||||||
r"/api/v2/users/([0-9]+)/?",
|
r"/api/v2/users/([0-9]+)/?",
|
||||||
ApiUsersUserIndexHandler,
|
ApiUsersUserIndexHandler,
|
||||||
@ -107,11 +196,41 @@ def api_handlers(handler_args):
|
|||||||
ApiServersIndexHandler,
|
ApiServersIndexHandler,
|
||||||
handler_args,
|
handler_args,
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/status/?",
|
||||||
|
ApiServersServerStatusHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
(
|
(
|
||||||
r"/api/v2/servers/([0-9]+)/?",
|
r"/api/v2/servers/([0-9]+)/?",
|
||||||
ApiServersServerIndexHandler,
|
ApiServersServerIndexHandler,
|
||||||
handler_args,
|
handler_args,
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/backups/?",
|
||||||
|
ApiServersServerBackupsIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/backups/backup/?",
|
||||||
|
ApiServersServerBackupsBackupIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/files/?",
|
||||||
|
ApiServersServerFilesIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/files/create/?",
|
||||||
|
ApiServersServerFilesCreateHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/files/zip/?",
|
||||||
|
ApiServersServerFilesZipHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
(
|
(
|
||||||
r"/api/v2/servers/([0-9]+)/tasks/?",
|
r"/api/v2/servers/([0-9]+)/tasks/?",
|
||||||
ApiServersServerTasksIndexHandler,
|
ApiServersServerTasksIndexHandler,
|
||||||
@ -132,6 +251,21 @@ def api_handlers(handler_args):
|
|||||||
ApiServersServerStatsHandler,
|
ApiServersServerStatsHandler,
|
||||||
handler_args,
|
handler_args,
|
||||||
),
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/history/?",
|
||||||
|
ApiServersServerHistoryHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/webhook/([0-9]+)/?",
|
||||||
|
ApiServersServerWebhooksManagementIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/api/v2/servers/([0-9]+)/webhook/?",
|
||||||
|
ApiServersServerWebhooksIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
(
|
(
|
||||||
r"/api/v2/servers/([0-9]+)/action/([a-z_]+)/?",
|
r"/api/v2/servers/([0-9]+)/action/([a-z_]+)/?",
|
||||||
ApiServersServerActionHandler,
|
ApiServersServerActionHandler,
|
||||||
|
110
app/classes/web/routes/api/crafty/announcements/index.py
Normal file
110
app/classes/web/routes/api/crafty/announcements/index.py
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
import logging
|
||||||
|
import json
|
||||||
|
from jsonschema import ValidationError, validate
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
notif_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"id": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiAnnounceIndexHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_exec_user_crafty_permissions,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_user,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
data = self.helper.get_announcements()
|
||||||
|
cleared = str(
|
||||||
|
self.controller.users.get_user_by_id(auth_data[4]["user_id"])[
|
||||||
|
"cleared_notifs"
|
||||||
|
]
|
||||||
|
).split(",")
|
||||||
|
res = [d.get("id", None) for d in data]
|
||||||
|
# remove notifs that are no longer in Crafty.
|
||||||
|
for item in cleared[:]:
|
||||||
|
if item not in res:
|
||||||
|
cleared.remove(item)
|
||||||
|
updata = {"cleared_notifs": ",".join(cleared)}
|
||||||
|
self.controller.users.update_user(auth_data[4]["user_id"], updata)
|
||||||
|
if len(cleared) > 0:
|
||||||
|
for item in data[:]:
|
||||||
|
if item["id"] in cleared:
|
||||||
|
data.remove(item)
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": data,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def post(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_exec_user_crafty_permissions,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_user,
|
||||||
|
) = auth_data
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, notif_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
announcements = self.helper.get_announcements()
|
||||||
|
res = [d.get("id", None) for d in announcements]
|
||||||
|
cleared_notifs = str(
|
||||||
|
self.controller.users.get_user_by_id(auth_data[4]["user_id"])[
|
||||||
|
"cleared_notifs"
|
||||||
|
]
|
||||||
|
).split(",")
|
||||||
|
# remove notifs that are no longer in Crafty.
|
||||||
|
for item in cleared_notifs[:]:
|
||||||
|
if item not in res:
|
||||||
|
cleared_notifs.remove(item)
|
||||||
|
if str(data["id"]) in str(res):
|
||||||
|
cleared_notifs.append(data["id"])
|
||||||
|
else:
|
||||||
|
self.finish_json(200, {"status": "error", "error": "INVALID_DATA"})
|
||||||
|
return
|
||||||
|
updata = {"cleared_notifs": ",".join(cleared_notifs)}
|
||||||
|
self.controller.users.update_user(auth_data[4]["user_id"], updata)
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": {},
|
||||||
|
},
|
||||||
|
)
|
34
app/classes/web/routes/api/crafty/clogs/index.py
Normal file
34
app/classes/web/routes/api/crafty/clogs/index.py
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
|
||||||
|
class ApiCraftyLogIndexHandler(BaseApiHandler):
|
||||||
|
def get(self, log_type: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
superuser,
|
||||||
|
_,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
if not superuser:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
log_types = ["audit", "session", "schedule"]
|
||||||
|
if log_type not in log_types:
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
if log_type == "audit":
|
||||||
|
return self.finish_json(
|
||||||
|
200,
|
||||||
|
{"status": "ok", "data": self.controller.management.get_activity_log()},
|
||||||
|
)
|
||||||
|
|
||||||
|
if log_type == "session":
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
if log_type == "schedule":
|
||||||
|
raise NotImplementedError
|
312
app/classes/web/routes/api/crafty/config/index.py
Normal file
312
app/classes/web/routes/api/crafty/config/index.py
Normal file
@ -0,0 +1,312 @@
|
|||||||
|
import os
|
||||||
|
import json
|
||||||
|
from jsonschema import ValidationError, validate
|
||||||
|
import orjson
|
||||||
|
from playhouse.shortcuts import model_to_dict
|
||||||
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
config_json_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"http_port": {"type": "integer"},
|
||||||
|
"https_port": {"type": "integer"},
|
||||||
|
"language": {
|
||||||
|
"type": "string",
|
||||||
|
},
|
||||||
|
"cookie_expire": {"type": "integer"},
|
||||||
|
"show_errors": {"type": "boolean"},
|
||||||
|
"history_max_age": {"type": "integer"},
|
||||||
|
"stats_update_frequency_seconds": {"type": "integer"},
|
||||||
|
"delete_default_json": {"type": "boolean"},
|
||||||
|
"show_contribute_link": {"type": "boolean"},
|
||||||
|
"virtual_terminal_lines": {"type": "integer"},
|
||||||
|
"max_log_lines": {"type": "integer"},
|
||||||
|
"max_audit_entries": {"type": "integer"},
|
||||||
|
"disabled_language_files": {"type": "array"},
|
||||||
|
"stream_size_GB": {"type": "integer"},
|
||||||
|
"keywords": {"type": "array"},
|
||||||
|
"allow_nsfw_profile_pictures": {"type": "boolean"},
|
||||||
|
"enable_user_self_delete": {"type": "boolean"},
|
||||||
|
"reset_secrets_on_next_boot": {"type": "boolean"},
|
||||||
|
"monitored_mounts": {"type": "array"},
|
||||||
|
"dir_size_poll_freq_minutes": {"type": "integer"},
|
||||||
|
"crafty_logs_delete_after_days": {"type": "integer"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
customize_json_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"photo": {"type": "string"},
|
||||||
|
"opacity": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
photo_delete_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"photo": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
DEFAULT_PHOTO = "login_1.jpg"
|
||||||
|
|
||||||
|
|
||||||
|
class ApiCraftyConfigIndexHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
superuser,
|
||||||
|
_,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
# GET /api/v2/roles?ids=true
|
||||||
|
get_only_ids = self.get_query_argument("ids", None) == "true"
|
||||||
|
|
||||||
|
if not superuser:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": self.controller.roles.get_all_role_ids()
|
||||||
|
if get_only_ids
|
||||||
|
else [model_to_dict(r) for r in self.controller.roles.get_all_roles()],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def patch(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
superuser,
|
||||||
|
user,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
if not superuser:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = orjson.loads(self.request.body)
|
||||||
|
except orjson.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, config_json_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
self.controller.set_config_json(data)
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
user["user_id"],
|
||||||
|
"edited config.json",
|
||||||
|
server_id=0,
|
||||||
|
source_ip=self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{"status": "ok"},
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiCraftyCustomizeIndexHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
superuser,
|
||||||
|
_,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
# GET /api/v2/roles?ids=true
|
||||||
|
get_only_ids = self.get_query_argument("ids", None) == "true"
|
||||||
|
|
||||||
|
if not superuser:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": self.controller.roles.get_all_role_ids()
|
||||||
|
if get_only_ids
|
||||||
|
else [model_to_dict(r) for r in self.controller.roles.get_all_roles()],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def patch(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
superuser,
|
||||||
|
user,
|
||||||
|
) = auth_data
|
||||||
|
if not superuser:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = orjson.loads(self.request.body)
|
||||||
|
except orjson.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, customize_json_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if not self.helper.validate_traversal(
|
||||||
|
os.path.join(
|
||||||
|
self.controller.project_root,
|
||||||
|
"app/frontend/static/assets/images/auth/",
|
||||||
|
),
|
||||||
|
os.path.join(
|
||||||
|
self.controller.project_root,
|
||||||
|
f"app/frontend/static/assets/images/auth/{data['photo']}",
|
||||||
|
),
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": "TRIED TO REACH FILES THAT ARE"
|
||||||
|
" NOT SUPPOSED TO BE ACCESSIBLE",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
user["user_id"],
|
||||||
|
f"customized login photo: {data['photo']}/{data['opacity']}",
|
||||||
|
server_id=0,
|
||||||
|
source_ip=self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
self.controller.management.set_login_opacity(int(data["opacity"]))
|
||||||
|
if data["photo"] == DEFAULT_PHOTO:
|
||||||
|
self.controller.management.set_login_image(DEFAULT_PHOTO)
|
||||||
|
self.controller.cached_login = f"{data['photo']}"
|
||||||
|
else:
|
||||||
|
self.controller.management.set_login_image(f"custom/{data['photo']}")
|
||||||
|
self.controller.cached_login = f"custom/{data['photo']}"
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": {"photo": data["photo"], "opacity": data["opacity"]},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def delete(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if not auth_data[4]["superuser"]:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, photo_delete_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if not self.helper.validate_traversal(
|
||||||
|
os.path.join(
|
||||||
|
self.controller.project_root,
|
||||||
|
"app",
|
||||||
|
"frontend",
|
||||||
|
"/static/assets/images/auth/",
|
||||||
|
),
|
||||||
|
os.path.join(
|
||||||
|
self.controller.project_root,
|
||||||
|
"app",
|
||||||
|
"frontend",
|
||||||
|
"/static/assets/images/auth/",
|
||||||
|
data["photo"],
|
||||||
|
),
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": "TRIED TO REACH FILES THAT ARE"
|
||||||
|
" NOT SUPPOSED TO BE ACCESSIBLE",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if data["photo"] == DEFAULT_PHOTO:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID FILE",
|
||||||
|
"error_data": "CANNOT DELETE DEFAULT",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
FileHelpers.del_file(
|
||||||
|
os.path.join(
|
||||||
|
self.controller.project_root,
|
||||||
|
f"app/frontend/static/assets/images/auth/custom/{data['photo']}",
|
||||||
|
)
|
||||||
|
)
|
||||||
|
current = self.controller.cached_login
|
||||||
|
split = current.split("/")
|
||||||
|
if len(split) == 1:
|
||||||
|
current_photo = current
|
||||||
|
else:
|
||||||
|
current_photo = split[1]
|
||||||
|
if current_photo == data["photo"]:
|
||||||
|
self.controller.management.set_login_image(DEFAULT_PHOTO)
|
||||||
|
self.controller.cached_login = DEFAULT_PHOTO
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
115
app/classes/web/routes/api/crafty/config/server_dir.py
Normal file
115
app/classes/web/routes/api/crafty/config/server_dir.py
Normal file
@ -0,0 +1,115 @@
|
|||||||
|
from jsonschema import ValidationError, validate
|
||||||
|
import orjson
|
||||||
|
from playhouse.shortcuts import model_to_dict
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
server_dir_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"new_dir": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiCraftyConfigServerDirHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
superuser,
|
||||||
|
_,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
# GET /api/v2/roles?ids=true
|
||||||
|
get_only_ids = self.get_query_argument("ids", None) == "true"
|
||||||
|
|
||||||
|
if not superuser:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": self.controller.roles.get_all_role_ids()
|
||||||
|
if get_only_ids
|
||||||
|
else [model_to_dict(r) for r in self.controller.roles.get_all_roles()],
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def patch(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
if not auth_data:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if not auth_data[4]["superuser"]:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
if self.helper.is_env_docker():
|
||||||
|
raise NotImplementedError
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = orjson.loads(self.request.body)
|
||||||
|
except orjson.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, server_dir_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if self.helper.dir_migration:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "IN PROGRESS",
|
||||||
|
"error_data": "Migration already in progress. Please be patient",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
for server in self.controller.servers.get_all_servers_stats():
|
||||||
|
if server["stats"]["running"]:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "SERVER RUNNING",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
new_dir = data["new_dir"]
|
||||||
|
self.controller.update_master_server_dir(new_dir, auth_data[4]["user_id"])
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"updated master servers dir to {new_dir}/servers",
|
||||||
|
server_id=0,
|
||||||
|
source_ip=self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{"status": "ok"},
|
||||||
|
)
|
27
app/classes/web/routes/api/crafty/exe_cache.py
Normal file
27
app/classes/web/routes/api/crafty/exe_cache.py
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
|
||||||
|
class ApiCraftyJarCacheIndexHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
if not auth_data[4]["superuser"]:
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.controller.server_jars.manual_refresh_cache()
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": self.controller.server_jars.get_serverjar_data(),
|
||||||
|
},
|
||||||
|
)
|
130
app/classes/web/routes/api/crafty/imports/index.py
Normal file
130
app/classes/web/routes/api/crafty/imports/index.py
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
import os
|
||||||
|
import logging
|
||||||
|
import json
|
||||||
|
import html
|
||||||
|
from jsonschema import validate
|
||||||
|
from jsonschema.exceptions import ValidationError
|
||||||
|
from app.classes.models.crafty_permissions import EnumPermissionsCrafty
|
||||||
|
from app.classes.shared.helpers import Helpers
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
files_get_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"page": {"type": "string", "minLength": 1},
|
||||||
|
"folder": {"type": "string"},
|
||||||
|
"upload": {"type": "boolean", "default": "False"},
|
||||||
|
"unzip": {"type": "boolean", "default": "True"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiImportFilesIndexHandler(BaseApiHandler):
|
||||||
|
def post(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsCrafty.SERVER_CREATION
|
||||||
|
not in self.controller.crafty_perms.get_crafty_permissions_list(
|
||||||
|
auth_data[4]["user_id"]
|
||||||
|
)
|
||||||
|
and not auth_data[4]["superuser"]
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files or Backup permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_get_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
# TODO: limit some columns for specific permissions?
|
||||||
|
folder = data["folder"]
|
||||||
|
user_id = auth_data[4]["user_id"]
|
||||||
|
root_path = False
|
||||||
|
if data["unzip"]:
|
||||||
|
# This is awful. Once uploads go to return
|
||||||
|
# JSON we need to remove this and just send
|
||||||
|
# the path.
|
||||||
|
if data["upload"]:
|
||||||
|
folder = os.path.join(
|
||||||
|
self.controller.project_root, "import", "upload", folder
|
||||||
|
)
|
||||||
|
if Helpers.check_file_exists(folder):
|
||||||
|
folder = self.file_helper.unzip_server(folder, user_id)
|
||||||
|
root_path = True
|
||||||
|
else:
|
||||||
|
if user_id:
|
||||||
|
user_lang = self.controller.users.get_user_lang_by_id(user_id)
|
||||||
|
self.helper.websocket_helper.broadcast_user(
|
||||||
|
user_id,
|
||||||
|
"send_start_error",
|
||||||
|
{
|
||||||
|
"error": self.helper.translation.translate(
|
||||||
|
"error", "no-file", user_lang
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
if not self.helper.check_path_exists(folder) and user_id:
|
||||||
|
user_lang = self.controller.users.get_user_lang_by_id(user_id)
|
||||||
|
self.helper.websocket_helper.broadcast_user(
|
||||||
|
user_id,
|
||||||
|
"send_start_error",
|
||||||
|
{
|
||||||
|
"error": self.helper.translation.translate(
|
||||||
|
"error", "no-file", user_lang
|
||||||
|
)
|
||||||
|
},
|
||||||
|
)
|
||||||
|
return_json = {
|
||||||
|
"root_path": {
|
||||||
|
"path": folder,
|
||||||
|
"top": root_path,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dir_list = []
|
||||||
|
unsorted_files = []
|
||||||
|
file_list = os.listdir(folder)
|
||||||
|
for item in file_list:
|
||||||
|
if os.path.isdir(os.path.join(folder, item)):
|
||||||
|
dir_list.append(item)
|
||||||
|
else:
|
||||||
|
unsorted_files.append(item)
|
||||||
|
file_list = sorted(dir_list, key=str.casefold) + sorted(
|
||||||
|
unsorted_files, key=str.casefold
|
||||||
|
)
|
||||||
|
for raw_filename in file_list:
|
||||||
|
filename = html.escape(raw_filename)
|
||||||
|
rel = os.path.join(folder, raw_filename)
|
||||||
|
dpath = os.path.join(folder, filename)
|
||||||
|
dpath = self.helper.wtol_path(dpath)
|
||||||
|
if os.path.isdir(rel):
|
||||||
|
return_json[filename] = {
|
||||||
|
"path": dpath,
|
||||||
|
"dir": True,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
return_json[filename] = {
|
||||||
|
"path": dpath,
|
||||||
|
"dir": False,
|
||||||
|
}
|
||||||
|
self.finish_json(200, {"status": "ok", "data": return_json})
|
21
app/classes/web/routes/api/crafty/stats/stats.py
Normal file
21
app/classes/web/routes/api/crafty/stats/stats.py
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import logging
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiCraftyHostStatsHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
latest = self.controller.management.get_latest_hosts_stats()
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": latest,
|
||||||
|
},
|
||||||
|
)
|
@ -28,9 +28,39 @@ create_role_schema = {
|
|||||||
"required": ["server_id", "permissions"],
|
"required": ["server_id", "permissions"],
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
"manager": {"type": ["integer", "null"]},
|
||||||
},
|
},
|
||||||
"required": ["name"],
|
|
||||||
"additionalProperties": False,
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
basic_create_role_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"name": {
|
||||||
|
"type": "string",
|
||||||
|
"minLength": 1,
|
||||||
|
},
|
||||||
|
"servers": {
|
||||||
|
"type": "array",
|
||||||
|
"items": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"server_id": {
|
||||||
|
"type": "integer",
|
||||||
|
"minimum": 1,
|
||||||
|
},
|
||||||
|
"permissions": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[01]{8}$", # 8 bits, see EnumPermissionsServer
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"required": ["server_id", "permissions"],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -86,7 +116,10 @@ class ApiRolesIndexHandler(BaseApiHandler):
|
|||||||
)
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
validate(data, create_role_schema)
|
if auth_data[4]["superuser"]:
|
||||||
|
validate(data, create_role_schema)
|
||||||
|
else:
|
||||||
|
validate(data, basic_create_role_schema)
|
||||||
except ValidationError as e:
|
except ValidationError as e:
|
||||||
return self.finish_json(
|
return self.finish_json(
|
||||||
400,
|
400,
|
||||||
@ -98,6 +131,9 @@ class ApiRolesIndexHandler(BaseApiHandler):
|
|||||||
)
|
)
|
||||||
|
|
||||||
role_name = data["name"]
|
role_name = data["name"]
|
||||||
|
manager = data.get("manager", None)
|
||||||
|
if manager == self.controller.users.get_id_by_name("SYSTEM") or manager == 0:
|
||||||
|
manager = None
|
||||||
|
|
||||||
# Get the servers
|
# Get the servers
|
||||||
servers_dict = {server["server_id"]: server for server in data["servers"]}
|
servers_dict = {server["server_id"]: server for server in data["servers"]}
|
||||||
@ -116,9 +152,7 @@ class ApiRolesIndexHandler(BaseApiHandler):
|
|||||||
400, {"status": "error", "error": "ROLE_NAME_ALREADY_EXISTS"}
|
400, {"status": "error", "error": "ROLE_NAME_ALREADY_EXISTS"}
|
||||||
)
|
)
|
||||||
|
|
||||||
role_id = self.controller.roles.add_role_advanced(
|
role_id = self.controller.roles.add_role_advanced(role_name, servers, manager)
|
||||||
role_name, servers, user["user_id"]
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
self.controller.management.add_to_audit_log(
|
||||||
user["user_id"],
|
user["user_id"],
|
||||||
|
@ -153,9 +153,18 @@ class ApiRolesRoleIndexHandler(BaseApiHandler):
|
|||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
manager = data.get(
|
||||||
|
"manager", self.controller.roles.get_role(role_id)["manager"]
|
||||||
|
)
|
||||||
|
if manager == self.controller.users.get_id_by_name("system") or manager == 0:
|
||||||
|
manager = None
|
||||||
|
|
||||||
try:
|
try:
|
||||||
self.controller.roles.update_role_advanced(
|
self.controller.roles.update_role_advanced(
|
||||||
role_id, data.get("role_name", None), data.get("servers", None)
|
role_id,
|
||||||
|
data.get("name", None),
|
||||||
|
data.get("servers", None),
|
||||||
|
manager,
|
||||||
)
|
)
|
||||||
except DoesNotExist:
|
except DoesNotExist:
|
||||||
return self.finish_json(404, {"status": "error", "error": "ROLE_NOT_FOUND"})
|
return self.finish_json(404, {"status": "error", "error": "ROLE_NOT_FOUND"})
|
||||||
|
@ -9,7 +9,7 @@ logger = logging.getLogger(__name__)
|
|||||||
|
|
||||||
new_server_schema = {
|
new_server_schema = {
|
||||||
"definitions": {},
|
"definitions": {},
|
||||||
"$schema": "http://json-schema.org/draft-07/schema#",
|
"$schema": "https://json-schema.org/draft-07/schema#",
|
||||||
"title": "Root",
|
"title": "Root",
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"required": [
|
"required": [
|
||||||
@ -24,6 +24,7 @@ new_server_schema = {
|
|||||||
"examples": ["My Server"],
|
"examples": ["My Server"],
|
||||||
"minLength": 2,
|
"minLength": 2,
|
||||||
},
|
},
|
||||||
|
"roles": {"title": "Roles to add", "type": "array", "examples": [1, 2, 3]},
|
||||||
"stop_command": {
|
"stop_command": {
|
||||||
"title": "Stop command",
|
"title": "Stop command",
|
||||||
"description": '"" means the default for the server creation type.',
|
"description": '"" means the default for the server creation type.',
|
||||||
@ -133,8 +134,13 @@ new_server_schema = {
|
|||||||
"mem_min",
|
"mem_min",
|
||||||
"mem_max",
|
"mem_max",
|
||||||
"server_properties_port",
|
"server_properties_port",
|
||||||
"agree_to_eula",
|
"category",
|
||||||
],
|
],
|
||||||
|
"category": {
|
||||||
|
"title": "Jar Category",
|
||||||
|
"type": "string",
|
||||||
|
"examples": ["modded", "vanilla"],
|
||||||
|
},
|
||||||
"properties": {
|
"properties": {
|
||||||
"type": {
|
"type": {
|
||||||
"title": "Server JAR Type",
|
"title": "Server JAR Type",
|
||||||
@ -185,7 +191,6 @@ new_server_schema = {
|
|||||||
"mem_min",
|
"mem_min",
|
||||||
"mem_max",
|
"mem_max",
|
||||||
"server_properties_port",
|
"server_properties_port",
|
||||||
"agree_to_eula",
|
|
||||||
],
|
],
|
||||||
"properties": {
|
"properties": {
|
||||||
"existing_server_path": {
|
"existing_server_path": {
|
||||||
@ -240,7 +245,6 @@ new_server_schema = {
|
|||||||
"mem_min",
|
"mem_min",
|
||||||
"mem_max",
|
"mem_max",
|
||||||
"server_properties_port",
|
"server_properties_port",
|
||||||
"agree_to_eula",
|
|
||||||
],
|
],
|
||||||
"properties": {
|
"properties": {
|
||||||
"zip_path": {
|
"zip_path": {
|
||||||
@ -336,12 +340,24 @@ new_server_schema = {
|
|||||||
"title": "Creation type",
|
"title": "Creation type",
|
||||||
"type": "string",
|
"type": "string",
|
||||||
"default": "import_server",
|
"default": "import_server",
|
||||||
"enum": ["import_server", "import_zip"],
|
"enum": ["download_exe", "import_server", "import_zip"],
|
||||||
|
},
|
||||||
|
"download_exe_create_data": {
|
||||||
|
"title": "Import server data",
|
||||||
|
"type": "object",
|
||||||
|
"required": [],
|
||||||
|
"properties": {
|
||||||
|
"agree_to_eula": {
|
||||||
|
"title": "Agree to the EULA",
|
||||||
|
"type": "boolean",
|
||||||
|
"enum": [True],
|
||||||
|
},
|
||||||
|
},
|
||||||
},
|
},
|
||||||
"import_server_create_data": {
|
"import_server_create_data": {
|
||||||
"title": "Import server data",
|
"title": "Import server data",
|
||||||
"type": "object",
|
"type": "object",
|
||||||
"required": ["existing_server_path", "command"],
|
"required": ["existing_server_path", "executable"],
|
||||||
"properties": {
|
"properties": {
|
||||||
"existing_server_path": {
|
"existing_server_path": {
|
||||||
"title": "Server path",
|
"title": "Server path",
|
||||||
@ -350,6 +366,14 @@ new_server_schema = {
|
|||||||
"examples": ["/var/opt/server"],
|
"examples": ["/var/opt/server"],
|
||||||
"minLength": 1,
|
"minLength": 1,
|
||||||
},
|
},
|
||||||
|
"executable": {
|
||||||
|
"title": "Executable File",
|
||||||
|
"description": "File Crafty should execute"
|
||||||
|
"on server launch",
|
||||||
|
"type": "string",
|
||||||
|
"examples": ["bedrock_server.exe"],
|
||||||
|
"minlength": 1,
|
||||||
|
},
|
||||||
"command": {
|
"command": {
|
||||||
"title": "Command",
|
"title": "Command",
|
||||||
"type": "string",
|
"type": "string",
|
||||||
@ -371,6 +395,14 @@ new_server_schema = {
|
|||||||
"examples": ["/var/opt/server.zip"],
|
"examples": ["/var/opt/server.zip"],
|
||||||
"minLength": 1,
|
"minLength": 1,
|
||||||
},
|
},
|
||||||
|
"executable": {
|
||||||
|
"title": "Executable File",
|
||||||
|
"description": "File Crafty should execute"
|
||||||
|
"on server launch",
|
||||||
|
"type": "string",
|
||||||
|
"examples": ["bedrock_server.exe"],
|
||||||
|
"minlength": 1,
|
||||||
|
},
|
||||||
"zip_root": {
|
"zip_root": {
|
||||||
"title": "Server root directory",
|
"title": "Server root directory",
|
||||||
"description": "The server root in the ZIP archive",
|
"description": "The server root in the ZIP archive",
|
||||||
@ -394,7 +426,9 @@ new_server_schema = {
|
|||||||
"allOf": [
|
"allOf": [
|
||||||
{
|
{
|
||||||
"if": {
|
"if": {
|
||||||
"properties": {"create_type": {"const": "import_exec"}}
|
"properties": {
|
||||||
|
"create_type": {"const": "import_server"}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"then": {"required": ["import_server_create_data"]},
|
"then": {"required": ["import_server_create_data"]},
|
||||||
},
|
},
|
||||||
@ -404,6 +438,16 @@ new_server_schema = {
|
|||||||
},
|
},
|
||||||
"then": {"required": ["import_zip_create_data"]},
|
"then": {"required": ["import_zip_create_data"]},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"if": {
|
||||||
|
"properties": {"create_type": {"const": "download_exe"}}
|
||||||
|
},
|
||||||
|
"then": {
|
||||||
|
"required": [
|
||||||
|
"download_exe_create_data",
|
||||||
|
]
|
||||||
|
},
|
||||||
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -411,6 +455,7 @@ new_server_schema = {
|
|||||||
"oneOf": [
|
"oneOf": [
|
||||||
{"required": ["import_server_create_data"]},
|
{"required": ["import_server_create_data"]},
|
||||||
{"required": ["import_zip_create_data"]},
|
{"required": ["import_zip_create_data"]},
|
||||||
|
{"required": ["download_exe_create_data"]},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -651,7 +696,6 @@ class ApiServersIndexHandler(BaseApiHandler):
|
|||||||
return self.finish_json(
|
return self.finish_json(
|
||||||
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
)
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
validate(data, new_server_schema)
|
validate(data, new_server_schema)
|
||||||
except ValidationError as e:
|
except ValidationError as e:
|
||||||
|
@ -31,6 +31,8 @@ class ApiServersServerActionHandler(BaseApiHandler):
|
|||||||
|
|
||||||
if action == "clone_server":
|
if action == "clone_server":
|
||||||
return self._clone_server(server_id, auth_data[4]["user_id"])
|
return self._clone_server(server_id, auth_data[4]["user_id"])
|
||||||
|
if action == "eula":
|
||||||
|
return self._agree_eula(server_id, auth_data[4]["user_id"])
|
||||||
|
|
||||||
self.controller.management.send_command(
|
self.controller.management.send_command(
|
||||||
auth_data[4]["user_id"], server_id, self.get_remote_ip(), action
|
auth_data[4]["user_id"], server_id, self.get_remote_ip(), action
|
||||||
@ -41,6 +43,11 @@ class ApiServersServerActionHandler(BaseApiHandler):
|
|||||||
{"status": "ok"},
|
{"status": "ok"},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def _agree_eula(self, server_id, user):
|
||||||
|
svr = self.controller.servers.get_server_instance_by_id(server_id)
|
||||||
|
svr.agree_eula(user)
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
def _clone_server(self, server_id, user_id):
|
def _clone_server(self, server_id, user_id):
|
||||||
def is_name_used(name):
|
def is_name_used(name):
|
||||||
return Servers.select().where(Servers.server_name == name).exists()
|
return Servers.select().where(Servers.server_name == name).exists()
|
||||||
|
@ -0,0 +1,217 @@
|
|||||||
|
import logging
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
from apscheduler.jobstores.base import JobLookupError
|
||||||
|
from jsonschema import validate
|
||||||
|
from jsonschema.exceptions import ValidationError
|
||||||
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
from app.classes.shared.helpers import Helpers
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
backup_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"filename": {"type": "string", "minLength": 5},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerBackupsBackupIndexHandler(BaseApiHandler):
|
||||||
|
def get(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.BACKUP
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
self.finish_json(200, self.controller.management.get_backup_config(server_id))
|
||||||
|
|
||||||
|
def delete(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
backup_conf = self.controller.management.get_backup_config(server_id)
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.BACKUP
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, backup_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
FileHelpers.del_file(
|
||||||
|
os.path.join(backup_conf["backup_path"], data["filename"])
|
||||||
|
)
|
||||||
|
except Exception:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "NO BACKUP FOUND"}
|
||||||
|
)
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Edited server {server_id}: removed backup {data['filename']}",
|
||||||
|
server_id,
|
||||||
|
self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
def post(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.BACKUP
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, backup_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
svr_obj = self.controller.servers.get_server_obj(server_id)
|
||||||
|
server_data = self.controller.servers.get_server_data_by_id(server_id)
|
||||||
|
zip_name = data["filename"]
|
||||||
|
# import the server again based on zipfile
|
||||||
|
backup_path = svr_obj.backup_path
|
||||||
|
if Helpers.validate_traversal(backup_path, zip_name):
|
||||||
|
temp_dir = Helpers.unzip_backup_archive(backup_path, zip_name)
|
||||||
|
if server_data["type"] == "minecraft-java":
|
||||||
|
new_server = self.controller.restore_java_zip_server(
|
||||||
|
svr_obj.server_name,
|
||||||
|
temp_dir,
|
||||||
|
server_data["executable"],
|
||||||
|
"1",
|
||||||
|
"2",
|
||||||
|
server_data["server_port"],
|
||||||
|
server_data["created_by"],
|
||||||
|
)
|
||||||
|
elif server_data["type"] == "minecraft-bedrock":
|
||||||
|
new_server = self.controller.restore_bedrock_zip_server(
|
||||||
|
svr_obj.server_name,
|
||||||
|
temp_dir,
|
||||||
|
server_data["executable"],
|
||||||
|
server_data["server_port"],
|
||||||
|
server_data["created_by"],
|
||||||
|
)
|
||||||
|
new_server_id = new_server
|
||||||
|
new_server = self.controller.servers.get_server_data(new_server)
|
||||||
|
self.controller.rename_backup_dir(
|
||||||
|
server_id, new_server_id, new_server["server_uuid"]
|
||||||
|
)
|
||||||
|
# preserve current schedules
|
||||||
|
for schedule in self.controller.management.get_schedules_by_server(
|
||||||
|
server_id
|
||||||
|
):
|
||||||
|
job_data = self.controller.management.get_scheduled_task(
|
||||||
|
schedule.schedule_id
|
||||||
|
)
|
||||||
|
job_data["server_id"] = new_server_id
|
||||||
|
del job_data["schedule_id"]
|
||||||
|
self.tasks_manager.update_job(schedule.schedule_id, job_data)
|
||||||
|
# preserve execution command
|
||||||
|
new_server_obj = self.controller.servers.get_server_obj(new_server_id)
|
||||||
|
new_server_obj.execution_command = server_data["execution_command"]
|
||||||
|
# reset executable path
|
||||||
|
if svr_obj.path in svr_obj.executable:
|
||||||
|
new_server_obj.executable = str(svr_obj.executable).replace(
|
||||||
|
svr_obj.path, new_server_obj.path
|
||||||
|
)
|
||||||
|
# reset run command path
|
||||||
|
if svr_obj.path in svr_obj.execution_command:
|
||||||
|
new_server_obj.execution_command = str(
|
||||||
|
svr_obj.execution_command
|
||||||
|
).replace(svr_obj.path, new_server_obj.path)
|
||||||
|
# reset log path
|
||||||
|
if svr_obj.path in svr_obj.log_path:
|
||||||
|
new_server_obj.log_path = str(svr_obj.log_path).replace(
|
||||||
|
svr_obj.path, new_server_obj.path
|
||||||
|
)
|
||||||
|
self.controller.servers.update_server(new_server_obj)
|
||||||
|
|
||||||
|
# preserve backup config
|
||||||
|
backup_config = self.controller.management.get_backup_config(server_id)
|
||||||
|
excluded_dirs = []
|
||||||
|
server_obj = self.controller.servers.get_server_obj(server_id)
|
||||||
|
loop_backup_path = self.helper.wtol_path(server_obj.path)
|
||||||
|
for item in self.controller.management.get_excluded_backup_dirs(
|
||||||
|
server_id
|
||||||
|
):
|
||||||
|
item_path = self.helper.wtol_path(item)
|
||||||
|
bu_path = os.path.relpath(item_path, loop_backup_path)
|
||||||
|
bu_path = os.path.join(new_server_obj.path, bu_path)
|
||||||
|
excluded_dirs.append(bu_path)
|
||||||
|
self.controller.management.set_backup_config(
|
||||||
|
new_server_id,
|
||||||
|
new_server_obj.backup_path,
|
||||||
|
backup_config["max_backups"],
|
||||||
|
excluded_dirs,
|
||||||
|
backup_config["compress"],
|
||||||
|
backup_config["shutdown"],
|
||||||
|
)
|
||||||
|
# remove old server's tasks
|
||||||
|
try:
|
||||||
|
self.tasks_manager.remove_all_server_tasks(server_id)
|
||||||
|
except JobLookupError as e:
|
||||||
|
logger.info("No active tasks found for server: {e}")
|
||||||
|
self.controller.remove_server(server_id, True)
|
||||||
|
except Exception as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": f"NO BACKUP FOUND {e}"}
|
||||||
|
)
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Restored server {server_id} backup {data['filename']}",
|
||||||
|
server_id,
|
||||||
|
self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
123
app/classes/web/routes/api/servers/server/backups/index.py
Normal file
123
app/classes/web/routes/api/servers/server/backups/index.py
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
import logging
|
||||||
|
import json
|
||||||
|
from jsonschema import validate
|
||||||
|
from jsonschema.exceptions import ValidationError
|
||||||
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
backup_patch_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"backup_path": {"type": "string", "minLength": 1},
|
||||||
|
"max_backups": {"type": "integer"},
|
||||||
|
"compress": {"type": "boolean"},
|
||||||
|
"shutdown": {"type": "boolean"},
|
||||||
|
"backup_before": {"type": "string"},
|
||||||
|
"backup_after": {"type": "string"},
|
||||||
|
"exclusions": {"type": "array"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
basic_backup_patch_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"max_backups": {"type": "integer"},
|
||||||
|
"compress": {"type": "boolean"},
|
||||||
|
"shutdown": {"type": "boolean"},
|
||||||
|
"backup_before": {"type": "string"},
|
||||||
|
"backup_after": {"type": "string"},
|
||||||
|
"exclusions": {"type": "array"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerBackupsIndexHandler(BaseApiHandler):
|
||||||
|
def get(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.BACKUP
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
self.finish_json(200, self.controller.management.get_backup_config(server_id))
|
||||||
|
|
||||||
|
def patch(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
if auth_data[4]["superuser"]:
|
||||||
|
validate(data, backup_patch_schema)
|
||||||
|
else:
|
||||||
|
validate(data, basic_backup_patch_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.BACKUP
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.controller.management.set_backup_config(
|
||||||
|
server_id,
|
||||||
|
data.get(
|
||||||
|
"backup_path",
|
||||||
|
self.controller.management.get_backup_config(server_id)["backup_path"],
|
||||||
|
),
|
||||||
|
data.get(
|
||||||
|
"max_backups",
|
||||||
|
self.controller.management.get_backup_config(server_id)["max_backups"],
|
||||||
|
),
|
||||||
|
data.get("exclusions"),
|
||||||
|
data.get(
|
||||||
|
"compress",
|
||||||
|
self.controller.management.get_backup_config(server_id)["compress"],
|
||||||
|
),
|
||||||
|
data.get(
|
||||||
|
"shutdown",
|
||||||
|
self.controller.management.get_backup_config(server_id)["shutdown"],
|
||||||
|
),
|
||||||
|
data.get(
|
||||||
|
"backup_before",
|
||||||
|
self.controller.management.get_backup_config(server_id)["before"],
|
||||||
|
),
|
||||||
|
data.get(
|
||||||
|
"backup_after",
|
||||||
|
self.controller.management.get_backup_config(server_id)["after"],
|
||||||
|
),
|
||||||
|
)
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
555
app/classes/web/routes/api/servers/server/files.py
Normal file
555
app/classes/web/routes/api/servers/server/files.py
Normal file
@ -0,0 +1,555 @@
|
|||||||
|
import os
|
||||||
|
import logging
|
||||||
|
import json
|
||||||
|
import html
|
||||||
|
from jsonschema import validate
|
||||||
|
from jsonschema.exceptions import ValidationError
|
||||||
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
|
from app.classes.shared.helpers import Helpers
|
||||||
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
files_get_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"page": {"type": "string", "minLength": 1},
|
||||||
|
"path": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
files_patch_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"path": {"type": "string"},
|
||||||
|
"contents": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
files_unzip_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"folder": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
files_create_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"parent": {"type": "string"},
|
||||||
|
"name": {"type": "string"},
|
||||||
|
"directory": {"type": "boolean"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
files_rename_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"path": {"type": "string"},
|
||||||
|
"new_name": {"type": "string"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
file_delete_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"filename": {"type": "string", "minLength": 5},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerFilesIndexHandler(BaseApiHandler):
|
||||||
|
def post(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
or EnumPermissionsServer.BACKUP
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files or Backup permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_get_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
data["path"],
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if os.path.isdir(data["path"]):
|
||||||
|
# TODO: limit some columns for specific permissions?
|
||||||
|
folder = data["path"]
|
||||||
|
return_json = {
|
||||||
|
"root_path": {
|
||||||
|
"path": folder,
|
||||||
|
"top": data["path"]
|
||||||
|
== self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dir_list = []
|
||||||
|
unsorted_files = []
|
||||||
|
file_list = os.listdir(folder)
|
||||||
|
for item in file_list:
|
||||||
|
if os.path.isdir(os.path.join(folder, item)):
|
||||||
|
dir_list.append(item)
|
||||||
|
else:
|
||||||
|
unsorted_files.append(item)
|
||||||
|
file_list = sorted(dir_list, key=str.casefold) + sorted(
|
||||||
|
unsorted_files, key=str.casefold
|
||||||
|
)
|
||||||
|
for raw_filename in file_list:
|
||||||
|
filename = html.escape(raw_filename)
|
||||||
|
rel = os.path.join(folder, raw_filename)
|
||||||
|
dpath = os.path.join(folder, filename)
|
||||||
|
if str(dpath) in self.controller.management.get_excluded_backup_dirs(
|
||||||
|
server_id
|
||||||
|
):
|
||||||
|
if os.path.isdir(rel):
|
||||||
|
return_json[filename] = {
|
||||||
|
"path": dpath,
|
||||||
|
"dir": True,
|
||||||
|
"excluded": True,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
return_json[filename] = {
|
||||||
|
"path": dpath,
|
||||||
|
"dir": False,
|
||||||
|
"excluded": True,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
if os.path.isdir(rel):
|
||||||
|
return_json[filename] = {
|
||||||
|
"path": dpath,
|
||||||
|
"dir": True,
|
||||||
|
"excluded": False,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
return_json[filename] = {
|
||||||
|
"path": dpath,
|
||||||
|
"dir": False,
|
||||||
|
"excluded": False,
|
||||||
|
}
|
||||||
|
self.finish_json(200, {"status": "ok", "data": return_json})
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
with open(data["path"], encoding="utf-8") as file:
|
||||||
|
file_contents = file.read()
|
||||||
|
except UnicodeDecodeError as ex:
|
||||||
|
self.finish_json(
|
||||||
|
400,
|
||||||
|
{"status": "error", "error": "DECODE_ERROR", "error_data": str(ex)},
|
||||||
|
)
|
||||||
|
self.finish_json(200, {"status": "ok", "data": file_contents})
|
||||||
|
|
||||||
|
def delete(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, file_delete_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
data["filename"],
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if os.path.isdir(data["filename"]):
|
||||||
|
FileHelpers.del_dirs(data["filename"])
|
||||||
|
else:
|
||||||
|
FileHelpers.del_file(data["filename"])
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
def patch(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_patch_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
data["path"],
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
file_path = Helpers.get_os_understandable_path(data["path"])
|
||||||
|
file_contents = data["contents"]
|
||||||
|
# Open the file in write mode and store the content in file_object
|
||||||
|
with open(file_path, "w", encoding="utf-8") as file_object:
|
||||||
|
file_object.write(file_contents)
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
def put(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_create_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
path = os.path.join(data["parent"], data["name"])
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
path,
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if Helpers.check_path_exists(os.path.abspath(path)):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "FILE EXISTS",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if data["directory"]:
|
||||||
|
os.mkdir(path)
|
||||||
|
else:
|
||||||
|
# Create the file by opening it
|
||||||
|
with open(path, "w", encoding="utf-8") as file_object:
|
||||||
|
file_object.close()
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerFilesCreateHandler(BaseApiHandler):
|
||||||
|
def patch(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_rename_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
path = data["path"]
|
||||||
|
new_item_name = data["new_name"]
|
||||||
|
new_item_path = os.path.join(os.path.split(path)[0], new_item_name)
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
path,
|
||||||
|
) or not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
new_item_path,
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if Helpers.check_path_exists(os.path.abspath(new_item_path)):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "FILE EXISTS",
|
||||||
|
"error_data": {},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
os.rename(path, new_item_path)
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
def put(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_create_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
path = os.path.join(data["parent"], data["name"])
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
path,
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if Helpers.check_path_exists(os.path.abspath(path)):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "FILE EXISTS",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if data["directory"]:
|
||||||
|
os.mkdir(path)
|
||||||
|
else:
|
||||||
|
# Create the file by opening it
|
||||||
|
with open(path, "w", encoding="utf-8") as file_object:
|
||||||
|
file_object.close()
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerFilesZipHandler(BaseApiHandler):
|
||||||
|
def post(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.FILES
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Files permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
try:
|
||||||
|
validate(data, files_unzip_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
folder = data["folder"]
|
||||||
|
user_id = auth_data[4]["user_id"]
|
||||||
|
if not Helpers.validate_traversal(
|
||||||
|
self.controller.servers.get_server_data_by_id(server_id)["path"],
|
||||||
|
folder,
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "TRAVERSAL DETECTED",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if Helpers.check_file_exists(folder):
|
||||||
|
folder = self.file_helper.unzip_file(folder, user_id)
|
||||||
|
else:
|
||||||
|
if user_id:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "FILE_DOES_NOT_EXIST",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
28
app/classes/web/routes/api/servers/server/history.py
Normal file
28
app/classes/web/routes/api/servers/server/history.py
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import logging
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
from app.classes.controllers.servers_controller import ServersController
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerHistoryHandler(BaseApiHandler):
|
||||||
|
def get(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
srv = ServersController().get_server_instance_by_id(server_id)
|
||||||
|
history = srv.get_server_history()
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": history,
|
||||||
|
},
|
||||||
|
)
|
@ -74,6 +74,6 @@ class ApiServersServerLogsHandler(BaseApiHandler):
|
|||||||
|
|
||||||
if use_html:
|
if use_html:
|
||||||
for line in lines:
|
for line in lines:
|
||||||
self.write(f"{line}<br />")
|
line = f"{line}<br />"
|
||||||
else:
|
|
||||||
self.finish_json(200, {"status": "ok", "data": lines})
|
self.finish_json(200, {"status": "ok", "data": lines})
|
||||||
|
32
app/classes/web/routes/api/servers/server/status.py
Normal file
32
app/classes/web/routes/api/servers/server/status.py
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
import logging
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerStatusHandler(BaseApiHandler):
|
||||||
|
def get(self):
|
||||||
|
servers_status = []
|
||||||
|
servers_list = self.controller.servers.get_all_servers_stats()
|
||||||
|
for server in servers_list:
|
||||||
|
if server.get("server_data").get("show_status") is True:
|
||||||
|
servers_status.append(
|
||||||
|
{
|
||||||
|
"id": server.get("server_data").get("server_id"),
|
||||||
|
"world_name": server.get("stats").get("world_name"),
|
||||||
|
"running": server.get("stats").get("running"),
|
||||||
|
"online": server.get("stats").get("online"),
|
||||||
|
"max": server.get("stats").get("max"),
|
||||||
|
"version": server.get("stats").get("version"),
|
||||||
|
"desc": server.get("stats").get("desc"),
|
||||||
|
"icon": server.get("stats").get("icon"),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": servers_status,
|
||||||
|
},
|
||||||
|
)
|
108
app/classes/web/routes/api/servers/server/webhooks/index.py
Normal file
108
app/classes/web/routes/api/servers/server/webhooks/index.py
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
# TODO: create and read
|
||||||
|
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from jsonschema import ValidationError, validate
|
||||||
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
from app.classes.web.webhooks.webhook_factory import WebhookFactory
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
new_webhook_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"webhook_type": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": WebhookFactory.get_supported_providers(),
|
||||||
|
},
|
||||||
|
"name": {"type": "string"},
|
||||||
|
"url": {"type": "string"},
|
||||||
|
"bot_name": {"type": "string"},
|
||||||
|
"trigger": {"type": "array"},
|
||||||
|
"body": {"type": "string"},
|
||||||
|
"color": {"type": "string", "default": "#005cd1"},
|
||||||
|
"enabled": {
|
||||||
|
"type": "boolean",
|
||||||
|
"default": True,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 7,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerWebhooksIndexHandler(BaseApiHandler):
|
||||||
|
def get(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.CONFIG
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": self.controller.management.get_webhooks_by_server(server_id),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def post(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, new_webhook_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.CONFIG
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
data["server_id"] = server_id
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Edited server {server_id}: added webhook",
|
||||||
|
server_id,
|
||||||
|
self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
triggers = ""
|
||||||
|
for item in data["trigger"]:
|
||||||
|
string = item + ","
|
||||||
|
triggers += string
|
||||||
|
data["trigger"] = triggers
|
||||||
|
webhook_id = self.controller.management.create_webhook(data)
|
||||||
|
|
||||||
|
self.finish_json(200, {"status": "ok", "data": {"webhook_id": webhook_id}})
|
@ -0,0 +1,187 @@
|
|||||||
|
# TODO: read and delete
|
||||||
|
|
||||||
|
import json
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from jsonschema import ValidationError, validate
|
||||||
|
from app.classes.models.server_permissions import EnumPermissionsServer
|
||||||
|
from app.classes.web.webhooks.webhook_factory import WebhookFactory
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
webhook_patch_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"webhook_type": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": WebhookFactory.get_supported_providers(),
|
||||||
|
},
|
||||||
|
"name": {"type": "string"},
|
||||||
|
"url": {"type": "string"},
|
||||||
|
"bot_name": {"type": "string"},
|
||||||
|
"trigger": {"type": "array"},
|
||||||
|
"body": {"type": "string"},
|
||||||
|
"color": {"type": "string", "default": "#005cd1"},
|
||||||
|
"enabled": {
|
||||||
|
"type": "boolean",
|
||||||
|
"default": True,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class ApiServersServerWebhooksManagementIndexHandler(BaseApiHandler):
|
||||||
|
def get(self, server_id: str, webhook_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.CONFIG
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
if (
|
||||||
|
not str(webhook_id)
|
||||||
|
in self.controller.management.get_webhooks_by_server(server_id).keys()
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "NO WEBHOOK FOUND"}
|
||||||
|
)
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": self.controller.management.get_webhook_by_id(webhook_id),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def delete(self, server_id: str, webhook_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.CONFIG
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
try:
|
||||||
|
self.controller.management.delete_webhook(webhook_id)
|
||||||
|
except Exception:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "NO WEBHOOK FOUND"}
|
||||||
|
)
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Edited server {server_id}: removed webhook",
|
||||||
|
server_id,
|
||||||
|
self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
def patch(self, server_id: str, webhook_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, webhook_patch_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.CONFIG
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
data["server_id"] = server_id
|
||||||
|
if "trigger" in data.keys():
|
||||||
|
triggers = ""
|
||||||
|
for item in data["trigger"]:
|
||||||
|
string = item + ","
|
||||||
|
triggers += string
|
||||||
|
data["trigger"] = triggers
|
||||||
|
self.controller.management.modify_webhook(webhook_id, data)
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Edited server {server_id}: updated webhook",
|
||||||
|
server_id,
|
||||||
|
self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
self.finish_json(200, {"status": "ok"})
|
||||||
|
|
||||||
|
def post(self, server_id: str, webhook_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
"Tested webhook",
|
||||||
|
server_id,
|
||||||
|
self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
if (
|
||||||
|
EnumPermissionsServer.CONFIG
|
||||||
|
not in self.controller.server_perms.get_user_id_permissions_list(
|
||||||
|
auth_data[4]["user_id"], server_id
|
||||||
|
)
|
||||||
|
):
|
||||||
|
# if the user doesn't have Schedule permission, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
webhook = self.controller.management.get_webhook_by_id(webhook_id)
|
||||||
|
try:
|
||||||
|
webhook_provider = WebhookFactory.create_provider(webhook["webhook_type"])
|
||||||
|
webhook_provider.send(
|
||||||
|
server_name=self.controller.servers.get_server_data_by_id(server_id)[
|
||||||
|
"server_name"
|
||||||
|
],
|
||||||
|
title=f"Test Webhook: {webhook['name']}",
|
||||||
|
url=webhook["url"],
|
||||||
|
message=webhook["body"],
|
||||||
|
color=webhook["color"], # Prestigious purple!
|
||||||
|
bot_name="Crafty Webhooks Tester",
|
||||||
|
)
|
||||||
|
except Exception as e:
|
||||||
|
self.finish_json(500, {"status": "error", "error": str(e)})
|
||||||
|
|
||||||
|
self.finish_json(200, {"status": "ok"})
|
@ -93,10 +93,17 @@ class ApiUsersIndexHandler(BaseApiHandler):
|
|||||||
"error_data": str(e),
|
"error_data": str(e),
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
username = data["username"]
|
username = data["username"]
|
||||||
username = str(username).lower()
|
username = str(username).lower()
|
||||||
manager = int(user["user_id"])
|
manager = data.get("manager", None)
|
||||||
|
if user["superuser"]:
|
||||||
|
if (
|
||||||
|
manager == self.controller.users.get_id_by_name("SYSTEM")
|
||||||
|
or manager == 0
|
||||||
|
):
|
||||||
|
manager = None
|
||||||
|
else:
|
||||||
|
manager = int(user["user_id"])
|
||||||
password = data["password"]
|
password = data["password"]
|
||||||
email = data.get("email", "default@example.com")
|
email = data.get("email", "default@example.com")
|
||||||
enabled = data.get("enabled", True)
|
enabled = data.get("enabled", True)
|
||||||
|
243
app/classes/web/routes/api/users/user/api.py
Normal file
243
app/classes/web/routes/api/users/user/api.py
Normal file
@ -0,0 +1,243 @@
|
|||||||
|
import json
|
||||||
|
import logging
|
||||||
|
|
||||||
|
from jsonschema import ValidationError, validate
|
||||||
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
|
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
|
class ApiUsersUserKeyHandler(BaseApiHandler):
|
||||||
|
def get(self, user_id: str, key_id=None):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
if key_id:
|
||||||
|
key = self.controller.users.get_user_api_key(key_id)
|
||||||
|
# does this user id exist?
|
||||||
|
if key is None:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID DATA",
|
||||||
|
"error_data": "INVALID KEY",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (
|
||||||
|
str(key.user_id) != str(auth_data[4]["user_id"])
|
||||||
|
and not auth_data[4]["superuser"]
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "NOT AUTHORIZED",
|
||||||
|
"error_data": "TRIED TO EDIT KEY WIHTOUT AUTH",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Generated a new API token for the key {key.name} "
|
||||||
|
f"from user with UID: {key.user_id}",
|
||||||
|
server_id=0,
|
||||||
|
source_ip=self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
data_key = self.controller.authentication.generate(
|
||||||
|
key.user_id_id, {"token_id": key.token_id}
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.finish_json(
|
||||||
|
200,
|
||||||
|
{"status": "ok", "data": data_key},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (
|
||||||
|
str(user_id) != str(auth_data[4]["user_id"])
|
||||||
|
and not auth_data[4]["superuser"]
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "NOT AUTHORIZED",
|
||||||
|
"error_data": "TRIED TO EDIT KEY WIHTOUT AUTH",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
keys = []
|
||||||
|
for key in self.controller.users.get_user_api_keys(str(user_id)):
|
||||||
|
keys.append(
|
||||||
|
{
|
||||||
|
"id": key.token_id,
|
||||||
|
"name": key.name,
|
||||||
|
"server_permissions": key.server_permissions,
|
||||||
|
"crafty_permissions": key.crafty_permissions,
|
||||||
|
"superuser": key.superuser,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
self.finish_json(
|
||||||
|
200,
|
||||||
|
{
|
||||||
|
"status": "ok",
|
||||||
|
"data": keys,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
def patch(self, user_id: str):
|
||||||
|
user_key_schema = {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"name": {"type": "string", "minLength": 3},
|
||||||
|
"server_permissions_mask": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[01]{8}$", # 8 bits, see EnumPermissionsServer
|
||||||
|
},
|
||||||
|
"crafty_permissions_mask": {
|
||||||
|
"type": "string",
|
||||||
|
"pattern": "^[01]{3}$", # 8 bits, see EnumPermissionsCrafty
|
||||||
|
},
|
||||||
|
"superuser": {"type": "boolean"},
|
||||||
|
},
|
||||||
|
"additionalProperties": False,
|
||||||
|
"minProperties": 1,
|
||||||
|
}
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_exec_user_crafty_permissions,
|
||||||
|
_,
|
||||||
|
_superuser,
|
||||||
|
user,
|
||||||
|
) = auth_data
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(self.request.body)
|
||||||
|
except json.decoder.JSONDecodeError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_JSON", "error_data": str(e)}
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
validate(data, user_key_schema)
|
||||||
|
except ValidationError as e:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID_JSON_SCHEMA",
|
||||||
|
"error_data": str(e),
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if user_id == "@me":
|
||||||
|
user_id = user["user_id"]
|
||||||
|
# does this user id exist?
|
||||||
|
if not self.controller.users.user_id_exists(user_id):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "USER NOT FOUND",
|
||||||
|
"error_data": "USER_NOT_FOUND",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (
|
||||||
|
str(user_id) != str(auth_data[4]["user_id"])
|
||||||
|
and not auth_data[4]["superuser"]
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "NOT AUTHORIZED",
|
||||||
|
"error_data": "TRIED TO EDIT KEY WIHTOUT AUTH",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
key_id = self.controller.users.add_user_api_key(
|
||||||
|
data["name"],
|
||||||
|
user_id,
|
||||||
|
data["superuser"],
|
||||||
|
data["server_permissions_mask"],
|
||||||
|
data["crafty_permissions_mask"],
|
||||||
|
)
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Added API key {data['name']} with crafty permissions "
|
||||||
|
f"{data['crafty_permissions_mask']}"
|
||||||
|
f" and {data['server_permissions_mask']} for user with UID: {user_id}",
|
||||||
|
server_id=0,
|
||||||
|
source_ip=self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
self.finish_json(200, {"status": "ok", "data": {"id": key_id}})
|
||||||
|
|
||||||
|
def delete(self, _user_id: str, key_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
(
|
||||||
|
_,
|
||||||
|
_exec_user_crafty_permissions,
|
||||||
|
_,
|
||||||
|
_,
|
||||||
|
_user,
|
||||||
|
) = auth_data
|
||||||
|
if key_id:
|
||||||
|
key = self.controller.users.get_user_api_key(key_id)
|
||||||
|
# does this user id exist?
|
||||||
|
if key is None:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID DATA",
|
||||||
|
"error_data": "INVALID KEY",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
# does this user id exist?
|
||||||
|
target_key = self.controller.users.get_user_api_key(key_id)
|
||||||
|
if not target_key:
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "INVALID KEY",
|
||||||
|
"error_data": "INVALID KEY ID",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
if (
|
||||||
|
target_key.user_id != auth_data[4]["user_id"]
|
||||||
|
and not auth_data[4]["superuser"]
|
||||||
|
):
|
||||||
|
return self.finish_json(
|
||||||
|
400,
|
||||||
|
{
|
||||||
|
"status": "error",
|
||||||
|
"error": "NOT AUTHORIZED",
|
||||||
|
"error_data": "TRIED TO EDIT KEY WIHTOUT AUTH",
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
self.controller.users.delete_user_api_key(key_id)
|
||||||
|
|
||||||
|
self.controller.management.add_to_audit_log(
|
||||||
|
auth_data[4]["user_id"],
|
||||||
|
f"Removed API key {target_key} "
|
||||||
|
f"(ID: {key_id}) from user {auth_data[4]['user_id']}",
|
||||||
|
server_id=0,
|
||||||
|
source_ip=self.get_remote_ip(),
|
||||||
|
)
|
||||||
|
|
||||||
|
return self.finish_json(
|
||||||
|
200,
|
||||||
|
{"status": "ok", "data": {"id": key_id}},
|
||||||
|
)
|
@ -4,10 +4,7 @@ import typing as t
|
|||||||
|
|
||||||
from jsonschema import ValidationError, validate
|
from jsonschema import ValidationError, validate
|
||||||
from app.classes.controllers.users_controller import UsersController
|
from app.classes.controllers.users_controller import UsersController
|
||||||
from app.classes.models.crafty_permissions import (
|
from app.classes.models.crafty_permissions import EnumPermissionsCrafty
|
||||||
EnumPermissionsCrafty,
|
|
||||||
PermissionsCrafty,
|
|
||||||
)
|
|
||||||
from app.classes.models.roles import HelperRoles
|
from app.classes.models.roles import HelperRoles
|
||||||
from app.classes.models.users import HelperUsers
|
from app.classes.models.users import HelperUsers
|
||||||
from app.classes.web.base_api_handler import BaseApiHandler
|
from app.classes.web.base_api_handler import BaseApiHandler
|
||||||
@ -166,7 +163,13 @@ class ApiUsersUserIndexHandler(BaseApiHandler):
|
|||||||
return self.finish_json(
|
return self.finish_json(
|
||||||
400, {"status": "error", "error": "INVALID_USERNAME"}
|
400, {"status": "error", "error": "INVALID_USERNAME"}
|
||||||
)
|
)
|
||||||
if self.controller.users.get_id_by_name(data["username"]) is not None:
|
if self.controller.users.get_id_by_name(
|
||||||
|
data["username"]
|
||||||
|
) is not None and self.controller.users.get_id_by_name(
|
||||||
|
data["username"]
|
||||||
|
) != int(
|
||||||
|
user_id
|
||||||
|
):
|
||||||
return self.finish_json(
|
return self.finish_json(
|
||||||
400, {"status": "error", "error": "USER_EXISTS"}
|
400, {"status": "error", "error": "USER_EXISTS"}
|
||||||
)
|
)
|
||||||
@ -210,13 +213,13 @@ class ApiUsersUserIndexHandler(BaseApiHandler):
|
|||||||
400, {"status": "error", "error": "INVALID_ROLES_MODIFY"}
|
400, {"status": "error", "error": "INVALID_ROLES_MODIFY"}
|
||||||
)
|
)
|
||||||
|
|
||||||
if "password" in data and str(user["user_id"] == str(user_id)):
|
|
||||||
# TODO: edit your own password
|
|
||||||
return self.finish_json(
|
|
||||||
400, {"status": "error", "error": "INVALID_PASSWORD_MODIFY"}
|
|
||||||
)
|
|
||||||
|
|
||||||
user_obj = HelperUsers.get_user_model(user_id)
|
user_obj = HelperUsers.get_user_model(user_id)
|
||||||
|
if "password" in data and str(user["user_id"]) != str(user_id):
|
||||||
|
if str(user["user_id"]) != str(user_obj.manager):
|
||||||
|
# TODO: edit your own password
|
||||||
|
return self.finish_json(
|
||||||
|
400, {"status": "error", "error": "INVALID_PASSWORD_MODIFY"}
|
||||||
|
)
|
||||||
|
|
||||||
if "roles" in data:
|
if "roles" in data:
|
||||||
roles: t.Set[str] = set(data.pop("roles"))
|
roles: t.Set[str] = set(data.pop("roles"))
|
||||||
@ -236,30 +239,30 @@ class ApiUsersUserIndexHandler(BaseApiHandler):
|
|||||||
user_id, removed_roles
|
user_id, removed_roles
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if "manager" in data and (
|
||||||
|
data["manager"] == self.controller.users.get_id_by_name("SYSTEM")
|
||||||
|
or data["manager"] == 0
|
||||||
|
):
|
||||||
|
data["manager"] = None
|
||||||
|
crafty_perms = None
|
||||||
if "permissions" in data:
|
if "permissions" in data:
|
||||||
permissions: t.List[UsersController.ApiPermissionDict] = data.pop(
|
permissions: t.List[UsersController.ApiPermissionDict] = data.pop(
|
||||||
"permissions"
|
"permissions"
|
||||||
)
|
)
|
||||||
permissions_mask = "0" * len(EnumPermissionsCrafty)
|
permissions_mask = "0" * len(EnumPermissionsCrafty)
|
||||||
limit_server_creation = 0
|
if permissions is not None:
|
||||||
limit_user_creation = 0
|
server_quantity = {}
|
||||||
limit_role_creation = 0
|
permissions_mask = list(permissions_mask)
|
||||||
|
for permission in permissions:
|
||||||
for permission in permissions:
|
server_quantity[permission["name"]] = permission["quantity"]
|
||||||
self.controller.crafty_perms.set_permission(
|
permissions_mask[
|
||||||
permissions_mask,
|
EnumPermissionsCrafty[permission["name"]].value
|
||||||
EnumPermissionsCrafty.__members__[permission["name"]],
|
] = ("1" if permission["enabled"] else "0")
|
||||||
"1" if permission["enabled"] else "0",
|
permissions_mask = "".join(permissions_mask)
|
||||||
)
|
crafty_perms = {
|
||||||
|
"permissions_mask": permissions_mask,
|
||||||
PermissionsCrafty.add_or_update_user(
|
"server_quantity": server_quantity,
|
||||||
user_id,
|
}
|
||||||
permissions_mask,
|
|
||||||
limit_server_creation,
|
|
||||||
limit_user_creation,
|
|
||||||
limit_role_creation,
|
|
||||||
)
|
|
||||||
|
|
||||||
# TODO: make this more efficient
|
# TODO: make this more efficient
|
||||||
if len(data) != 0:
|
if len(data) != 0:
|
||||||
for key in data:
|
for key in data:
|
||||||
@ -268,7 +271,11 @@ class ApiUsersUserIndexHandler(BaseApiHandler):
|
|||||||
if key == "password":
|
if key == "password":
|
||||||
value = self.helper.encode_pass(value)
|
value = self.helper.encode_pass(value)
|
||||||
setattr(user_obj, key, value)
|
setattr(user_obj, key, value)
|
||||||
user_obj.save()
|
self.controller.users.update_user(
|
||||||
|
user_id,
|
||||||
|
data,
|
||||||
|
crafty_perms,
|
||||||
|
)
|
||||||
|
|
||||||
self.controller.management.add_to_audit_log(
|
self.controller.management.add_to_audit_log(
|
||||||
user["user_id"],
|
user["user_id"],
|
||||||
|
31
app/classes/web/routes/metrics/host.py
Normal file
31
app/classes/web/routes/metrics/host.py
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
from prometheus_client.exposition import _bake_output
|
||||||
|
from prometheus_client.exposition import parse_qs, urlparse
|
||||||
|
|
||||||
|
from app.classes.web.metrics_handler import BaseMetricsHandler
|
||||||
|
|
||||||
|
|
||||||
|
# Decorate function with metric.
|
||||||
|
class ApiOpenMetricsCraftyHandler(BaseMetricsHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if not auth_data[3]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.get_registry()
|
||||||
|
|
||||||
|
def get_registry(self) -> None:
|
||||||
|
# Prepare parameters
|
||||||
|
registry = self.controller.management.host_registry
|
||||||
|
accept_header = self.request.headers.get("Accept")
|
||||||
|
accept_encoding_header = self.request.headers.get("Accept-Encoding")
|
||||||
|
params = parse_qs(urlparse(self.request.path).query)
|
||||||
|
# Bake output
|
||||||
|
status, headers, output = _bake_output(
|
||||||
|
registry, accept_header, accept_encoding_header, params, False
|
||||||
|
)
|
||||||
|
# Return output
|
||||||
|
self.finish_metrics(int(status.split(" ", maxsplit=1)[0]), headers, output)
|
21
app/classes/web/routes/metrics/index.py
Normal file
21
app/classes/web/routes/metrics/index.py
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
from prometheus_client import Info
|
||||||
|
from app.classes.web.metrics_handler import BaseMetricsHandler
|
||||||
|
|
||||||
|
CRAFTY_INFO = Info("Crafty_Controller", "Infos of this Crafty Instance")
|
||||||
|
|
||||||
|
|
||||||
|
# Decorate function with metric.
|
||||||
|
class ApiOpenMetricsIndexHandler(BaseMetricsHandler):
|
||||||
|
def get(self):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
version = f"{self.helper.get_version().get('major')} \
|
||||||
|
.{self.helper.get_version().get('minor')} \
|
||||||
|
.{self.helper.get_version().get('sub')}"
|
||||||
|
CRAFTY_INFO.info(
|
||||||
|
{"version": version, "docker": f"{self.helper.is_env_docker()}"}
|
||||||
|
)
|
||||||
|
|
||||||
|
self.get_registry()
|
24
app/classes/web/routes/metrics/metrics_handlers.py
Normal file
24
app/classes/web/routes/metrics/metrics_handlers.py
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
from app.classes.web.routes.metrics.index import ApiOpenMetricsIndexHandler
|
||||||
|
from app.classes.web.routes.metrics.host import ApiOpenMetricsCraftyHandler
|
||||||
|
from app.classes.web.routes.metrics.servers import ApiOpenMetricsServersHandler
|
||||||
|
|
||||||
|
|
||||||
|
def metrics_handlers(handler_args):
|
||||||
|
return [
|
||||||
|
# OpenMetrics routes
|
||||||
|
(
|
||||||
|
r"/metrics/?",
|
||||||
|
ApiOpenMetricsIndexHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/metrics/host/?",
|
||||||
|
ApiOpenMetricsCraftyHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
(
|
||||||
|
r"/metrics/servers/([0-9]+)/?",
|
||||||
|
ApiOpenMetricsServersHandler,
|
||||||
|
handler_args,
|
||||||
|
),
|
||||||
|
]
|
37
app/classes/web/routes/metrics/servers.py
Normal file
37
app/classes/web/routes/metrics/servers.py
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
from prometheus_client.exposition import _bake_output
|
||||||
|
from prometheus_client.exposition import parse_qs, urlparse
|
||||||
|
|
||||||
|
from app.classes.web.metrics_handler import BaseMetricsHandler
|
||||||
|
from app.classes.controllers.servers_controller import ServersController
|
||||||
|
|
||||||
|
|
||||||
|
# Decorate function with metric.
|
||||||
|
class ApiOpenMetricsServersHandler(BaseMetricsHandler):
|
||||||
|
def get(self, server_id: str):
|
||||||
|
auth_data = self.authenticate_user()
|
||||||
|
if not auth_data:
|
||||||
|
return
|
||||||
|
|
||||||
|
if server_id not in [str(x["server_id"]) for x in auth_data[0]]:
|
||||||
|
# if the user doesn't have access to the server, return an error
|
||||||
|
return self.finish_json(400, {"status": "error", "error": "NOT_AUTHORIZED"})
|
||||||
|
|
||||||
|
self.get_registry(server_id)
|
||||||
|
|
||||||
|
def get_registry(self, server_id=None) -> None:
|
||||||
|
if server_id is None:
|
||||||
|
return self.finish_json(500, {"status": "error", "error": "UNKNOWN_SERVER"})
|
||||||
|
|
||||||
|
# Prepare parameters
|
||||||
|
registry = (
|
||||||
|
ServersController().get_server_instance_by_id(server_id).server_registry
|
||||||
|
)
|
||||||
|
accept_header = self.request.headers.get("Accept")
|
||||||
|
accept_encoding_header = self.request.headers.get("Accept-Encoding")
|
||||||
|
params = parse_qs(urlparse(self.request.path).query)
|
||||||
|
# Bake output
|
||||||
|
status, headers, output = _bake_output(
|
||||||
|
registry, accept_header, accept_encoding_header, params, False
|
||||||
|
)
|
||||||
|
# Return output
|
||||||
|
self.finish_metrics(int(status.split(" ", maxsplit=1)[0]), headers, output)
|
@ -1,14 +1,10 @@
|
|||||||
import json
|
import json
|
||||||
import logging
|
import logging
|
||||||
import os
|
|
||||||
import time
|
|
||||||
import tornado.web
|
import tornado.web
|
||||||
import tornado.escape
|
import tornado.escape
|
||||||
import bleach
|
|
||||||
|
|
||||||
from app.classes.models.crafty_permissions import EnumPermissionsCrafty
|
from app.classes.models.crafty_permissions import EnumPermissionsCrafty
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.file_helpers import FileHelpers
|
|
||||||
from app.classes.shared.main_models import DatabaseShortcuts
|
from app.classes.shared.main_models import DatabaseShortcuts
|
||||||
from app.classes.web.base_handler import BaseHandler
|
from app.classes.web.base_handler import BaseHandler
|
||||||
|
|
||||||
@ -174,441 +170,3 @@ class ServerHandler(BaseHandler):
|
|||||||
data=page_data,
|
data=page_data,
|
||||||
translate=self.translator.translate,
|
translate=self.translator.translate,
|
||||||
)
|
)
|
||||||
|
|
||||||
@tornado.web.authenticated
|
|
||||||
def post(self, page):
|
|
||||||
api_key, _token_data, exec_user = self.current_user
|
|
||||||
superuser = exec_user["superuser"]
|
|
||||||
if api_key is not None:
|
|
||||||
superuser = superuser and api_key.superuser
|
|
||||||
|
|
||||||
template = "public/404.html"
|
|
||||||
page_data = {
|
|
||||||
"version_data": "version_data_here", # TODO
|
|
||||||
"user_data": exec_user,
|
|
||||||
"show_contribute": self.helper.get_setting("show_contribute_link", True),
|
|
||||||
"background": self.controller.cached_login,
|
|
||||||
"lang": self.controller.users.get_user_lang_by_id(exec_user["user_id"]),
|
|
||||||
"lang_page": Helpers.get_lang_page(
|
|
||||||
self.controller.users.get_user_lang_by_id(exec_user["user_id"])
|
|
||||||
),
|
|
||||||
}
|
|
||||||
|
|
||||||
if page == "command":
|
|
||||||
server_id = bleach.clean(self.get_argument("id", None))
|
|
||||||
command = bleach.clean(self.get_argument("command", None))
|
|
||||||
|
|
||||||
if server_id is not None:
|
|
||||||
if command == "clone_server":
|
|
||||||
if (
|
|
||||||
not superuser
|
|
||||||
and not self.controller.crafty_perms.can_create_server(
|
|
||||||
exec_user["user_id"]
|
|
||||||
)
|
|
||||||
):
|
|
||||||
time.sleep(3)
|
|
||||||
self.helper.websocket_helper.broadcast_user(
|
|
||||||
exec_user["user_id"],
|
|
||||||
"send_start_error",
|
|
||||||
{
|
|
||||||
"error": "<i class='fas fa-exclamation-triangle'"
|
|
||||||
" style='font-size:48px;color:red'>"
|
|
||||||
"</i> Not a server creator or server limit reached."
|
|
||||||
},
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
def is_name_used(name):
|
|
||||||
for server in self.controller.servers.get_all_defined_servers():
|
|
||||||
if server["server_name"] == name:
|
|
||||||
return True
|
|
||||||
return
|
|
||||||
|
|
||||||
template = "/panel/dashboard"
|
|
||||||
server_data = self.controller.servers.get_server_data_by_id(
|
|
||||||
server_id
|
|
||||||
)
|
|
||||||
new_server_name = server_data.get("server_name") + " (Copy)"
|
|
||||||
|
|
||||||
name_counter = 1
|
|
||||||
while is_name_used(new_server_name):
|
|
||||||
name_counter += 1
|
|
||||||
new_server_name = (
|
|
||||||
server_data.get("server_name") + f" (Copy {name_counter})"
|
|
||||||
)
|
|
||||||
|
|
||||||
new_server_uuid = Helpers.create_uuid()
|
|
||||||
while os.path.exists(
|
|
||||||
os.path.join(self.helper.servers_dir, new_server_uuid)
|
|
||||||
):
|
|
||||||
new_server_uuid = Helpers.create_uuid()
|
|
||||||
new_server_path = os.path.join(
|
|
||||||
self.helper.servers_dir, new_server_uuid
|
|
||||||
)
|
|
||||||
|
|
||||||
# copy the old server
|
|
||||||
FileHelpers.copy_dir(server_data.get("path"), new_server_path)
|
|
||||||
|
|
||||||
# TODO get old server DB data to individual variables
|
|
||||||
stop_command = server_data.get("stop_command")
|
|
||||||
new_server_command = str(server_data.get("execution_command"))
|
|
||||||
new_executable = server_data.get("executable")
|
|
||||||
new_server_log_file = str(
|
|
||||||
Helpers.get_os_understandable_path(server_data.get("log_path"))
|
|
||||||
)
|
|
||||||
backup_path = os.path.join(self.helper.backup_path, new_server_uuid)
|
|
||||||
server_port = server_data.get("server_port")
|
|
||||||
server_type = server_data.get("type")
|
|
||||||
created_by = exec_user["user_id"]
|
|
||||||
|
|
||||||
new_server_id = self.controller.servers.create_server(
|
|
||||||
new_server_name,
|
|
||||||
new_server_uuid,
|
|
||||||
new_server_path,
|
|
||||||
backup_path,
|
|
||||||
new_server_command,
|
|
||||||
new_executable,
|
|
||||||
new_server_log_file,
|
|
||||||
stop_command,
|
|
||||||
server_type,
|
|
||||||
created_by,
|
|
||||||
server_port,
|
|
||||||
)
|
|
||||||
if not exec_user["superuser"]:
|
|
||||||
new_server_uuid = self.controller.servers.get_server_data_by_id(
|
|
||||||
new_server_id
|
|
||||||
).get("server_uuid")
|
|
||||||
role_id = self.controller.roles.add_role(
|
|
||||||
f"Creator of Server with uuid={new_server_uuid}",
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.server_perms.add_role_server(
|
|
||||||
new_server_id, role_id, "11111111"
|
|
||||||
)
|
|
||||||
self.controller.users.add_role_to_user(
|
|
||||||
exec_user["user_id"], role_id
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.servers.init_all_servers()
|
|
||||||
|
|
||||||
return
|
|
||||||
|
|
||||||
self.controller.management.send_command(
|
|
||||||
exec_user["user_id"], server_id, self.get_remote_ip(), command
|
|
||||||
)
|
|
||||||
|
|
||||||
if page == "step1":
|
|
||||||
if not superuser and not self.controller.crafty_perms.can_create_server(
|
|
||||||
exec_user["user_id"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: "
|
|
||||||
"not a server creator or server limit reached"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
if not superuser:
|
|
||||||
user_roles = self.controller.roles.get_all_roles()
|
|
||||||
else:
|
|
||||||
user_roles = self.get_user_roles()
|
|
||||||
server = bleach.clean(self.get_argument("server", ""))
|
|
||||||
server_name = bleach.clean(self.get_argument("server_name", ""))
|
|
||||||
min_mem = bleach.clean(self.get_argument("min_memory", ""))
|
|
||||||
max_mem = bleach.clean(self.get_argument("max_memory", ""))
|
|
||||||
port = bleach.clean(self.get_argument("port", ""))
|
|
||||||
if int(port) < 1 or int(port) > 65535:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Constraint Error: "
|
|
||||||
"Port must be greater than 0 and less than 65535"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
import_type = bleach.clean(self.get_argument("create_type", ""))
|
|
||||||
import_server_path = bleach.clean(self.get_argument("server_path", ""))
|
|
||||||
import_server_jar = bleach.clean(self.get_argument("server_jar", ""))
|
|
||||||
server_parts = server.split("|")
|
|
||||||
captured_roles = []
|
|
||||||
for role in user_roles:
|
|
||||||
if bleach.clean(self.get_argument(str(role), "")) == "on":
|
|
||||||
captured_roles.append(role)
|
|
||||||
|
|
||||||
if not server_name:
|
|
||||||
self.redirect("/panel/error?error=Server name cannot be empty!")
|
|
||||||
return
|
|
||||||
|
|
||||||
if import_type == "import_jar":
|
|
||||||
if self.helper.is_subdir(
|
|
||||||
self.controller.project_root, import_server_path
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Loop Error: The selected path will cause"
|
|
||||||
" an infinite copy loop. Make sure Crafty's directory is not"
|
|
||||||
" in your server path."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
good_path = self.controller.verify_jar_server(
|
|
||||||
import_server_path, import_server_jar
|
|
||||||
)
|
|
||||||
|
|
||||||
if not good_path:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Server path or Server Jar not found!"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
new_server_id = self.controller.import_jar_server(
|
|
||||||
server_name,
|
|
||||||
import_server_path,
|
|
||||||
import_server_jar,
|
|
||||||
min_mem,
|
|
||||||
max_mem,
|
|
||||||
port,
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f'imported a jar server named "{server_name}"',
|
|
||||||
new_server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
elif import_type == "import_zip":
|
|
||||||
# here import_server_path means the zip path
|
|
||||||
zip_path = bleach.clean(self.get_argument("root_path"))
|
|
||||||
good_path = Helpers.check_path_exists(zip_path)
|
|
||||||
if not good_path:
|
|
||||||
self.redirect("/panel/error?error=Temp path not found!")
|
|
||||||
return
|
|
||||||
|
|
||||||
new_server_id = self.controller.import_zip_server(
|
|
||||||
server_name,
|
|
||||||
zip_path,
|
|
||||||
import_server_jar,
|
|
||||||
min_mem,
|
|
||||||
max_mem,
|
|
||||||
port,
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
if new_server_id == "false":
|
|
||||||
self.redirect(
|
|
||||||
f"/panel/error?error=Zip file not accessible! "
|
|
||||||
f"You can fix this permissions issue with "
|
|
||||||
f"sudo chown -R crafty:crafty {import_server_path} "
|
|
||||||
f"And sudo chmod 2775 -R {import_server_path}"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f'imported a zip server named "{server_name}"',
|
|
||||||
new_server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
if len(server_parts) != 3:
|
|
||||||
self.redirect("/panel/error?error=Invalid server data")
|
|
||||||
return
|
|
||||||
jar_type, server_type, server_version = server_parts
|
|
||||||
# TODO: add server type check here and call the correct server
|
|
||||||
# add functions if not a jar
|
|
||||||
if server_type == "forge" and not self.helper.detect_java():
|
|
||||||
translation = self.helper.translation.translate(
|
|
||||||
"error",
|
|
||||||
"installerJava",
|
|
||||||
self.controller.users.get_user_lang_by_id(exec_user["user_id"]),
|
|
||||||
).format(server_name)
|
|
||||||
self.redirect(f"/panel/error?error={translation}")
|
|
||||||
return
|
|
||||||
new_server_id = self.controller.create_jar_server(
|
|
||||||
jar_type,
|
|
||||||
server_type,
|
|
||||||
server_version,
|
|
||||||
server_name,
|
|
||||||
min_mem,
|
|
||||||
max_mem,
|
|
||||||
port,
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f"created a {server_version} {str(server_type).capitalize()}"
|
|
||||||
f' server named "{server_name}"',
|
|
||||||
# Example: Admin created a 1.16.5 Bukkit server named "survival"
|
|
||||||
new_server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
|
|
||||||
# These lines create a new Role for the Server with full permissions
|
|
||||||
# and add the user to it if he's not a superuser
|
|
||||||
if len(captured_roles) == 0:
|
|
||||||
if not superuser:
|
|
||||||
new_server_uuid = self.controller.servers.get_server_data_by_id(
|
|
||||||
new_server_id
|
|
||||||
).get("server_uuid")
|
|
||||||
role_id = self.controller.roles.add_role(
|
|
||||||
f"Creator of Server with uuid={new_server_uuid}",
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.server_perms.add_role_server(
|
|
||||||
new_server_id, role_id, "11111111"
|
|
||||||
)
|
|
||||||
self.controller.users.add_role_to_user(
|
|
||||||
exec_user["user_id"], role_id
|
|
||||||
)
|
|
||||||
|
|
||||||
else:
|
|
||||||
for role in captured_roles:
|
|
||||||
role_id = role
|
|
||||||
self.controller.server_perms.add_role_server(
|
|
||||||
new_server_id, role_id, "11111111"
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.servers.stats.record_stats()
|
|
||||||
self.redirect("/panel/dashboard")
|
|
||||||
|
|
||||||
if page == "bedrock_step1":
|
|
||||||
if not superuser and not self.controller.crafty_perms.can_create_server(
|
|
||||||
exec_user["user_id"]
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Unauthorized access: "
|
|
||||||
"not a server creator or server limit reached"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
if not superuser:
|
|
||||||
user_roles = self.controller.roles.get_all_roles()
|
|
||||||
else:
|
|
||||||
user_roles = self.controller.roles.get_all_roles()
|
|
||||||
server = bleach.clean(self.get_argument("server", ""))
|
|
||||||
server_name = bleach.clean(self.get_argument("server_name", ""))
|
|
||||||
port = bleach.clean(self.get_argument("port", ""))
|
|
||||||
|
|
||||||
if not port:
|
|
||||||
port = 19132
|
|
||||||
if int(port) < 1 or int(port) > 65535:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Constraint Error: "
|
|
||||||
"Port must be greater than 0 and less than 65535"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
import_type = bleach.clean(self.get_argument("create_type", ""))
|
|
||||||
import_server_path = bleach.clean(self.get_argument("server_path", ""))
|
|
||||||
import_server_exe = bleach.clean(self.get_argument("server_jar", ""))
|
|
||||||
server_parts = server.split("|")
|
|
||||||
captured_roles = []
|
|
||||||
for role in user_roles:
|
|
||||||
if bleach.clean(self.get_argument(str(role), "")) == "on":
|
|
||||||
captured_roles.append(role)
|
|
||||||
|
|
||||||
if not server_name:
|
|
||||||
self.redirect("/panel/error?error=Server name cannot be empty!")
|
|
||||||
return
|
|
||||||
|
|
||||||
if import_type == "import_jar":
|
|
||||||
if self.helper.is_subdir(
|
|
||||||
self.controller.project_root, import_server_path
|
|
||||||
):
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Loop Error: The selected path will cause"
|
|
||||||
" an infinite copy loop. Make sure Crafty's directory is not"
|
|
||||||
" in your server path."
|
|
||||||
)
|
|
||||||
return
|
|
||||||
good_path = self.controller.verify_jar_server(
|
|
||||||
import_server_path, import_server_exe
|
|
||||||
)
|
|
||||||
|
|
||||||
if not good_path:
|
|
||||||
self.redirect(
|
|
||||||
"/panel/error?error=Server path or Server Jar not found!"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
|
|
||||||
new_server_id = self.controller.import_bedrock_server(
|
|
||||||
server_name,
|
|
||||||
import_server_path,
|
|
||||||
import_server_exe,
|
|
||||||
port,
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f'imported a jar server named "{server_name}"',
|
|
||||||
new_server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
elif import_type == "import_zip":
|
|
||||||
# here import_server_path means the zip path
|
|
||||||
zip_path = bleach.clean(self.get_argument("root_path"))
|
|
||||||
good_path = Helpers.check_path_exists(zip_path)
|
|
||||||
if not good_path:
|
|
||||||
self.redirect("/panel/error?error=Temp path not found!")
|
|
||||||
return
|
|
||||||
|
|
||||||
new_server_id = self.controller.import_bedrock_zip_server(
|
|
||||||
server_name,
|
|
||||||
zip_path,
|
|
||||||
import_server_exe,
|
|
||||||
port,
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
if new_server_id == "false":
|
|
||||||
self.redirect(
|
|
||||||
f"/panel/error?error=Zip file not accessible! "
|
|
||||||
f"You can fix this permissions issue with"
|
|
||||||
f"sudo chown -R crafty:crafty {import_server_path} "
|
|
||||||
f"And sudo chmod 2775 -R {import_server_path}"
|
|
||||||
)
|
|
||||||
return
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
f'imported a zip server named "{server_name}"',
|
|
||||||
new_server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
new_server_id = self.controller.create_bedrock_server(
|
|
||||||
server_name,
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.management.add_to_audit_log(
|
|
||||||
exec_user["user_id"],
|
|
||||||
"created a Bedrock " f'server named "{server_name}"',
|
|
||||||
# Example: Admin created a 1.16.5 Bukkit server named "survival"
|
|
||||||
new_server_id,
|
|
||||||
self.get_remote_ip(),
|
|
||||||
)
|
|
||||||
|
|
||||||
# These lines create a new Role for the Server with full permissions
|
|
||||||
# and add the user to it if he's not a superuser
|
|
||||||
if len(captured_roles) == 0:
|
|
||||||
if not superuser:
|
|
||||||
new_server_uuid = self.controller.servers.get_server_data_by_id(
|
|
||||||
new_server_id
|
|
||||||
).get("server_uuid")
|
|
||||||
role_id = self.controller.roles.add_role(
|
|
||||||
f"Creator of Server with uuid={new_server_uuid}",
|
|
||||||
exec_user["user_id"],
|
|
||||||
)
|
|
||||||
self.controller.server_perms.add_role_server(
|
|
||||||
new_server_id, role_id, "11111111"
|
|
||||||
)
|
|
||||||
self.controller.users.add_role_to_user(
|
|
||||||
exec_user["user_id"], role_id
|
|
||||||
)
|
|
||||||
|
|
||||||
else:
|
|
||||||
for role in captured_roles:
|
|
||||||
role_id = role
|
|
||||||
self.controller.server_perms.add_role_server(
|
|
||||||
new_server_id, role_id, "11111111"
|
|
||||||
)
|
|
||||||
|
|
||||||
self.controller.servers.stats.record_stats()
|
|
||||||
self.redirect("/panel/dashboard")
|
|
||||||
|
|
||||||
try:
|
|
||||||
self.render(
|
|
||||||
template,
|
|
||||||
data=page_data,
|
|
||||||
translate=self.translator.translate,
|
|
||||||
)
|
|
||||||
except RuntimeError:
|
|
||||||
self.redirect("/panel/dashboard")
|
|
||||||
|
@ -14,14 +14,14 @@ import tornado.httpserver
|
|||||||
from app.classes.models.management import HelpersManagement
|
from app.classes.models.management import HelpersManagement
|
||||||
from app.classes.shared.console import Console
|
from app.classes.shared.console import Console
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
|
from app.classes.shared.file_helpers import FileHelpers
|
||||||
from app.classes.shared.main_controller import Controller
|
from app.classes.shared.main_controller import Controller
|
||||||
from app.classes.web.file_handler import FileHandler
|
|
||||||
from app.classes.web.public_handler import PublicHandler
|
from app.classes.web.public_handler import PublicHandler
|
||||||
from app.classes.web.panel_handler import PanelHandler
|
from app.classes.web.panel_handler import PanelHandler
|
||||||
from app.classes.web.default_handler import DefaultHandler
|
from app.classes.web.default_handler import DefaultHandler
|
||||||
from app.classes.web.routes.api.api_handlers import api_handlers
|
from app.classes.web.routes.api.api_handlers import api_handlers
|
||||||
|
from app.classes.web.routes.metrics.metrics_handlers import metrics_handlers
|
||||||
from app.classes.web.server_handler import ServerHandler
|
from app.classes.web.server_handler import ServerHandler
|
||||||
from app.classes.web.ajax_handler import AjaxHandler
|
|
||||||
from app.classes.web.api_handler import (
|
from app.classes.web.api_handler import (
|
||||||
ServersStats,
|
ServersStats,
|
||||||
NodeStats,
|
NodeStats,
|
||||||
@ -34,7 +34,7 @@ from app.classes.web.api_handler import (
|
|||||||
ListServers,
|
ListServers,
|
||||||
SendCommand,
|
SendCommand,
|
||||||
)
|
)
|
||||||
from app.classes.web.websocket_handler import SocketHandler
|
from app.classes.web.websocket_handler import WebSocketHandler
|
||||||
from app.classes.web.static_handler import CustomStaticHandler
|
from app.classes.web.static_handler import CustomStaticHandler
|
||||||
from app.classes.web.upload_handler import UploadHandler
|
from app.classes.web.upload_handler import UploadHandler
|
||||||
from app.classes.web.http_handler import HTTPHandler, HTTPHandlerPage
|
from app.classes.web.http_handler import HTTPHandler, HTTPHandlerPage
|
||||||
@ -48,13 +48,20 @@ class Webserver:
|
|||||||
controller: Controller
|
controller: Controller
|
||||||
helper: Helpers
|
helper: Helpers
|
||||||
|
|
||||||
def __init__(self, helper, controller, tasks_manager):
|
def __init__(
|
||||||
|
self,
|
||||||
|
helper: Helpers,
|
||||||
|
controller: Controller,
|
||||||
|
tasks_manager,
|
||||||
|
file_helper: FileHelpers,
|
||||||
|
):
|
||||||
self.ioloop = None
|
self.ioloop = None
|
||||||
self.http_server = None
|
self.http_server = None
|
||||||
self.https_server = None
|
self.https_server = None
|
||||||
self.helper = helper
|
self.helper = helper
|
||||||
self.controller = controller
|
self.controller = controller
|
||||||
self.tasks_manager = tasks_manager
|
self.tasks_manager = tasks_manager
|
||||||
|
self.file_helper = file_helper
|
||||||
self._asyncio_patch()
|
self._asyncio_patch()
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
@ -146,14 +153,13 @@ class Webserver:
|
|||||||
"controller": self.controller,
|
"controller": self.controller,
|
||||||
"tasks_manager": self.tasks_manager,
|
"tasks_manager": self.tasks_manager,
|
||||||
"translator": self.helper.translation,
|
"translator": self.helper.translation,
|
||||||
|
"file_helper": self.file_helper,
|
||||||
}
|
}
|
||||||
handlers = [
|
handlers = [
|
||||||
(r"/", DefaultHandler, handler_args),
|
(r"/", DefaultHandler, handler_args),
|
||||||
(r"/panel/(.*)", PanelHandler, handler_args),
|
(r"/panel/(.*)", PanelHandler, handler_args),
|
||||||
(r"/server/(.*)", ServerHandler, handler_args),
|
(r"/server/(.*)", ServerHandler, handler_args),
|
||||||
(r"/ajax/(.*)", AjaxHandler, handler_args),
|
(r"/ws", WebSocketHandler, handler_args),
|
||||||
(r"/files/(.*)", FileHandler, handler_args),
|
|
||||||
(r"/ws", SocketHandler, handler_args),
|
|
||||||
(r"/upload", UploadHandler, handler_args),
|
(r"/upload", UploadHandler, handler_args),
|
||||||
(r"/status", StatusHandler, handler_args),
|
(r"/status", StatusHandler, handler_args),
|
||||||
# API Routes V1
|
# API Routes V1
|
||||||
@ -169,6 +175,8 @@ class Webserver:
|
|||||||
(r"/api/v1/users/delete_user", DeleteUser, handler_args),
|
(r"/api/v1/users/delete_user", DeleteUser, handler_args),
|
||||||
# API Routes V2
|
# API Routes V2
|
||||||
*api_handlers(handler_args),
|
*api_handlers(handler_args),
|
||||||
|
# API Routes OpenMetrics
|
||||||
|
*metrics_handlers(handler_args),
|
||||||
# Using this one at the end
|
# Using this one at the end
|
||||||
# to catch all the other requests to Public Handler
|
# to catch all the other requests to Public Handler
|
||||||
(r"/(.*)", PublicHandler, handler_args),
|
(r"/(.*)", PublicHandler, handler_args),
|
||||||
|
@ -12,6 +12,7 @@ from app.classes.shared.console import Console
|
|||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
from app.classes.shared.main_controller import Controller
|
from app.classes.shared.main_controller import Controller
|
||||||
from app.classes.web.base_handler import BaseHandler
|
from app.classes.web.base_handler import BaseHandler
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
@ -25,11 +26,13 @@ class UploadHandler(BaseHandler):
|
|||||||
controller: Controller = None,
|
controller: Controller = None,
|
||||||
tasks_manager=None,
|
tasks_manager=None,
|
||||||
translator=None,
|
translator=None,
|
||||||
|
file_helper=None,
|
||||||
):
|
):
|
||||||
self.helper = helper
|
self.helper = helper
|
||||||
self.controller = controller
|
self.controller = controller
|
||||||
self.tasks_manager = tasks_manager
|
self.tasks_manager = tasks_manager
|
||||||
self.translator = translator
|
self.translator = translator
|
||||||
|
self.file_helper = file_helper
|
||||||
|
|
||||||
def prepare(self):
|
def prepare(self):
|
||||||
# Class & Function Defination
|
# Class & Function Defination
|
||||||
@ -99,7 +102,8 @@ class UploadHandler(BaseHandler):
|
|||||||
)
|
)
|
||||||
self.do_upload = False
|
self.do_upload = False
|
||||||
|
|
||||||
path = os.path.join(self.controller.project_root, "imports")
|
path = os.path.join(self.controller.project_root, "import", "upload")
|
||||||
|
self.helper.ensure_dir_exists(path)
|
||||||
# Delete existing files
|
# Delete existing files
|
||||||
if len(os.listdir(path)) > 0:
|
if len(os.listdir(path)) > 0:
|
||||||
for item in os.listdir():
|
for item in os.listdir():
|
||||||
@ -113,7 +117,7 @@ class UploadHandler(BaseHandler):
|
|||||||
self.request.headers.get("X-FileName", None)
|
self.request.headers.get("X-FileName", None)
|
||||||
)
|
)
|
||||||
if not str(filename).endswith(".zip"):
|
if not str(filename).endswith(".zip"):
|
||||||
self.helper.websocket_helper.broadcast("close_upload_box", "error")
|
WebSocketManager().broadcast("close_upload_box", "error")
|
||||||
self.finish("error")
|
self.finish("error")
|
||||||
full_path = os.path.join(path, filename)
|
full_path = os.path.join(path, filename)
|
||||||
|
|
||||||
@ -313,13 +317,13 @@ class UploadHandler(BaseHandler):
|
|||||||
if self.do_upload:
|
if self.do_upload:
|
||||||
time.sleep(5)
|
time.sleep(5)
|
||||||
if files_left == 0:
|
if files_left == 0:
|
||||||
self.helper.websocket_helper.broadcast("close_upload_box", "success")
|
WebSocketManager().broadcast("close_upload_box", "success")
|
||||||
self.finish("success") # Nope, I'm sending "success"
|
self.finish("success") # Nope, I'm sending "success"
|
||||||
self.f.close()
|
self.f.close()
|
||||||
else:
|
else:
|
||||||
time.sleep(5)
|
time.sleep(5)
|
||||||
if files_left == 0:
|
if files_left == 0:
|
||||||
self.helper.websocket_helper.broadcast("close_upload_box", "error")
|
WebSocketManager().broadcast("close_upload_box", "error")
|
||||||
self.finish("error")
|
self.finish("error")
|
||||||
|
|
||||||
def data_received(self, chunk):
|
def data_received(self, chunk):
|
||||||
|
39
app/classes/web/webhooks/base_webhook.py
Normal file
39
app/classes/web/webhooks/base_webhook.py
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
from abc import ABC, abstractmethod
|
||||||
|
import logging
|
||||||
|
import requests
|
||||||
|
|
||||||
|
from app.classes.shared.helpers import Helpers
|
||||||
|
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
helper = Helpers()
|
||||||
|
|
||||||
|
|
||||||
|
class WebhookProvider(ABC):
|
||||||
|
"""
|
||||||
|
Base class for all webhook providers.
|
||||||
|
|
||||||
|
Provides a common interface for all webhook provider implementations,
|
||||||
|
ensuring that each provider will have a send method.
|
||||||
|
"""
|
||||||
|
|
||||||
|
WEBHOOK_USERNAME = "Crafty Webhooks"
|
||||||
|
WEBHOOK_PFP_URL = (
|
||||||
|
"https://gitlab.com/crafty-controller/crafty-4/-"
|
||||||
|
+ "/raw/master/app/frontend/static/assets/images/"
|
||||||
|
+ "Crafty_4-0.png"
|
||||||
|
)
|
||||||
|
CRAFTY_VERSION = helper.get_version_string()
|
||||||
|
|
||||||
|
def _send_request(self, url, payload, headers=None):
|
||||||
|
"""Send a POST request to the given URL with the provided payload."""
|
||||||
|
try:
|
||||||
|
response = requests.post(url, json=payload, headers=headers, timeout=10)
|
||||||
|
response.raise_for_status()
|
||||||
|
return "Dispatch successful"
|
||||||
|
except requests.RequestException as error:
|
||||||
|
logger.error(error)
|
||||||
|
raise RuntimeError(f"Failed to dispatch notification: {error}") from error
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
def send(self, server_name, title, url, message, **kwargs):
|
||||||
|
"""Abstract method that derived classes will implement for sending webhooks."""
|
82
app/classes/web/webhooks/discord_webhook.py
Normal file
82
app/classes/web/webhooks/discord_webhook.py
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from app.classes.web.webhooks.base_webhook import WebhookProvider
|
||||||
|
|
||||||
|
|
||||||
|
class DiscordWebhook(WebhookProvider):
|
||||||
|
def _construct_discord_payload(self, server_name, title, message, color, bot_name):
|
||||||
|
"""
|
||||||
|
Constructs the payload required for sending a Discord webhook notification.
|
||||||
|
|
||||||
|
This method prepares a payload for the Discord webhook API using the provided
|
||||||
|
message content, the Crafty Controller version, and the current UTC datetime.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
message (str): The main content of the notification message.
|
||||||
|
color (int): The color code for the side stripe in the Discord embed message.
|
||||||
|
bot_name (str): Override for the Webhook's name set on creation
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
tuple: A tuple containing the constructed payload (dict) incl headers (dict).
|
||||||
|
|
||||||
|
Note:
|
||||||
|
- Discord embed designer
|
||||||
|
- https://discohook.org/
|
||||||
|
"""
|
||||||
|
current_datetime = datetime.utcnow()
|
||||||
|
formatted_datetime = (
|
||||||
|
current_datetime.strftime("%Y-%m-%dT%H:%M:%S.%f")[:-3] + "Z"
|
||||||
|
)
|
||||||
|
|
||||||
|
# Convert the hex to an integer
|
||||||
|
sanitized_hex = color[1:] if color.startswith("#") else color
|
||||||
|
color_int = int(sanitized_hex, 16)
|
||||||
|
|
||||||
|
headers = {"Content-type": "application/json"}
|
||||||
|
payload = {
|
||||||
|
"username": bot_name,
|
||||||
|
"avatar_url": self.WEBHOOK_PFP_URL,
|
||||||
|
"embeds": [
|
||||||
|
{
|
||||||
|
"title": title,
|
||||||
|
"description": message,
|
||||||
|
"color": color_int,
|
||||||
|
"author": {"name": server_name},
|
||||||
|
"footer": {"text": f"Crafty Controller v.{self.CRAFTY_VERSION}"},
|
||||||
|
"timestamp": formatted_datetime,
|
||||||
|
}
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
return payload, headers
|
||||||
|
|
||||||
|
def send(self, server_name, title, url, message, **kwargs):
|
||||||
|
"""
|
||||||
|
Sends a Discord webhook notification using the given details.
|
||||||
|
|
||||||
|
The method constructs and dispatches a payload suitable for
|
||||||
|
Discords's webhook system.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
url (str): The webhook URL to send the notification to.
|
||||||
|
message (str): The main content or body of the notification message.
|
||||||
|
color (str, optional): The color code for the embed's side stripe.
|
||||||
|
Defaults to a pretty blue if not provided.
|
||||||
|
bot_name (str): Override for the Webhook's name set on creation
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: "Dispatch successful!" if the message is sent successfully, otherwise an
|
||||||
|
exception is raised.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
Exception: If there's an error in dispatching the webhook.
|
||||||
|
"""
|
||||||
|
color = kwargs.get("color", "#005cd1") # Default to a color if not provided.
|
||||||
|
bot_name = kwargs.get("bot_name", self.WEBHOOK_USERNAME)
|
||||||
|
payload, headers = self._construct_discord_payload(
|
||||||
|
server_name, title, message, color, bot_name
|
||||||
|
)
|
||||||
|
return self._send_request(url, payload, headers)
|
74
app/classes/web/webhooks/mattermost_webhook.py
Normal file
74
app/classes/web/webhooks/mattermost_webhook.py
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
from app.classes.web.webhooks.base_webhook import WebhookProvider
|
||||||
|
|
||||||
|
|
||||||
|
class MattermostWebhook(WebhookProvider):
|
||||||
|
def _construct_mattermost_payload(self, server_name, title, message, bot_name):
|
||||||
|
"""
|
||||||
|
Constructs the payload required for sending a Mattermost webhook notification.
|
||||||
|
|
||||||
|
The method formats the given information into a Markdown-styled message for MM,
|
||||||
|
including an information card containing the Crafty version.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
message (str): The main content of the notification message.
|
||||||
|
bot_name (str): Override for the Webhook's name set on creation.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
tuple: A tuple containing the constructed payload (dict) incl headers (dict).
|
||||||
|
"""
|
||||||
|
formatted_text = (
|
||||||
|
f"-----\n\n"
|
||||||
|
f"#### {title}\n"
|
||||||
|
f"##### Server: ```{server_name}```\n\n"
|
||||||
|
f"```\n{message}\n```\n\n"
|
||||||
|
f"-----"
|
||||||
|
)
|
||||||
|
|
||||||
|
headers = {"Content-Type": "application/json"}
|
||||||
|
payload = {
|
||||||
|
"text": formatted_text,
|
||||||
|
"username": bot_name,
|
||||||
|
"icon_url": self.WEBHOOK_PFP_URL,
|
||||||
|
"props": {
|
||||||
|
"card": (
|
||||||
|
f"[Crafty Controller "
|
||||||
|
f"v.{self.CRAFTY_VERSION}](https://craftycontrol.com)"
|
||||||
|
)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
return payload, headers
|
||||||
|
|
||||||
|
def send(self, server_name, title, url, message, **kwargs):
|
||||||
|
"""
|
||||||
|
Sends a Mattermost webhook notification using the given details.
|
||||||
|
|
||||||
|
The method constructs and dispatches a payload suitable for
|
||||||
|
Mattermost's webhook system.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
url (str): The webhook URL to send the notification to.
|
||||||
|
message (str): The main content or body of the notification message.
|
||||||
|
bot_name (str): Override for the Webhook's name set on creation, see note!
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: "Dispatch successful!" if the message is sent successfully, otherwise an
|
||||||
|
exception is raised.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
Exception: If there's an error in dispatching the webhook.
|
||||||
|
|
||||||
|
Note:
|
||||||
|
- To set webhook username & pfp Mattermost needs to be configured to allow this!
|
||||||
|
- Mattermost's `config.json` setting is `"EnablePostUsernameOverride": true`
|
||||||
|
- Mattermost's `config.json` setting is `"EnablePostIconOverride": true`
|
||||||
|
"""
|
||||||
|
bot_name = kwargs.get("bot_name", self.WEBHOOK_USERNAME)
|
||||||
|
payload, headers = self._construct_mattermost_payload(
|
||||||
|
server_name, title, message, bot_name
|
||||||
|
)
|
||||||
|
return self._send_request(url, payload, headers)
|
98
app/classes/web/webhooks/slack_webhook.py
Normal file
98
app/classes/web/webhooks/slack_webhook.py
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
from app.classes.web.webhooks.base_webhook import WebhookProvider
|
||||||
|
|
||||||
|
|
||||||
|
class SlackWebhook(WebhookProvider):
|
||||||
|
def _construct_slack_payload(self, server_name, title, message, color, bot_name):
|
||||||
|
"""
|
||||||
|
Constructs the payload required for sending a Slack webhook notification.
|
||||||
|
|
||||||
|
The method formats the given information into a Markdown-styled message for MM,
|
||||||
|
including an information card containing the Crafty version.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
message (str): The main content of the notification message.
|
||||||
|
color (int): The color code for the side stripe in the Slack block.
|
||||||
|
bot_name (str): Override for the Webhook's name set on creation, (not working).
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
tuple: A tuple containing the constructed payload (dict) incl headers (dict).
|
||||||
|
|
||||||
|
Note:
|
||||||
|
- Block Builder/designer
|
||||||
|
- https://app.slack.com/block-kit-builder/
|
||||||
|
"""
|
||||||
|
headers = {"Content-Type": "application/json"}
|
||||||
|
payload = {
|
||||||
|
"username": bot_name,
|
||||||
|
"attachments": [
|
||||||
|
{
|
||||||
|
"color": color,
|
||||||
|
"blocks": [
|
||||||
|
{
|
||||||
|
"type": "section",
|
||||||
|
"text": {"type": "plain_text", "text": server_name},
|
||||||
|
},
|
||||||
|
{"type": "divider"},
|
||||||
|
{
|
||||||
|
"type": "section",
|
||||||
|
"text": {
|
||||||
|
"type": "mrkdwn",
|
||||||
|
"text": f"*{title}*\n{message}",
|
||||||
|
},
|
||||||
|
"accessory": {
|
||||||
|
"type": "image",
|
||||||
|
"image_url": self.WEBHOOK_PFP_URL,
|
||||||
|
"alt_text": "Crafty Controller Logo",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "context",
|
||||||
|
"elements": [
|
||||||
|
{
|
||||||
|
"type": "mrkdwn",
|
||||||
|
"text": (
|
||||||
|
f"*Crafty Controller "
|
||||||
|
f"v{self.CRAFTY_VERSION}*"
|
||||||
|
),
|
||||||
|
}
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{"type": "divider"},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
return payload, headers
|
||||||
|
|
||||||
|
def send(self, server_name, title, url, message, **kwargs):
|
||||||
|
"""
|
||||||
|
Sends a Slack webhook notification using the given details.
|
||||||
|
|
||||||
|
The method constructs and dispatches a payload suitable for
|
||||||
|
Slack's webhook system.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
url (str): The webhook URL to send the notification to.
|
||||||
|
message (str): The main content or body of the notification message.
|
||||||
|
color (str, optional): The color code for the blocks's colour accent.
|
||||||
|
Defaults to a pretty blue if not provided.
|
||||||
|
bot_name (str): Override for the Webhook's name set on creation, (not working).
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: "Dispatch successful!" if the message is sent successfully, otherwise an
|
||||||
|
exception is raised.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
Exception: If there's an error in dispatching the webhook.
|
||||||
|
"""
|
||||||
|
color = kwargs.get("color", "#005cd1") # Default to a color if not provided.
|
||||||
|
bot_name = kwargs.get("bot_name", self.WEBHOOK_USERNAME)
|
||||||
|
payload, headers = self._construct_slack_payload(
|
||||||
|
server_name, title, message, color, bot_name
|
||||||
|
)
|
||||||
|
return self._send_request(url, payload, headers)
|
126
app/classes/web/webhooks/teams_adaptive_webhook.py
Normal file
126
app/classes/web/webhooks/teams_adaptive_webhook.py
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
from datetime import datetime
|
||||||
|
from app.classes.web.webhooks.base_webhook import WebhookProvider
|
||||||
|
|
||||||
|
|
||||||
|
class TeamsWebhook(WebhookProvider):
|
||||||
|
def _construct_teams_payload(self, server_name, title, message):
|
||||||
|
"""
|
||||||
|
Constructs the payload required for sending a Teams Adaptive card notification.
|
||||||
|
|
||||||
|
This method prepares a payload for the Teams webhook API using the provided
|
||||||
|
message content, the Crafty Controller version, and the current UTC datetime.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
message (str): The main content of the notification message.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
tuple: A tuple containing the constructed payload (dict) incl headers (dict).
|
||||||
|
|
||||||
|
Note:
|
||||||
|
- Adaptive Card Designer
|
||||||
|
- https://www.adaptivecards.io/designer/
|
||||||
|
"""
|
||||||
|
current_datetime = datetime.utcnow()
|
||||||
|
formatted_datetime = current_datetime.strftime("%Y-%m-%dT%H:%M:%SZ")
|
||||||
|
|
||||||
|
headers = {"Content-type": "application/json"}
|
||||||
|
payload = {
|
||||||
|
"type": "message",
|
||||||
|
"attachments": [
|
||||||
|
{
|
||||||
|
"contentType": "application/vnd.microsoft.card.adaptive",
|
||||||
|
"content": {
|
||||||
|
"body": [
|
||||||
|
{
|
||||||
|
"type": "TextBlock",
|
||||||
|
"size": "Medium",
|
||||||
|
"weight": "Bolder",
|
||||||
|
"text": f"{title}",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "ColumnSet",
|
||||||
|
"columns": [
|
||||||
|
{
|
||||||
|
"type": "Column",
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"type": "Image",
|
||||||
|
"style": "Person",
|
||||||
|
"url": f"{self.WEBHOOK_PFP_URL}",
|
||||||
|
"size": "Small",
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"width": "auto",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "Column",
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"type": "TextBlock",
|
||||||
|
"weight": "Bolder",
|
||||||
|
"text": f"{server_name}",
|
||||||
|
"wrap": True,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "TextBlock",
|
||||||
|
"spacing": "None",
|
||||||
|
"text": "{{DATE("
|
||||||
|
+ f"{formatted_datetime}"
|
||||||
|
+ ",SHORT)}}",
|
||||||
|
"isSubtle": True,
|
||||||
|
"wrap": True,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"width": "stretch",
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "TextBlock",
|
||||||
|
"text": f"{message}",
|
||||||
|
"wrap": True,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "TextBlock",
|
||||||
|
"text": f"Crafty Controller v{self.CRAFTY_VERSION}",
|
||||||
|
"wrap": True,
|
||||||
|
"separator": True,
|
||||||
|
"isSubtle": True,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
"$schema": (
|
||||||
|
"https://adaptivecards.io/schemas/adaptive-card.json"
|
||||||
|
),
|
||||||
|
"version": "1.6",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
return payload, headers
|
||||||
|
|
||||||
|
def send(self, server_name, title, url, message, **kwargs):
|
||||||
|
"""
|
||||||
|
Sends a Teams Adaptive card notification using the given details.
|
||||||
|
|
||||||
|
The method constructs and dispatches a payload suitable for
|
||||||
|
Discords's webhook system.
|
||||||
|
|
||||||
|
Parameters:
|
||||||
|
server_name (str): The name of the server triggering the notification.
|
||||||
|
title (str): The title for the notification message.
|
||||||
|
url (str): The webhook URL to send the notification to.
|
||||||
|
message (str): The main content or body of the notification message.
|
||||||
|
Defaults to a pretty blue if not provided.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: "Dispatch successful!" if the message is sent successfully, otherwise an
|
||||||
|
exception is raised.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
Exception: If there's an error in dispatching the webhook.
|
||||||
|
"""
|
||||||
|
payload, headers = self._construct_teams_payload(server_name, title, message)
|
||||||
|
return self._send_request(url, payload, headers)
|
85
app/classes/web/webhooks/webhook_factory.py
Normal file
85
app/classes/web/webhooks/webhook_factory.py
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
from app.classes.web.webhooks.discord_webhook import DiscordWebhook
|
||||||
|
from app.classes.web.webhooks.mattermost_webhook import MattermostWebhook
|
||||||
|
from app.classes.web.webhooks.slack_webhook import SlackWebhook
|
||||||
|
from app.classes.web.webhooks.teams_adaptive_webhook import TeamsWebhook
|
||||||
|
|
||||||
|
|
||||||
|
class WebhookFactory:
|
||||||
|
"""
|
||||||
|
A factory class responsible for the creation and management of webhook providers.
|
||||||
|
|
||||||
|
This class provides methods to instantiate specific webhook providers based on
|
||||||
|
their name and to retrieve a list of supported providers. It uses a registry pattern
|
||||||
|
to manage the available providers.
|
||||||
|
|
||||||
|
Attributes:
|
||||||
|
- _registry (dict): A dictionary mapping provider names to their classes.
|
||||||
|
"""
|
||||||
|
|
||||||
|
_registry = {
|
||||||
|
"Discord": DiscordWebhook,
|
||||||
|
"Mattermost": MattermostWebhook,
|
||||||
|
"Slack": SlackWebhook,
|
||||||
|
"Teams": TeamsWebhook,
|
||||||
|
# "Custom",
|
||||||
|
}
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def create_provider(cls, provider_name, *args, **kwargs):
|
||||||
|
"""
|
||||||
|
Creates and returns an instance of the specified webhook provider.
|
||||||
|
|
||||||
|
This method looks up the provider in the registry, then instantiates it w/ the
|
||||||
|
provided arguments. If the provider is not recognized, a ValueError is raised.
|
||||||
|
|
||||||
|
Arguments:
|
||||||
|
- provider_name (str): The name of the desired webhook provider.
|
||||||
|
|
||||||
|
Additional arguments supported that we may use for if a provider
|
||||||
|
requires initialization:
|
||||||
|
- *args: Positional arguments to pass to the provider's constructor.
|
||||||
|
- **kwargs: Keyword arguments to pass to the provider's constructor.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
WebhookProvider: An instance of the desired webhook provider.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: If the specified provider name is not recognized.
|
||||||
|
"""
|
||||||
|
if provider_name not in cls._registry:
|
||||||
|
raise ValueError(f"Provider {provider_name} is not supported.")
|
||||||
|
return cls._registry[provider_name](*args, **kwargs)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_supported_providers(cls):
|
||||||
|
"""
|
||||||
|
Retrieves the names of all supported webhook providers.
|
||||||
|
|
||||||
|
This method returns a list containing the names of all providers
|
||||||
|
currently registered in the factory's registry.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List[str]: A list of supported provider names.
|
||||||
|
"""
|
||||||
|
return list(cls._registry.keys())
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def get_monitored_events():
|
||||||
|
"""
|
||||||
|
Retrieves the list of supported events for monitoring.
|
||||||
|
|
||||||
|
This method provides a list of common server events that the webhook system can
|
||||||
|
monitor and notify about.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List[str]: A list of supported monitored actions.
|
||||||
|
"""
|
||||||
|
return [
|
||||||
|
"start_server",
|
||||||
|
"stop_server",
|
||||||
|
"crash_detected",
|
||||||
|
"backup_server",
|
||||||
|
"jar_update",
|
||||||
|
"send_command",
|
||||||
|
"kill",
|
||||||
|
]
|
@ -4,26 +4,34 @@ import asyncio
|
|||||||
from urllib.parse import parse_qsl
|
from urllib.parse import parse_qsl
|
||||||
import tornado.websocket
|
import tornado.websocket
|
||||||
|
|
||||||
|
from app.classes.shared.main_controller import Controller
|
||||||
from app.classes.shared.helpers import Helpers
|
from app.classes.shared.helpers import Helpers
|
||||||
|
from app.classes.shared.websocket_manager import WebSocketManager
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class SocketHandler(tornado.websocket.WebSocketHandler):
|
class WebSocketHandler(tornado.websocket.WebSocketHandler):
|
||||||
page = None
|
page = None
|
||||||
page_query_params = None
|
page_query_params = None
|
||||||
controller = None
|
controller: Controller = None
|
||||||
tasks_manager = None
|
tasks_manager = None
|
||||||
translator = None
|
translator = None
|
||||||
io_loop = None
|
io_loop = None
|
||||||
|
|
||||||
def initialize(
|
def initialize(
|
||||||
self, helper=None, controller=None, tasks_manager=None, translator=None
|
self,
|
||||||
|
helper=None,
|
||||||
|
controller=None,
|
||||||
|
tasks_manager=None,
|
||||||
|
translator=None,
|
||||||
|
file_helper=None,
|
||||||
):
|
):
|
||||||
self.helper = helper
|
self.helper = helper
|
||||||
self.controller = controller
|
self.controller = controller
|
||||||
self.tasks_manager = tasks_manager
|
self.tasks_manager = tasks_manager
|
||||||
self.translator = translator
|
self.translator = translator
|
||||||
|
self.file_helper = file_helper
|
||||||
self.io_loop = tornado.ioloop.IOLoop.current()
|
self.io_loop = tornado.ioloop.IOLoop.current()
|
||||||
|
|
||||||
def get_remote_ip(self):
|
def get_remote_ip(self):
|
||||||
@ -34,23 +42,16 @@ class SocketHandler(tornado.websocket.WebSocketHandler):
|
|||||||
)
|
)
|
||||||
return remote_ip
|
return remote_ip
|
||||||
|
|
||||||
def get_user_id(self):
|
|
||||||
_, _, user = self.controller.authentication.check(self.get_cookie("token"))
|
|
||||||
return user["user_id"]
|
|
||||||
|
|
||||||
def check_auth(self):
|
|
||||||
return self.controller.authentication.check_bool(self.get_cookie("token"))
|
|
||||||
|
|
||||||
# pylint: disable=arguments-differ
|
# pylint: disable=arguments-differ
|
||||||
def open(self):
|
def open(self):
|
||||||
logger.debug("Checking WebSocket authentication")
|
logger.debug("Checking WebSocket authentication")
|
||||||
if self.check_auth():
|
if self.check_auth():
|
||||||
self.handle()
|
self.handle()
|
||||||
else:
|
else:
|
||||||
self.helper.websocket_helper.send_message(
|
WebSocketManager().broadcast_to_admins(
|
||||||
self, "notification", "Not authenticated for WebSocket connection"
|
self, "notification", "Not authenticated for WebSocket connection"
|
||||||
)
|
)
|
||||||
self.close()
|
self.close(1011, "Forbidden WS Access")
|
||||||
self.controller.management.add_to_audit_log_raw(
|
self.controller.management.add_to_audit_log_raw(
|
||||||
"unknown",
|
"unknown",
|
||||||
0,
|
0,
|
||||||
@ -58,7 +59,7 @@ class SocketHandler(tornado.websocket.WebSocketHandler):
|
|||||||
"Someone tried to connect via WebSocket without proper authentication",
|
"Someone tried to connect via WebSocket without proper authentication",
|
||||||
self.get_remote_ip(),
|
self.get_remote_ip(),
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.broadcast(
|
WebSocketManager().broadcast(
|
||||||
"notification",
|
"notification",
|
||||||
"Someone tried to connect via WebSocket without proper authentication",
|
"Someone tried to connect via WebSocket without proper authentication",
|
||||||
)
|
)
|
||||||
@ -73,24 +74,34 @@ class SocketHandler(tornado.websocket.WebSocketHandler):
|
|||||||
Helpers.remove_prefix(self.get_query_argument("page_query_params"), "?")
|
Helpers.remove_prefix(self.get_query_argument("page_query_params"), "?")
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
self.helper.websocket_helper.add_client(self)
|
WebSocketManager().add_client(self)
|
||||||
logger.debug("Opened WebSocket connection")
|
logger.debug("Opened WebSocket connection")
|
||||||
|
|
||||||
# pylint: disable=arguments-renamed
|
# pylint: disable=arguments-renamed
|
||||||
@staticmethod
|
def on_message(self, raw_message):
|
||||||
def on_message(raw_message):
|
|
||||||
logger.debug(f"Got message from WebSocket connection {raw_message}")
|
logger.debug(f"Got message from WebSocket connection {raw_message}")
|
||||||
message = json.loads(raw_message)
|
message = json.loads(raw_message)
|
||||||
logger.debug(f"Event Type: {message['event']}, Data: {message['data']}")
|
logger.debug(f"Event Type: {message['event']}, Data: {message['data']}")
|
||||||
|
|
||||||
def on_close(self):
|
def on_close(self):
|
||||||
self.helper.websocket_helper.remove_client(self)
|
WebSocketManager().remove_client(self)
|
||||||
logger.debug("Closed WebSocket connection")
|
logger.debug("Closed WebSocket connection")
|
||||||
|
|
||||||
async def write_message_int(self, message):
|
async def write_message_int(self, message):
|
||||||
self.write_message(message)
|
self.write_message(message)
|
||||||
|
|
||||||
def write_message_helper(self, message):
|
def write_message_async(self, message):
|
||||||
asyncio.run_coroutine_threadsafe(
|
asyncio.run_coroutine_threadsafe(
|
||||||
self.write_message_int(message), self.io_loop.asyncio_loop
|
self.write_message_int(message), self.io_loop.asyncio_loop
|
||||||
)
|
)
|
||||||
|
|
||||||
|
def send_message(self, event_type: str, data):
|
||||||
|
message = str(json.dumps({"event": event_type, "data": data}))
|
||||||
|
self.write_message_async(message)
|
||||||
|
|
||||||
|
def get_user_id(self):
|
||||||
|
_, _, user = self.controller.authentication.check(self.get_cookie("token"))
|
||||||
|
return user["user_id"]
|
||||||
|
|
||||||
|
def check_auth(self):
|
||||||
|
return self.controller.authentication.check_bool(self.get_cookie("token"))
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"major": 4,
|
"major": 4,
|
||||||
"minor": 1,
|
"minor": 2,
|
||||||
"sub": 3
|
"sub": 0
|
||||||
}
|
}
|
||||||
|
@ -22979,27 +22979,42 @@ ul li {
|
|||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled {
|
.nav-tabs.tab-simple-styled {
|
||||||
border-bottom: none;
|
border-bottom: none;
|
||||||
margin-top: 30px;
|
margin-top: 30px;
|
||||||
margin-bottom: 30px;
|
margin-bottom: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled .nav-item {
|
.nav-tabs.tab-simple-styled .nav-item {
|
||||||
margin-right: 30px;
|
margin-right: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled .nav-item .nav-link {
|
.nav-tabs.tab-simple-styled .nav-item .nav-link {
|
||||||
border: none;
|
border: none;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
color: var(--base-text);
|
color: var(--base-text);
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled .nav-item .nav-link.active {
|
.nav-tabs.tab-simple-styled .nav-item .nav-link.active {
|
||||||
background: var(--dropdown-bg);
|
background: var(--dropdown-bg);
|
||||||
color: var(--info);
|
color: var(--info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.nav-pills.tab-simple-styled {
|
||||||
|
border-bottom: none;
|
||||||
|
/*margin-top: 1.5rem;*/
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*.nav-pills.tab-simple-styled .nav-item {
|
||||||
|
margin-right: 1.5rem;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
.nav-pills.tab-simple-styled .nav-item .nav-link.active {
|
||||||
|
background: var(--info);
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
.tab-tile-style {
|
.tab-tile-style {
|
||||||
display: -webkit-box;
|
display: -webkit-box;
|
||||||
display: -ms-flexbox;
|
display: -ms-flexbox;
|
||||||
|
@ -21494,27 +21494,42 @@ ul li {
|
|||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled {
|
.nav-tabs.tab-simple-styled {
|
||||||
border-bottom: none;
|
border-bottom: none;
|
||||||
margin-top: 30px;
|
margin-top: 30px;
|
||||||
margin-bottom: 30px;
|
margin-bottom: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled .nav-item {
|
.nav-tabs.tab-simple-styled .nav-item {
|
||||||
margin-right: 30px;
|
margin-right: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled .nav-item .nav-link {
|
.nav-tabs.tab-simple-styled .nav-item .nav-link {
|
||||||
border: none;
|
border: none;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
color: var(--gray);
|
color: var(--gray);
|
||||||
}
|
}
|
||||||
|
|
||||||
.tab-simple-styled .nav-item .nav-link.active {
|
.nav-tabs.tab-simple-styled .nav-item .nav-link.active {
|
||||||
background: #fff;
|
background: #fff;
|
||||||
color: var(--info);
|
color: var(--info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.nav-pills.tab-simple-styled {
|
||||||
|
border-bottom: none;
|
||||||
|
/*margin-top: 1.5rem;*/
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*.nav-pills.tab-simple-styled .nav-item {
|
||||||
|
margin-right: 1.5rem;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
.nav-pills.tab-simple-styled .nav-item .nav-link.active {
|
||||||
|
background: var(--info);
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
|
||||||
.tab-tile-style {
|
.tab-tile-style {
|
||||||
display: -webkit-box;
|
display: -webkit-box;
|
||||||
display: -ms-flexbox;
|
display: -ms-flexbox;
|
||||||
|
137
app/frontend/static/assets/js/shared/root-dir.js
Normal file
137
app/frontend/static/assets/js/shared/root-dir.js
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
|
||||||
|
function show_file_tree() {
|
||||||
|
$("#dir_select").modal();
|
||||||
|
}
|
||||||
|
function getDirView(event = false) {
|
||||||
|
if (event) {
|
||||||
|
try {
|
||||||
|
let path = event.target.parentElement.getAttribute('data-path');
|
||||||
|
if (event.target.parentElement.classList.contains('clicked')) {
|
||||||
|
|
||||||
|
if ($(`#${path}span`).hasClass('files-tree-title')) {
|
||||||
|
$(`#${path}ul`).toggleClass("d-block");
|
||||||
|
$(`#${path}span`).toggleClass("tree-caret-down");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
getTreeView(path);
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
console.log("Well that failed");
|
||||||
|
}
|
||||||
|
} else if ($("#root_files_button").hasClass("clicked")) {
|
||||||
|
getTreeView($("#zip_server_path").val(), true);
|
||||||
|
} else {
|
||||||
|
getTreeView($("#file-uploaded").val(), true, true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async function getTreeView(path, unzip = false, upload = false) {
|
||||||
|
const token = getCookie("_xsrf");
|
||||||
|
console.log("IN TREE VIEW")
|
||||||
|
console.log({ "page": "import", "folder": path, "upload": upload, "unzip": unzip });
|
||||||
|
let res = await fetch(`/api/v2/import/file/unzip/`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: JSON.stringify({ "page": "import", "folder": path, "upload": upload, "unzip": unzip }),
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
console.log(responseData);
|
||||||
|
process_tree_response(responseData);
|
||||||
|
let x = document.querySelector('.bootbox');
|
||||||
|
if (x) {
|
||||||
|
x.remove()
|
||||||
|
}
|
||||||
|
x = document.querySelector('.modal-backdrop');
|
||||||
|
if (x) {
|
||||||
|
x.remove()
|
||||||
|
}
|
||||||
|
show_file_tree();
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function process_tree_response(response) {
|
||||||
|
const styles = window.getComputedStyle(document.getElementById("lower_half"));
|
||||||
|
//If this value is still hidden we know the user is executing a zip import and not an upload
|
||||||
|
if (styles.visibility === "hidden") {
|
||||||
|
document.getElementById('zip_submit').disabled = false;
|
||||||
|
} else {
|
||||||
|
document.getElementById('upload_submit').disabled = false;
|
||||||
|
}
|
||||||
|
let path = response.data.root_path.path;
|
||||||
|
$(".root-input").val(response.data.root_path.path);
|
||||||
|
let text = `<ul class="tree-nested d-block" id="${path}ul">`;
|
||||||
|
Object.entries(response.data).forEach(([key, value]) => {
|
||||||
|
if (key === "root_path" || key === "db_stats") {
|
||||||
|
//continue is not valid in for each. Return acts as a continue.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let dpath = value.path;
|
||||||
|
let filename = key;
|
||||||
|
if (value.dir) {
|
||||||
|
text += `<li class="tree-item" id="${dpath}li" data-path="${dpath}">
|
||||||
|
<div id="${dpath}" data-path="${dpath}" data-name="${filename}" class="tree-caret tree-ctx-item tree-folder">
|
||||||
|
<input type="radio" name="root_path" value="${dpath}">
|
||||||
|
<span id="${dpath}span" class="files-tree-title" data-path="${dpath}" data-name="${filename}" onclick="getDirView(event)">
|
||||||
|
<i style="color: var(--info);" class="far fa-folder"></i>
|
||||||
|
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
||||||
|
${filename}
|
||||||
|
</span>
|
||||||
|
</input></div><li>`
|
||||||
|
} else {
|
||||||
|
text += `<li
|
||||||
|
class="d-block tree-ctx-item tree-file"
|
||||||
|
data-path="${dpath}"
|
||||||
|
data-name="${filename}"
|
||||||
|
onclick="" id="${dpath}li"><input type='radio' class="checkBoxClass d-none file-check" name='root_path' value="${dpath}" disabled><span style="margin-right: 6px;">
|
||||||
|
<i class="far fa-file"></i></span></input>${filename}</li>
|
||||||
|
`
|
||||||
|
}
|
||||||
|
});
|
||||||
|
text += `</ul>`;
|
||||||
|
|
||||||
|
if (response.data.root_path.top) {
|
||||||
|
try {
|
||||||
|
document.getElementById('main-tree-div').innerHTML += text;
|
||||||
|
document.getElementById('main-tree').parentElement.classList.add("clicked");
|
||||||
|
} catch {
|
||||||
|
document.getElementById('files-tree').innerHTML = text;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
document.getElementById(path + "span").classList.add('tree-caret-down');
|
||||||
|
document.getElementById(path).innerHTML += text;
|
||||||
|
document.getElementById(path).classList.add("clicked");
|
||||||
|
} catch {
|
||||||
|
console.log("Bad")
|
||||||
|
}
|
||||||
|
|
||||||
|
let toggler = document.getElementById(path + "span");
|
||||||
|
|
||||||
|
if (toggler.classList.contains('files-tree-title')) {
|
||||||
|
document.getElementById(path + "span").addEventListener("click", function caretListener() {
|
||||||
|
document.getElementById(path + "ul").classList.toggle("d-block");
|
||||||
|
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getToggleMain(event) {
|
||||||
|
const path = event.target.parentElement.getAttribute('data-path');
|
||||||
|
document.getElementById("files-tree").classList.toggle("d-block");
|
||||||
|
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
||||||
|
document.getElementById(path + "span").classList.toggle("tree-caret");
|
||||||
|
}
|
@ -1,46 +1,14 @@
|
|||||||
// This is the "Offline page" service worker
|
// This is the "Offline page" service worker
|
||||||
|
|
||||||
importScripts('https://storage.googleapis.com/workbox-cdn/releases/5.1.2/workbox-sw.js');
|
importScripts(
|
||||||
|
"https://storage.googleapis.com/workbox-cdn/releases/5.1.2/workbox-sw.js"
|
||||||
|
);
|
||||||
|
|
||||||
const CACHE = "crafty-controller";
|
const CACHE = "crafty-controller";
|
||||||
|
|
||||||
// TODO: replace the following with the correct offline fallback page i.e.: const offlineFallbackPage = "offline.html";
|
//This service worker is basically just here to make browsers
|
||||||
const offlineFallbackPage = "/offline";
|
//accept the PWA. It's not doing much anymore
|
||||||
|
|
||||||
self.addEventListener("message", (event) => {
|
|
||||||
if (event.data && event.data.type === "SKIP_WAITING") {
|
|
||||||
self.skipWaiting();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
self.addEventListener('install', async (event) => {
|
|
||||||
event.waitUntil(
|
|
||||||
caches.open(CACHE)
|
|
||||||
.then((cache) => cache.add(offlineFallbackPage))
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
if (workbox.navigationPreload.isSupported()) {
|
if (workbox.navigationPreload.isSupported()) {
|
||||||
workbox.navigationPreload.enable();
|
workbox.navigationPreload.enable();
|
||||||
}
|
}
|
||||||
|
|
||||||
self.addEventListener('fetch', (event) => {
|
|
||||||
if (event.request.mode === 'navigate') {
|
|
||||||
event.respondWith((async () => {
|
|
||||||
try {
|
|
||||||
const preloadResp = await event.preloadResponse;
|
|
||||||
|
|
||||||
if (preloadResp) {
|
|
||||||
return preloadResp;
|
|
||||||
}
|
|
||||||
const networkResp = await fetch(event.request);
|
|
||||||
return networkResp;
|
|
||||||
} catch (error) {
|
|
||||||
|
|
||||||
const cache = await caches.open(CACHE);
|
|
||||||
const cachedResp = await cache.match(offlineFallbackPage);
|
|
||||||
return cachedResp;
|
|
||||||
}
|
|
||||||
})());
|
|
||||||
}
|
|
||||||
});
|
|
28
app/frontend/static/assets/vendors/css/bootstrap-toggle.min.css
vendored
Normal file
28
app/frontend/static/assets/vendors/css/bootstrap-toggle.min.css
vendored
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
/*! ========================================================================
|
||||||
|
* Bootstrap Toggle: bootstrap-toggle.css v2.2.0
|
||||||
|
* http://www.bootstraptoggle.com
|
||||||
|
* ========================================================================
|
||||||
|
* Copyright 2014 Min Hur, The New York Times Company
|
||||||
|
* Licensed under MIT
|
||||||
|
* ======================================================================== */
|
||||||
|
.checkbox label .toggle,.checkbox-inline .toggle{margin-left:-20px;margin-right:5px}
|
||||||
|
.toggle{position:relative;overflow:hidden}
|
||||||
|
.toggle input[type=checkbox]{display:none}
|
||||||
|
.toggle-group{position:absolute;width:200%;top:0;bottom:0;left:0;transition:left .35s;-webkit-transition:left .35s;-moz-user-select:none;-webkit-user-select:none}
|
||||||
|
.toggle.off .toggle-group{left:-100%}
|
||||||
|
.toggle-on{position:absolute;top:0;bottom:0;left:0;right:50%;margin:0;border:0;border-radius:0}
|
||||||
|
.toggle-off{position:absolute;top:0;bottom:0;left:50%;right:0;margin:0;border:0;border-radius:0}
|
||||||
|
.toggle-handle{position:relative;margin:0 auto;padding-top:0;padding-bottom:0;height:100%;width:0;border-width:0 1px}
|
||||||
|
.toggle.btn{min-width:59px;min-height:34px}
|
||||||
|
.toggle-on.btn{padding-right:24px}
|
||||||
|
.toggle-off.btn{padding-left:24px}
|
||||||
|
.toggle.btn-lg{min-width:79px;min-height:45px}
|
||||||
|
.toggle-on.btn-lg{padding-right:31px}
|
||||||
|
.toggle-off.btn-lg{padding-left:31px}
|
||||||
|
.toggle-handle.btn-lg{width:40px}
|
||||||
|
.toggle.btn-sm{min-width:50px;min-height:30px}
|
||||||
|
.toggle-on.btn-sm{padding-right:20px}
|
||||||
|
.toggle-off.btn-sm{padding-left:20px}
|
||||||
|
.toggle.btn-xs{min-width:35px;min-height:22px}
|
||||||
|
.toggle-on.btn-xs{padding-right:12px}
|
||||||
|
.toggle-off.btn-xs{padding-left:12px}
|
1
app/frontend/static/assets/vendors/js/bootbox.min.js
vendored
Normal file
1
app/frontend/static/assets/vendors/js/bootbox.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
9
app/frontend/static/assets/vendors/js/bootstrap-select.min.js
vendored
Normal file
9
app/frontend/static/assets/vendors/js/bootstrap-select.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
9
app/frontend/static/assets/vendors/js/bootstrap-toggle.min.js
vendored
Normal file
9
app/frontend/static/assets/vendors/js/bootstrap-toggle.min.js
vendored
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
/*! ========================================================================
|
||||||
|
* Bootstrap Toggle: bootstrap-toggle.js v2.2.0
|
||||||
|
* http://www.bootstraptoggle.com
|
||||||
|
* ========================================================================
|
||||||
|
* Copyright 2014 Min Hur, The New York Times Company
|
||||||
|
* Licensed under MIT
|
||||||
|
* ======================================================================== */
|
||||||
|
+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.toggle"),f="object"==typeof b&&b;e||d.data("bs.toggle",e=new c(this,f)),"string"==typeof b&&e[b]&&e[b]()})}var c=function(b,c){this.$element=a(b),this.options=a.extend({},this.defaults(),c),this.render()};c.VERSION="2.2.0",c.DEFAULTS={on:"On",off:"Off",onstyle:"primary",offstyle:"default",size:"normal",style:"",width:null,height:null},c.prototype.defaults=function(){return{on:this.$element.attr("data-on")||c.DEFAULTS.on,off:this.$element.attr("data-off")||c.DEFAULTS.off,onstyle:this.$element.attr("data-onstyle")||c.DEFAULTS.onstyle,offstyle:this.$element.attr("data-offstyle")||c.DEFAULTS.offstyle,size:this.$element.attr("data-size")||c.DEFAULTS.size,style:this.$element.attr("data-style")||c.DEFAULTS.style,width:this.$element.attr("data-width")||c.DEFAULTS.width,height:this.$element.attr("data-height")||c.DEFAULTS.height}},c.prototype.render=function(){this._onstyle="btn-"+this.options.onstyle,this._offstyle="btn-"+this.options.offstyle;var b="large"===this.options.size?"btn-lg":"small"===this.options.size?"btn-sm":"mini"===this.options.size?"btn-xs":"",c=a('<label class="btn">').html(this.options.on).addClass(this._onstyle+" "+b),d=a('<label class="btn">').html(this.options.off).addClass(this._offstyle+" "+b+" active"),e=a('<span class="toggle-handle btn btn-default">').addClass(b),f=a('<div class="toggle-group">').append(c,d,e),g=a('<div class="toggle btn" data-toggle="toggle">').addClass(this.$element.prop("checked")?this._onstyle:this._offstyle+" off").addClass(b).addClass(this.options.style);this.$element.wrap(g),a.extend(this,{$toggle:this.$element.parent(),$toggleOn:c,$toggleOff:d,$toggleGroup:f}),this.$toggle.append(f);var h=this.options.width||Math.max(c.outerWidth(),d.outerWidth())+e.outerWidth()/2,i=this.options.height||Math.max(c.outerHeight(),d.outerHeight());c.addClass("toggle-on"),d.addClass("toggle-off"),this.$toggle.css({width:h,height:i}),this.options.height&&(c.css("line-height",c.height()+"px"),d.css("line-height",d.height()+"px")),this.update(!0),this.trigger(!0)},c.prototype.toggle=function(){this.$element.prop("checked")?this.off():this.on()},c.prototype.on=function(a){return this.$element.prop("disabled")?!1:(this.$toggle.removeClass(this._offstyle+" off").addClass(this._onstyle),this.$element.prop("checked",!0),void(a||this.trigger()))},c.prototype.off=function(a){return this.$element.prop("disabled")?!1:(this.$toggle.removeClass(this._onstyle).addClass(this._offstyle+" off"),this.$element.prop("checked",!1),void(a||this.trigger()))},c.prototype.enable=function(){this.$toggle.removeAttr("disabled"),this.$element.prop("disabled",!1)},c.prototype.disable=function(){this.$toggle.attr("disabled","disabled"),this.$element.prop("disabled",!0)},c.prototype.update=function(a){this.$element.prop("disabled")?this.disable():this.enable(),this.$element.prop("checked")?this.on(a):this.off(a)},c.prototype.trigger=function(b){this.$element.off("change.bs.toggle"),b||this.$element.change(),this.$element.on("change.bs.toggle",a.proxy(function(){this.update()},this))},c.prototype.destroy=function(){this.$element.off("change.bs.toggle"),this.$toggleGroup.remove(),this.$element.removeData("bs.toggle"),this.$element.unwrap()};var d=a.fn.bootstrapToggle;a.fn.bootstrapToggle=b,a.fn.bootstrapToggle.Constructor=c,a.fn.toggle.noConflict=function(){return a.fn.bootstrapToggle=d,this},a(function(){a("input[type=checkbox][data-toggle^=toggle]").bootstrapToggle()}),a(document).on("click.bs.toggle","div[data-toggle^=toggle]",function(b){var c=a(this).find("input[type=checkbox]");c.bootstrapToggle("toggle"),b.preventDefault()})}(jQuery);
|
||||||
|
//# sourceMappingURL=bootstrap-toggle.min.js.map
|
5
app/frontend/static/assets/vendors/js/cdn.min.js
vendored
Normal file
5
app/frontend/static/assets/vendors/js/cdn.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
13
app/frontend/static/assets/vendors/js/chart.min.js
vendored
Normal file
13
app/frontend/static/assets/vendors/js/chart.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
7
app/frontend/static/assets/vendors/js/chartjs-plugin-zoom.min.js
vendored
Normal file
7
app/frontend/static/assets/vendors/js/chartjs-plugin-zoom.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
491
app/frontend/static/assets/vendors/js/datatables.min.js
vendored
Normal file
491
app/frontend/static/assets/vendors/js/datatables.min.js
vendored
Normal file
@ -0,0 +1,491 @@
|
|||||||
|
/*
|
||||||
|
* This combined file was created by the DataTables downloader builder:
|
||||||
|
* https://datatables.net/download
|
||||||
|
*
|
||||||
|
* To rebuild or modify this file with the latest versions of the included
|
||||||
|
* software please visit:
|
||||||
|
* https://datatables.net/download/#bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2
|
||||||
|
*
|
||||||
|
* Included libraries:
|
||||||
|
* DataTables 1.10.22, FixedHeader 3.1.7, Responsive 2.2.6, Scroller 2.0.3, SearchPanes 1.2.2
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Copyright 2008-2020 SpryMedia Ltd.
|
||||||
|
|
||||||
|
This source file is free software, available under the following license:
|
||||||
|
MIT license - http://datatables.net/license
|
||||||
|
|
||||||
|
This source file is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the license files for details.
|
||||||
|
|
||||||
|
For details please refer to: http://www.datatables.net
|
||||||
|
DataTables 1.10.22
|
||||||
|
©2008-2020 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.findInternal=function(k,y,z){k instanceof String&&(k=String(k));for(var q=k.length,G=0;G<q;G++){var O=k[G];if(y.call(z,O,G,k))return{i:G,v:O}}return{i:-1,v:void 0}};$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;$jscomp.ISOLATE_POLYFILLS=!1;
|
||||||
|
$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(k,y,z){if(k==Array.prototype||k==Object.prototype)return k;k[y]=z.value;return k};$jscomp.getGlobal=function(k){k=["object"==typeof globalThis&&globalThis,k,"object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global];for(var y=0;y<k.length;++y){var z=k[y];if(z&&z.Math==Math)return z}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.IS_SYMBOL_NATIVE="function"===typeof Symbol&&"symbol"===typeof Symbol("x");$jscomp.TRUST_ES6_POLYFILLS=!$jscomp.ISOLATE_POLYFILLS||$jscomp.IS_SYMBOL_NATIVE;$jscomp.polyfills={};$jscomp.propertyToPolyfillSymbol={};$jscomp.POLYFILL_PREFIX="$jscp$";var $jscomp$lookupPolyfilledValue=function(k,y){var z=$jscomp.propertyToPolyfillSymbol[y];if(null==z)return k[y];z=k[z];return void 0!==z?z:k[y]};
|
||||||
|
$jscomp.polyfill=function(k,y,z,q){y&&($jscomp.ISOLATE_POLYFILLS?$jscomp.polyfillIsolated(k,y,z,q):$jscomp.polyfillUnisolated(k,y,z,q))};$jscomp.polyfillUnisolated=function(k,y,z,q){z=$jscomp.global;k=k.split(".");for(q=0;q<k.length-1;q++){var G=k[q];if(!(G in z))return;z=z[G]}k=k[k.length-1];q=z[k];y=y(q);y!=q&&null!=y&&$jscomp.defineProperty(z,k,{configurable:!0,writable:!0,value:y})};
|
||||||
|
$jscomp.polyfillIsolated=function(k,y,z,q){var G=k.split(".");k=1===G.length;q=G[0];q=!k&&q in $jscomp.polyfills?$jscomp.polyfills:$jscomp.global;for(var O=0;O<G.length-1;O++){var ma=G[O];if(!(ma in q))return;q=q[ma]}G=G[G.length-1];z=$jscomp.IS_SYMBOL_NATIVE&&"es6"===z?q[G]:null;y=y(z);null!=y&&(k?$jscomp.defineProperty($jscomp.polyfills,G,{configurable:!0,writable:!0,value:y}):y!==z&&($jscomp.propertyToPolyfillSymbol[G]=$jscomp.IS_SYMBOL_NATIVE?$jscomp.global.Symbol(G):$jscomp.POLYFILL_PREFIX+G,
|
||||||
|
G=$jscomp.propertyToPolyfillSymbol[G],$jscomp.defineProperty(q,G,{configurable:!0,writable:!0,value:y})))};$jscomp.polyfill("Array.prototype.find",function(k){return k?k:function(y,z){return $jscomp.findInternal(this,y,z).v}},"es6","es3");
|
||||||
|
(function(k){"function"===typeof define&&define.amd?define(["jquery"],function(y){return k(y,window,document)}):"object"===typeof exports?module.exports=function(y,z){y||(y=window);z||(z="undefined"!==typeof window?require("jquery"):require("jquery")(y));return k(z,y,y.document)}:k(jQuery,window,document)})(function(k,y,z,q){function G(a){var b,c,d={};k.each(a,function(f,e){(b=f.match(/^([^A-Z]+?)([A-Z])/))&&-1!=="a aa ai ao as b fn i m o s ".indexOf(b[1]+" ")&&(c=f.replace(b[0],b[2].toLowerCase()),
|
||||||
|
d[c]=f,"o"===b[1]&&G(a[f]))});a._hungarianMap=d}function O(a,b,c){a._hungarianMap||G(a);var d;k.each(b,function(f,e){d=a._hungarianMap[f];d===q||!c&&b[d]!==q||("o"===d.charAt(0)?(b[d]||(b[d]={}),k.extend(!0,b[d],b[f]),O(a[d],b[d],c)):b[d]=b[f])})}function ma(a){var b=u.defaults.oLanguage,c=b.sDecimal;c&&Va(c);if(a){var d=a.sZeroRecords;!a.sEmptyTable&&d&&"No data available in table"===b.sEmptyTable&&V(a,a,"sZeroRecords","sEmptyTable");!a.sLoadingRecords&&d&&"Loading..."===b.sLoadingRecords&&V(a,a,
|
||||||
|
"sZeroRecords","sLoadingRecords");a.sInfoThousands&&(a.sThousands=a.sInfoThousands);(a=a.sDecimal)&&c!==a&&Va(a)}}function yb(a){R(a,"ordering","bSort");R(a,"orderMulti","bSortMulti");R(a,"orderClasses","bSortClasses");R(a,"orderCellsTop","bSortCellsTop");R(a,"order","aaSorting");R(a,"orderFixed","aaSortingFixed");R(a,"paging","bPaginate");R(a,"pagingType","sPaginationType");R(a,"pageLength","iDisplayLength");R(a,"searching","bFilter");"boolean"===typeof a.sScrollX&&(a.sScrollX=a.sScrollX?"100%":
|
||||||
|
"");"boolean"===typeof a.scrollX&&(a.scrollX=a.scrollX?"100%":"");if(a=a.aoSearchCols)for(var b=0,c=a.length;b<c;b++)a[b]&&O(u.models.oSearch,a[b])}function zb(a){R(a,"orderable","bSortable");R(a,"orderData","aDataSort");R(a,"orderSequence","asSorting");R(a,"orderDataType","sortDataType");var b=a.aDataSort;"number"!==typeof b||Array.isArray(b)||(a.aDataSort=[b])}function Ab(a){if(!u.__browser){var b={};u.__browser=b;var c=k("<div/>").css({position:"fixed",top:0,left:-1*k(y).scrollLeft(),height:1,
|
||||||
|
width:1,overflow:"hidden"}).append(k("<div/>").css({position:"absolute",top:1,left:1,width:100,overflow:"scroll"}).append(k("<div/>").css({width:"100%",height:10}))).appendTo("body"),d=c.children(),f=d.children();b.barWidth=d[0].offsetWidth-d[0].clientWidth;b.bScrollOversize=100===f[0].offsetWidth&&100!==d[0].clientWidth;b.bScrollbarLeft=1!==Math.round(f.offset().left);b.bBounding=c[0].getBoundingClientRect().width?!0:!1;c.remove()}k.extend(a.oBrowser,u.__browser);a.oScroll.iBarWidth=u.__browser.barWidth}
|
||||||
|
function Bb(a,b,c,d,f,e){var g=!1;if(c!==q){var h=c;g=!0}for(;d!==f;)a.hasOwnProperty(d)&&(h=g?b(h,a[d],d,a):a[d],g=!0,d+=e);return h}function Wa(a,b){var c=u.defaults.column,d=a.aoColumns.length;c=k.extend({},u.models.oColumn,c,{nTh:b?b:z.createElement("th"),sTitle:c.sTitle?c.sTitle:b?b.innerHTML:"",aDataSort:c.aDataSort?c.aDataSort:[d],mData:c.mData?c.mData:d,idx:d});a.aoColumns.push(c);c=a.aoPreSearchCols;c[d]=k.extend({},u.models.oSearch,c[d]);Da(a,d,k(b).data())}function Da(a,b,c){b=a.aoColumns[b];
|
||||||
|
var d=a.oClasses,f=k(b.nTh);if(!b.sWidthOrig){b.sWidthOrig=f.attr("width")||null;var e=(f.attr("style")||"").match(/width:\s*(\d+[pxem%]+)/);e&&(b.sWidthOrig=e[1])}c!==q&&null!==c&&(zb(c),O(u.defaults.column,c,!0),c.mDataProp===q||c.mData||(c.mData=c.mDataProp),c.sType&&(b._sManualType=c.sType),c.className&&!c.sClass&&(c.sClass=c.className),c.sClass&&f.addClass(c.sClass),k.extend(b,c),V(b,c,"sWidth","sWidthOrig"),c.iDataSort!==q&&(b.aDataSort=[c.iDataSort]),V(b,c,"aDataSort"));var g=b.mData,h=ia(g),
|
||||||
|
l=b.mRender?ia(b.mRender):null;c=function(n){return"string"===typeof n&&-1!==n.indexOf("@")};b._bAttrSrc=k.isPlainObject(g)&&(c(g.sort)||c(g.type)||c(g.filter));b._setter=null;b.fnGetData=function(n,m,p){var t=h(n,m,q,p);return l&&m?l(t,m,n,p):t};b.fnSetData=function(n,m,p){return da(g)(n,m,p)};"number"!==typeof g&&(a._rowReadObject=!0);a.oFeatures.bSort||(b.bSortable=!1,f.addClass(d.sSortableNone));a=-1!==k.inArray("asc",b.asSorting);c=-1!==k.inArray("desc",b.asSorting);b.bSortable&&(a||c)?a&&!c?
|
||||||
|
(b.sSortingClass=d.sSortableAsc,b.sSortingClassJUI=d.sSortJUIAscAllowed):!a&&c?(b.sSortingClass=d.sSortableDesc,b.sSortingClassJUI=d.sSortJUIDescAllowed):(b.sSortingClass=d.sSortable,b.sSortingClassJUI=d.sSortJUI):(b.sSortingClass=d.sSortableNone,b.sSortingClassJUI="")}function ra(a){if(!1!==a.oFeatures.bAutoWidth){var b=a.aoColumns;Xa(a);for(var c=0,d=b.length;c<d;c++)b[c].nTh.style.width=b[c].sWidth}b=a.oScroll;""===b.sY&&""===b.sX||Ea(a);I(a,null,"column-sizing",[a])}function sa(a,b){a=Fa(a,"bVisible");
|
||||||
|
return"number"===typeof a[b]?a[b]:null}function ta(a,b){a=Fa(a,"bVisible");b=k.inArray(b,a);return-1!==b?b:null}function na(a){var b=0;k.each(a.aoColumns,function(c,d){d.bVisible&&"none"!==k(d.nTh).css("display")&&b++});return b}function Fa(a,b){var c=[];k.map(a.aoColumns,function(d,f){d[b]&&c.push(f)});return c}function Ya(a){var b=a.aoColumns,c=a.aoData,d=u.ext.type.detect,f,e,g;var h=0;for(f=b.length;h<f;h++){var l=b[h];var n=[];if(!l.sType&&l._sManualType)l.sType=l._sManualType;else if(!l.sType){var m=
|
||||||
|
0;for(e=d.length;m<e;m++){var p=0;for(g=c.length;p<g;p++){n[p]===q&&(n[p]=S(a,p,h,"type"));var t=d[m](n[p],a);if(!t&&m!==d.length-1)break;if("html"===t)break}if(t){l.sType=t;break}}l.sType||(l.sType="string")}}}function Cb(a,b,c,d){var f,e,g,h=a.aoColumns;if(b)for(f=b.length-1;0<=f;f--){var l=b[f];var n=l.targets!==q?l.targets:l.aTargets;Array.isArray(n)||(n=[n]);var m=0;for(e=n.length;m<e;m++)if("number"===typeof n[m]&&0<=n[m]){for(;h.length<=n[m];)Wa(a);d(n[m],l)}else if("number"===typeof n[m]&&
|
||||||
|
0>n[m])d(h.length+n[m],l);else if("string"===typeof n[m]){var p=0;for(g=h.length;p<g;p++)("_all"==n[m]||k(h[p].nTh).hasClass(n[m]))&&d(p,l)}}if(c)for(f=0,a=c.length;f<a;f++)d(f,c[f])}function ea(a,b,c,d){var f=a.aoData.length,e=k.extend(!0,{},u.models.oRow,{src:c?"dom":"data",idx:f});e._aData=b;a.aoData.push(e);for(var g=a.aoColumns,h=0,l=g.length;h<l;h++)g[h].sType=null;a.aiDisplayMaster.push(f);b=a.rowIdFn(b);b!==q&&(a.aIds[b]=e);!c&&a.oFeatures.bDeferRender||Za(a,f,c,d);return f}function Ga(a,
|
||||||
|
b){var c;b instanceof k||(b=k(b));return b.map(function(d,f){c=$a(a,f);return ea(a,c.data,f,c.cells)})}function S(a,b,c,d){var f=a.iDraw,e=a.aoColumns[c],g=a.aoData[b]._aData,h=e.sDefaultContent,l=e.fnGetData(g,d,{settings:a,row:b,col:c});if(l===q)return a.iDrawError!=f&&null===h&&(aa(a,0,"Requested unknown parameter "+("function"==typeof e.mData?"{function}":"'"+e.mData+"'")+" for row "+b+", column "+c,4),a.iDrawError=f),h;if((l===g||null===l)&&null!==h&&d!==q)l=h;else if("function"===typeof l)return l.call(g);
|
||||||
|
return null===l&&"display"==d?"":l}function Db(a,b,c,d){a.aoColumns[c].fnSetData(a.aoData[b]._aData,d,{settings:a,row:b,col:c})}function ab(a){return k.map(a.match(/(\\.|[^\.])+/g)||[""],function(b){return b.replace(/\\\./g,".")})}function ia(a){if(k.isPlainObject(a)){var b={};k.each(a,function(d,f){f&&(b[d]=ia(f))});return function(d,f,e,g){var h=b[f]||b._;return h!==q?h(d,f,e,g):d}}if(null===a)return function(d){return d};if("function"===typeof a)return function(d,f,e,g){return a(d,f,e,g)};if("string"!==
|
||||||
|
typeof a||-1===a.indexOf(".")&&-1===a.indexOf("[")&&-1===a.indexOf("("))return function(d,f){return d[a]};var c=function(d,f,e){if(""!==e){var g=ab(e);for(var h=0,l=g.length;h<l;h++){e=g[h].match(ua);var n=g[h].match(oa);if(e){g[h]=g[h].replace(ua,"");""!==g[h]&&(d=d[g[h]]);n=[];g.splice(0,h+1);g=g.join(".");if(Array.isArray(d))for(h=0,l=d.length;h<l;h++)n.push(c(d[h],f,g));d=e[0].substring(1,e[0].length-1);d=""===d?n:n.join(d);break}else if(n){g[h]=g[h].replace(oa,"");d=d[g[h]]();continue}if(null===
|
||||||
|
d||d[g[h]]===q)return q;d=d[g[h]]}}return d};return function(d,f){return c(d,f,a)}}function da(a){if(k.isPlainObject(a))return da(a._);if(null===a)return function(){};if("function"===typeof a)return function(c,d,f){a(c,"set",d,f)};if("string"!==typeof a||-1===a.indexOf(".")&&-1===a.indexOf("[")&&-1===a.indexOf("("))return function(c,d){c[a]=d};var b=function(c,d,f){f=ab(f);var e=f[f.length-1];for(var g,h,l=0,n=f.length-1;l<n;l++){if("__proto__"===f[l])throw Error("Cannot set prototype values");g=
|
||||||
|
f[l].match(ua);h=f[l].match(oa);if(g){f[l]=f[l].replace(ua,"");c[f[l]]=[];e=f.slice();e.splice(0,l+1);g=e.join(".");if(Array.isArray(d))for(h=0,n=d.length;h<n;h++)e={},b(e,d[h],g),c[f[l]].push(e);else c[f[l]]=d;return}h&&(f[l]=f[l].replace(oa,""),c=c[f[l]](d));if(null===c[f[l]]||c[f[l]]===q)c[f[l]]={};c=c[f[l]]}if(e.match(oa))c[e.replace(oa,"")](d);else c[e.replace(ua,"")]=d};return function(c,d){return b(c,d,a)}}function bb(a){return T(a.aoData,"_aData")}function Ha(a){a.aoData.length=0;a.aiDisplayMaster.length=
|
||||||
|
0;a.aiDisplay.length=0;a.aIds={}}function Ia(a,b,c){for(var d=-1,f=0,e=a.length;f<e;f++)a[f]==b?d=f:a[f]>b&&a[f]--; -1!=d&&c===q&&a.splice(d,1)}function va(a,b,c,d){var f=a.aoData[b],e,g=function(l,n){for(;l.childNodes.length;)l.removeChild(l.firstChild);l.innerHTML=S(a,b,n,"display")};if("dom"!==c&&(c&&"auto"!==c||"dom"!==f.src)){var h=f.anCells;if(h)if(d!==q)g(h[d],d);else for(c=0,e=h.length;c<e;c++)g(h[c],c)}else f._aData=$a(a,f,d,d===q?q:f._aData).data;f._aSortData=null;f._aFilterData=null;g=
|
||||||
|
a.aoColumns;if(d!==q)g[d].sType=null;else{c=0;for(e=g.length;c<e;c++)g[c].sType=null;cb(a,f)}}function $a(a,b,c,d){var f=[],e=b.firstChild,g,h=0,l,n=a.aoColumns,m=a._rowReadObject;d=d!==q?d:m?{}:[];var p=function(x,r){if("string"===typeof x){var A=x.indexOf("@");-1!==A&&(A=x.substring(A+1),da(x)(d,r.getAttribute(A)))}},t=function(x){if(c===q||c===h)g=n[h],l=x.innerHTML.trim(),g&&g._bAttrSrc?(da(g.mData._)(d,l),p(g.mData.sort,x),p(g.mData.type,x),p(g.mData.filter,x)):m?(g._setter||(g._setter=da(g.mData)),
|
||||||
|
g._setter(d,l)):d[h]=l;h++};if(e)for(;e;){var v=e.nodeName.toUpperCase();if("TD"==v||"TH"==v)t(e),f.push(e);e=e.nextSibling}else for(f=b.anCells,e=0,v=f.length;e<v;e++)t(f[e]);(b=b.firstChild?b:b.nTr)&&(b=b.getAttribute("id"))&&da(a.rowId)(d,b);return{data:d,cells:f}}function Za(a,b,c,d){var f=a.aoData[b],e=f._aData,g=[],h,l;if(null===f.nTr){var n=c||z.createElement("tr");f.nTr=n;f.anCells=g;n._DT_RowIndex=b;cb(a,f);var m=0;for(h=a.aoColumns.length;m<h;m++){var p=a.aoColumns[m];var t=(l=c?!1:!0)?
|
||||||
|
z.createElement(p.sCellType):d[m];t._DT_CellIndex={row:b,column:m};g.push(t);if(l||!(c&&!p.mRender&&p.mData===m||k.isPlainObject(p.mData)&&p.mData._===m+".display"))t.innerHTML=S(a,b,m,"display");p.sClass&&(t.className+=" "+p.sClass);p.bVisible&&!c?n.appendChild(t):!p.bVisible&&c&&t.parentNode.removeChild(t);p.fnCreatedCell&&p.fnCreatedCell.call(a.oInstance,t,S(a,b,m),e,b,m)}I(a,"aoRowCreatedCallback",null,[n,e,b,g])}f.nTr.setAttribute("role","row")}function cb(a,b){var c=b.nTr,d=b._aData;if(c){if(a=
|
||||||
|
a.rowIdFn(d))c.id=a;d.DT_RowClass&&(a=d.DT_RowClass.split(" "),b.__rowc=b.__rowc?Ja(b.__rowc.concat(a)):a,k(c).removeClass(b.__rowc.join(" ")).addClass(d.DT_RowClass));d.DT_RowAttr&&k(c).attr(d.DT_RowAttr);d.DT_RowData&&k(c).data(d.DT_RowData)}}function Eb(a){var b,c,d=a.nTHead,f=a.nTFoot,e=0===k("th, td",d).length,g=a.oClasses,h=a.aoColumns;e&&(c=k("<tr/>").appendTo(d));var l=0;for(b=h.length;l<b;l++){var n=h[l];var m=k(n.nTh).addClass(n.sClass);e&&m.appendTo(c);a.oFeatures.bSort&&(m.addClass(n.sSortingClass),
|
||||||
|
!1!==n.bSortable&&(m.attr("tabindex",a.iTabIndex).attr("aria-controls",a.sTableId),db(a,n.nTh,l)));n.sTitle!=m[0].innerHTML&&m.html(n.sTitle);eb(a,"header")(a,m,n,g)}e&&wa(a.aoHeader,d);k(d).children("tr").attr("role","row");k(d).children("tr").children("th, td").addClass(g.sHeaderTH);k(f).children("tr").children("th, td").addClass(g.sFooterTH);if(null!==f)for(a=a.aoFooter[0],l=0,b=a.length;l<b;l++)n=h[l],n.nTf=a[l].cell,n.sClass&&k(n.nTf).addClass(n.sClass)}function xa(a,b,c){var d,f,e=[],g=[],h=
|
||||||
|
a.aoColumns.length;if(b){c===q&&(c=!1);var l=0;for(d=b.length;l<d;l++){e[l]=b[l].slice();e[l].nTr=b[l].nTr;for(f=h-1;0<=f;f--)a.aoColumns[f].bVisible||c||e[l].splice(f,1);g.push([])}l=0;for(d=e.length;l<d;l++){if(a=e[l].nTr)for(;f=a.firstChild;)a.removeChild(f);f=0;for(b=e[l].length;f<b;f++){var n=h=1;if(g[l][f]===q){a.appendChild(e[l][f].cell);for(g[l][f]=1;e[l+h]!==q&&e[l][f].cell==e[l+h][f].cell;)g[l+h][f]=1,h++;for(;e[l][f+n]!==q&&e[l][f].cell==e[l][f+n].cell;){for(c=0;c<h;c++)g[l+c][f+n]=1;n++}k(e[l][f].cell).attr("rowspan",
|
||||||
|
h).attr("colspan",n)}}}}}function fa(a){var b=I(a,"aoPreDrawCallback","preDraw",[a]);if(-1!==k.inArray(!1,b))U(a,!1);else{b=[];var c=0,d=a.asStripeClasses,f=d.length,e=a.oLanguage,g=a.iInitDisplayStart,h="ssp"==P(a),l=a.aiDisplay;a.bDrawing=!0;g!==q&&-1!==g&&(a._iDisplayStart=h?g:g>=a.fnRecordsDisplay()?0:g,a.iInitDisplayStart=-1);g=a._iDisplayStart;var n=a.fnDisplayEnd();if(a.bDeferLoading)a.bDeferLoading=!1,a.iDraw++,U(a,!1);else if(!h)a.iDraw++;else if(!a.bDestroying&&!Fb(a))return;if(0!==l.length)for(e=
|
||||||
|
h?a.aoData.length:n,h=h?0:g;h<e;h++){var m=l[h],p=a.aoData[m];null===p.nTr&&Za(a,m);var t=p.nTr;if(0!==f){var v=d[c%f];p._sRowStripe!=v&&(k(t).removeClass(p._sRowStripe).addClass(v),p._sRowStripe=v)}I(a,"aoRowCallback",null,[t,p._aData,c,h,m]);b.push(t);c++}else c=e.sZeroRecords,1==a.iDraw&&"ajax"==P(a)?c=e.sLoadingRecords:e.sEmptyTable&&0===a.fnRecordsTotal()&&(c=e.sEmptyTable),b[0]=k("<tr/>",{"class":f?d[0]:""}).append(k("<td />",{valign:"top",colSpan:na(a),"class":a.oClasses.sRowEmpty}).html(c))[0];
|
||||||
|
I(a,"aoHeaderCallback","header",[k(a.nTHead).children("tr")[0],bb(a),g,n,l]);I(a,"aoFooterCallback","footer",[k(a.nTFoot).children("tr")[0],bb(a),g,n,l]);d=k(a.nTBody);d.children().detach();d.append(k(b));I(a,"aoDrawCallback","draw",[a]);a.bSorted=!1;a.bFiltered=!1;a.bDrawing=!1}}function ja(a,b){var c=a.oFeatures,d=c.bFilter;c.bSort&&Gb(a);d?ya(a,a.oPreviousSearch):a.aiDisplay=a.aiDisplayMaster.slice();!0!==b&&(a._iDisplayStart=0);a._drawHold=b;fa(a);a._drawHold=!1}function Hb(a){var b=a.oClasses,
|
||||||
|
c=k(a.nTable);c=k("<div/>").insertBefore(c);var d=a.oFeatures,f=k("<div/>",{id:a.sTableId+"_wrapper","class":b.sWrapper+(a.nTFoot?"":" "+b.sNoFooter)});a.nHolding=c[0];a.nTableWrapper=f[0];a.nTableReinsertBefore=a.nTable.nextSibling;for(var e=a.sDom.split(""),g,h,l,n,m,p,t=0;t<e.length;t++){g=null;h=e[t];if("<"==h){l=k("<div/>")[0];n=e[t+1];if("'"==n||'"'==n){m="";for(p=2;e[t+p]!=n;)m+=e[t+p],p++;"H"==m?m=b.sJUIHeader:"F"==m&&(m=b.sJUIFooter);-1!=m.indexOf(".")?(n=m.split("."),l.id=n[0].substr(1,
|
||||||
|
n[0].length-1),l.className=n[1]):"#"==m.charAt(0)?l.id=m.substr(1,m.length-1):l.className=m;t+=p}f.append(l);f=k(l)}else if(">"==h)f=f.parent();else if("l"==h&&d.bPaginate&&d.bLengthChange)g=Ib(a);else if("f"==h&&d.bFilter)g=Jb(a);else if("r"==h&&d.bProcessing)g=Kb(a);else if("t"==h)g=Lb(a);else if("i"==h&&d.bInfo)g=Mb(a);else if("p"==h&&d.bPaginate)g=Nb(a);else if(0!==u.ext.feature.length)for(l=u.ext.feature,p=0,n=l.length;p<n;p++)if(h==l[p].cFeature){g=l[p].fnInit(a);break}g&&(l=a.aanFeatures,l[h]||
|
||||||
|
(l[h]=[]),l[h].push(g),f.append(g))}c.replaceWith(f);a.nHolding=null}function wa(a,b){b=k(b).children("tr");var c,d,f;a.splice(0,a.length);var e=0;for(f=b.length;e<f;e++)a.push([]);e=0;for(f=b.length;e<f;e++){var g=b[e];for(c=g.firstChild;c;){if("TD"==c.nodeName.toUpperCase()||"TH"==c.nodeName.toUpperCase()){var h=1*c.getAttribute("colspan");var l=1*c.getAttribute("rowspan");h=h&&0!==h&&1!==h?h:1;l=l&&0!==l&&1!==l?l:1;var n=0;for(d=a[e];d[n];)n++;var m=n;var p=1===h?!0:!1;for(d=0;d<h;d++)for(n=0;n<
|
||||||
|
l;n++)a[e+n][m+d]={cell:c,unique:p},a[e+n].nTr=g}c=c.nextSibling}}}function Ka(a,b,c){var d=[];c||(c=a.aoHeader,b&&(c=[],wa(c,b)));b=0;for(var f=c.length;b<f;b++)for(var e=0,g=c[b].length;e<g;e++)!c[b][e].unique||d[e]&&a.bSortCellsTop||(d[e]=c[b][e].cell);return d}function La(a,b,c){I(a,"aoServerParams","serverParams",[b]);if(b&&Array.isArray(b)){var d={},f=/(.*?)\[\]$/;k.each(b,function(m,p){(m=p.name.match(f))?(m=m[0],d[m]||(d[m]=[]),d[m].push(p.value)):d[p.name]=p.value});b=d}var e=a.ajax,g=a.oInstance,
|
||||||
|
h=function(m){I(a,null,"xhr",[a,m,a.jqXHR]);c(m)};if(k.isPlainObject(e)&&e.data){var l=e.data;var n="function"===typeof l?l(b,a):l;b="function"===typeof l&&n?n:k.extend(!0,b,n);delete e.data}n={data:b,success:function(m){var p=m.error||m.sError;p&&aa(a,0,p);a.json=m;h(m)},dataType:"json",cache:!1,type:a.sServerMethod,error:function(m,p,t){t=I(a,null,"xhr",[a,null,a.jqXHR]);-1===k.inArray(!0,t)&&("parsererror"==p?aa(a,0,"Invalid JSON response",1):4===m.readyState&&aa(a,0,"Ajax error",7));U(a,!1)}};
|
||||||
|
a.oAjaxData=b;I(a,null,"preXhr",[a,b]);a.fnServerData?a.fnServerData.call(g,a.sAjaxSource,k.map(b,function(m,p){return{name:p,value:m}}),h,a):a.sAjaxSource||"string"===typeof e?a.jqXHR=k.ajax(k.extend(n,{url:e||a.sAjaxSource})):"function"===typeof e?a.jqXHR=e.call(g,b,h,a):(a.jqXHR=k.ajax(k.extend(n,e)),e.data=l)}function Fb(a){return a.bAjaxDataGet?(a.iDraw++,U(a,!0),La(a,Ob(a),function(b){Pb(a,b)}),!1):!0}function Ob(a){var b=a.aoColumns,c=b.length,d=a.oFeatures,f=a.oPreviousSearch,e=a.aoPreSearchCols,
|
||||||
|
g=[],h=pa(a);var l=a._iDisplayStart;var n=!1!==d.bPaginate?a._iDisplayLength:-1;var m=function(x,r){g.push({name:x,value:r})};m("sEcho",a.iDraw);m("iColumns",c);m("sColumns",T(b,"sName").join(","));m("iDisplayStart",l);m("iDisplayLength",n);var p={draw:a.iDraw,columns:[],order:[],start:l,length:n,search:{value:f.sSearch,regex:f.bRegex}};for(l=0;l<c;l++){var t=b[l];var v=e[l];n="function"==typeof t.mData?"function":t.mData;p.columns.push({data:n,name:t.sName,searchable:t.bSearchable,orderable:t.bSortable,
|
||||||
|
search:{value:v.sSearch,regex:v.bRegex}});m("mDataProp_"+l,n);d.bFilter&&(m("sSearch_"+l,v.sSearch),m("bRegex_"+l,v.bRegex),m("bSearchable_"+l,t.bSearchable));d.bSort&&m("bSortable_"+l,t.bSortable)}d.bFilter&&(m("sSearch",f.sSearch),m("bRegex",f.bRegex));d.bSort&&(k.each(h,function(x,r){p.order.push({column:r.col,dir:r.dir});m("iSortCol_"+x,r.col);m("sSortDir_"+x,r.dir)}),m("iSortingCols",h.length));b=u.ext.legacy.ajax;return null===b?a.sAjaxSource?g:p:b?g:p}function Pb(a,b){var c=function(g,h){return b[g]!==
|
||||||
|
q?b[g]:b[h]},d=Ma(a,b),f=c("sEcho","draw"),e=c("iTotalRecords","recordsTotal");c=c("iTotalDisplayRecords","recordsFiltered");if(f!==q){if(1*f<a.iDraw)return;a.iDraw=1*f}Ha(a);a._iRecordsTotal=parseInt(e,10);a._iRecordsDisplay=parseInt(c,10);f=0;for(e=d.length;f<e;f++)ea(a,d[f]);a.aiDisplay=a.aiDisplayMaster.slice();a.bAjaxDataGet=!1;fa(a);a._bInitComplete||Na(a,b);a.bAjaxDataGet=!0;U(a,!1)}function Ma(a,b){a=k.isPlainObject(a.ajax)&&a.ajax.dataSrc!==q?a.ajax.dataSrc:a.sAjaxDataProp;return"data"===
|
||||||
|
a?b.aaData||b[a]:""!==a?ia(a)(b):b}function Jb(a){var b=a.oClasses,c=a.sTableId,d=a.oLanguage,f=a.oPreviousSearch,e=a.aanFeatures,g='<input type="search" class="'+b.sFilterInput+'"/>',h=d.sSearch;h=h.match(/_INPUT_/)?h.replace("_INPUT_",g):h+g;b=k("<div/>",{id:e.f?null:c+"_filter","class":b.sFilter}).append(k("<label/>").append(h));var l=function(){var m=this.value?this.value:"";m!=f.sSearch&&(ya(a,{sSearch:m,bRegex:f.bRegex,bSmart:f.bSmart,bCaseInsensitive:f.bCaseInsensitive}),a._iDisplayStart=0,
|
||||||
|
fa(a))};e=null!==a.searchDelay?a.searchDelay:"ssp"===P(a)?400:0;var n=k("input",b).val(f.sSearch).attr("placeholder",d.sSearchPlaceholder).on("keyup.DT search.DT input.DT paste.DT cut.DT",e?fb(l,e):l).on("mouseup",function(m){setTimeout(function(){l.call(n[0])},10)}).on("keypress.DT",function(m){if(13==m.keyCode)return!1}).attr("aria-controls",c);k(a.nTable).on("search.dt.DT",function(m,p){if(a===p)try{n[0]!==z.activeElement&&n.val(f.sSearch)}catch(t){}});return b[0]}function ya(a,b,c){var d=a.oPreviousSearch,
|
||||||
|
f=a.aoPreSearchCols,e=function(h){d.sSearch=h.sSearch;d.bRegex=h.bRegex;d.bSmart=h.bSmart;d.bCaseInsensitive=h.bCaseInsensitive},g=function(h){return h.bEscapeRegex!==q?!h.bEscapeRegex:h.bRegex};Ya(a);if("ssp"!=P(a)){Qb(a,b.sSearch,c,g(b),b.bSmart,b.bCaseInsensitive);e(b);for(b=0;b<f.length;b++)Rb(a,f[b].sSearch,b,g(f[b]),f[b].bSmart,f[b].bCaseInsensitive);Sb(a)}else e(b);a.bFiltered=!0;I(a,null,"search",[a])}function Sb(a){for(var b=u.ext.search,c=a.aiDisplay,d,f,e=0,g=b.length;e<g;e++){for(var h=
|
||||||
|
[],l=0,n=c.length;l<n;l++)f=c[l],d=a.aoData[f],b[e](a,d._aFilterData,f,d._aData,l)&&h.push(f);c.length=0;k.merge(c,h)}}function Rb(a,b,c,d,f,e){if(""!==b){var g=[],h=a.aiDisplay;d=gb(b,d,f,e);for(f=0;f<h.length;f++)b=a.aoData[h[f]]._aFilterData[c],d.test(b)&&g.push(h[f]);a.aiDisplay=g}}function Qb(a,b,c,d,f,e){f=gb(b,d,f,e);var g=a.oPreviousSearch.sSearch,h=a.aiDisplayMaster;e=[];0!==u.ext.search.length&&(c=!0);var l=Tb(a);if(0>=b.length)a.aiDisplay=h.slice();else{if(l||c||d||g.length>b.length||0!==
|
||||||
|
b.indexOf(g)||a.bSorted)a.aiDisplay=h.slice();b=a.aiDisplay;for(c=0;c<b.length;c++)f.test(a.aoData[b[c]]._sFilterRow)&&e.push(b[c]);a.aiDisplay=e}}function gb(a,b,c,d){a=b?a:hb(a);c&&(a="^(?=.*?"+k.map(a.match(/"[^"]+"|[^ ]+/g)||[""],function(f){if('"'===f.charAt(0)){var e=f.match(/^"(.*)"$/);f=e?e[1]:f}return f.replace('"',"")}).join(")(?=.*?")+").*$");return new RegExp(a,d?"i":"")}function Tb(a){var b=a.aoColumns,c,d,f=u.ext.type.search;var e=!1;var g=0;for(c=a.aoData.length;g<c;g++){var h=a.aoData[g];
|
||||||
|
if(!h._aFilterData){var l=[];var n=0;for(d=b.length;n<d;n++){e=b[n];if(e.bSearchable){var m=S(a,g,n,"filter");f[e.sType]&&(m=f[e.sType](m));null===m&&(m="");"string"!==typeof m&&m.toString&&(m=m.toString())}else m="";m.indexOf&&-1!==m.indexOf("&")&&(Oa.innerHTML=m,m=rc?Oa.textContent:Oa.innerText);m.replace&&(m=m.replace(/[\r\n\u2028]/g,""));l.push(m)}h._aFilterData=l;h._sFilterRow=l.join(" ");e=!0}}return e}function Ub(a){return{search:a.sSearch,smart:a.bSmart,regex:a.bRegex,caseInsensitive:a.bCaseInsensitive}}
|
||||||
|
function Vb(a){return{sSearch:a.search,bSmart:a.smart,bRegex:a.regex,bCaseInsensitive:a.caseInsensitive}}function Mb(a){var b=a.sTableId,c=a.aanFeatures.i,d=k("<div/>",{"class":a.oClasses.sInfo,id:c?null:b+"_info"});c||(a.aoDrawCallback.push({fn:Wb,sName:"information"}),d.attr("role","status").attr("aria-live","polite"),k(a.nTable).attr("aria-describedby",b+"_info"));return d[0]}function Wb(a){var b=a.aanFeatures.i;if(0!==b.length){var c=a.oLanguage,d=a._iDisplayStart+1,f=a.fnDisplayEnd(),e=a.fnRecordsTotal(),
|
||||||
|
g=a.fnRecordsDisplay(),h=g?c.sInfo:c.sInfoEmpty;g!==e&&(h+=" "+c.sInfoFiltered);h+=c.sInfoPostFix;h=Xb(a,h);c=c.fnInfoCallback;null!==c&&(h=c.call(a.oInstance,a,d,f,e,g,h));k(b).html(h)}}function Xb(a,b){var c=a.fnFormatNumber,d=a._iDisplayStart+1,f=a._iDisplayLength,e=a.fnRecordsDisplay(),g=-1===f;return b.replace(/_START_/g,c.call(a,d)).replace(/_END_/g,c.call(a,a.fnDisplayEnd())).replace(/_MAX_/g,c.call(a,a.fnRecordsTotal())).replace(/_TOTAL_/g,c.call(a,e)).replace(/_PAGE_/g,c.call(a,g?1:Math.ceil(d/
|
||||||
|
f))).replace(/_PAGES_/g,c.call(a,g?1:Math.ceil(e/f)))}function za(a){var b=a.iInitDisplayStart,c=a.aoColumns;var d=a.oFeatures;var f=a.bDeferLoading;if(a.bInitialised){Hb(a);Eb(a);xa(a,a.aoHeader);xa(a,a.aoFooter);U(a,!0);d.bAutoWidth&&Xa(a);var e=0;for(d=c.length;e<d;e++){var g=c[e];g.sWidth&&(g.nTh.style.width=K(g.sWidth))}I(a,null,"preInit",[a]);ja(a);c=P(a);if("ssp"!=c||f)"ajax"==c?La(a,[],function(h){var l=Ma(a,h);for(e=0;e<l.length;e++)ea(a,l[e]);a.iInitDisplayStart=b;ja(a);U(a,!1);Na(a,h)},
|
||||||
|
a):(U(a,!1),Na(a))}else setTimeout(function(){za(a)},200)}function Na(a,b){a._bInitComplete=!0;(b||a.oInit.aaData)&&ra(a);I(a,null,"plugin-init",[a,b]);I(a,"aoInitComplete","init",[a,b])}function ib(a,b){b=parseInt(b,10);a._iDisplayLength=b;jb(a);I(a,null,"length",[a,b])}function Ib(a){var b=a.oClasses,c=a.sTableId,d=a.aLengthMenu,f=Array.isArray(d[0]),e=f?d[0]:d;d=f?d[1]:d;f=k("<select/>",{name:c+"_length","aria-controls":c,"class":b.sLengthSelect});for(var g=0,h=e.length;g<h;g++)f[0][g]=new Option("number"===
|
||||||
|
typeof d[g]?a.fnFormatNumber(d[g]):d[g],e[g]);var l=k("<div><label/></div>").addClass(b.sLength);a.aanFeatures.l||(l[0].id=c+"_length");l.children().append(a.oLanguage.sLengthMenu.replace("_MENU_",f[0].outerHTML));k("select",l).val(a._iDisplayLength).on("change.DT",function(n){ib(a,k(this).val());fa(a)});k(a.nTable).on("length.dt.DT",function(n,m,p){a===m&&k("select",l).val(p)});return l[0]}function Nb(a){var b=a.sPaginationType,c=u.ext.pager[b],d="function"===typeof c,f=function(g){fa(g)};b=k("<div/>").addClass(a.oClasses.sPaging+
|
||||||
|
b)[0];var e=a.aanFeatures;d||c.fnInit(a,b,f);e.p||(b.id=a.sTableId+"_paginate",a.aoDrawCallback.push({fn:function(g){if(d){var h=g._iDisplayStart,l=g._iDisplayLength,n=g.fnRecordsDisplay(),m=-1===l;h=m?0:Math.ceil(h/l);l=m?1:Math.ceil(n/l);n=c(h,l);var p;m=0;for(p=e.p.length;m<p;m++)eb(g,"pageButton")(g,e.p[m],m,n,h,l)}else c.fnUpdate(g,f)},sName:"pagination"}));return b}function kb(a,b,c){var d=a._iDisplayStart,f=a._iDisplayLength,e=a.fnRecordsDisplay();0===e||-1===f?d=0:"number"===typeof b?(d=b*
|
||||||
|
f,d>e&&(d=0)):"first"==b?d=0:"previous"==b?(d=0<=f?d-f:0,0>d&&(d=0)):"next"==b?d+f<e&&(d+=f):"last"==b?d=Math.floor((e-1)/f)*f:aa(a,0,"Unknown paging action: "+b,5);b=a._iDisplayStart!==d;a._iDisplayStart=d;b&&(I(a,null,"page",[a]),c&&fa(a));return b}function Kb(a){return k("<div/>",{id:a.aanFeatures.r?null:a.sTableId+"_processing","class":a.oClasses.sProcessing}).html(a.oLanguage.sProcessing).insertBefore(a.nTable)[0]}function U(a,b){a.oFeatures.bProcessing&&k(a.aanFeatures.r).css("display",b?"block":
|
||||||
|
"none");I(a,null,"processing",[a,b])}function Lb(a){var b=k(a.nTable);b.attr("role","grid");var c=a.oScroll;if(""===c.sX&&""===c.sY)return a.nTable;var d=c.sX,f=c.sY,e=a.oClasses,g=b.children("caption"),h=g.length?g[0]._captionSide:null,l=k(b[0].cloneNode(!1)),n=k(b[0].cloneNode(!1)),m=b.children("tfoot");m.length||(m=null);l=k("<div/>",{"class":e.sScrollWrapper}).append(k("<div/>",{"class":e.sScrollHead}).css({overflow:"hidden",position:"relative",border:0,width:d?d?K(d):null:"100%"}).append(k("<div/>",
|
||||||
|
{"class":e.sScrollHeadInner}).css({"box-sizing":"content-box",width:c.sXInner||"100%"}).append(l.removeAttr("id").css("margin-left",0).append("top"===h?g:null).append(b.children("thead"))))).append(k("<div/>",{"class":e.sScrollBody}).css({position:"relative",overflow:"auto",width:d?K(d):null}).append(b));m&&l.append(k("<div/>",{"class":e.sScrollFoot}).css({overflow:"hidden",border:0,width:d?d?K(d):null:"100%"}).append(k("<div/>",{"class":e.sScrollFootInner}).append(n.removeAttr("id").css("margin-left",
|
||||||
|
0).append("bottom"===h?g:null).append(b.children("tfoot")))));b=l.children();var p=b[0];e=b[1];var t=m?b[2]:null;if(d)k(e).on("scroll.DT",function(v){v=this.scrollLeft;p.scrollLeft=v;m&&(t.scrollLeft=v)});k(e).css("max-height",f);c.bCollapse||k(e).css("height",f);a.nScrollHead=p;a.nScrollBody=e;a.nScrollFoot=t;a.aoDrawCallback.push({fn:Ea,sName:"scrolling"});return l[0]}function Ea(a){var b=a.oScroll,c=b.sX,d=b.sXInner,f=b.sY;b=b.iBarWidth;var e=k(a.nScrollHead),g=e[0].style,h=e.children("div"),l=
|
||||||
|
h[0].style,n=h.children("table");h=a.nScrollBody;var m=k(h),p=h.style,t=k(a.nScrollFoot).children("div"),v=t.children("table"),x=k(a.nTHead),r=k(a.nTable),A=r[0],E=A.style,H=a.nTFoot?k(a.nTFoot):null,W=a.oBrowser,M=W.bScrollOversize,C=T(a.aoColumns,"nTh"),B=[],ba=[],X=[],lb=[],Aa,Yb=function(F){F=F.style;F.paddingTop="0";F.paddingBottom="0";F.borderTopWidth="0";F.borderBottomWidth="0";F.height=0};var ha=h.scrollHeight>h.clientHeight;if(a.scrollBarVis!==ha&&a.scrollBarVis!==q)a.scrollBarVis=ha,ra(a);
|
||||||
|
else{a.scrollBarVis=ha;r.children("thead, tfoot").remove();if(H){var ka=H.clone().prependTo(r);var la=H.find("tr");ka=ka.find("tr")}var mb=x.clone().prependTo(r);x=x.find("tr");ha=mb.find("tr");mb.find("th, td").removeAttr("tabindex");c||(p.width="100%",e[0].style.width="100%");k.each(Ka(a,mb),function(F,Y){Aa=sa(a,F);Y.style.width=a.aoColumns[Aa].sWidth});H&&Z(function(F){F.style.width=""},ka);e=r.outerWidth();""===c?(E.width="100%",M&&(r.find("tbody").height()>h.offsetHeight||"scroll"==m.css("overflow-y"))&&
|
||||||
|
(E.width=K(r.outerWidth()-b)),e=r.outerWidth()):""!==d&&(E.width=K(d),e=r.outerWidth());Z(Yb,ha);Z(function(F){X.push(F.innerHTML);B.push(K(k(F).css("width")))},ha);Z(function(F,Y){-1!==k.inArray(F,C)&&(F.style.width=B[Y])},x);k(ha).height(0);H&&(Z(Yb,ka),Z(function(F){lb.push(F.innerHTML);ba.push(K(k(F).css("width")))},ka),Z(function(F,Y){F.style.width=ba[Y]},la),k(ka).height(0));Z(function(F,Y){F.innerHTML='<div class="dataTables_sizing">'+X[Y]+"</div>";F.childNodes[0].style.height="0";F.childNodes[0].style.overflow=
|
||||||
|
"hidden";F.style.width=B[Y]},ha);H&&Z(function(F,Y){F.innerHTML='<div class="dataTables_sizing">'+lb[Y]+"</div>";F.childNodes[0].style.height="0";F.childNodes[0].style.overflow="hidden";F.style.width=ba[Y]},ka);r.outerWidth()<e?(la=h.scrollHeight>h.offsetHeight||"scroll"==m.css("overflow-y")?e+b:e,M&&(h.scrollHeight>h.offsetHeight||"scroll"==m.css("overflow-y"))&&(E.width=K(la-b)),""!==c&&""===d||aa(a,1,"Possible column misalignment",6)):la="100%";p.width=K(la);g.width=K(la);H&&(a.nScrollFoot.style.width=
|
||||||
|
K(la));!f&&M&&(p.height=K(A.offsetHeight+b));c=r.outerWidth();n[0].style.width=K(c);l.width=K(c);d=r.height()>h.clientHeight||"scroll"==m.css("overflow-y");f="padding"+(W.bScrollbarLeft?"Left":"Right");l[f]=d?b+"px":"0px";H&&(v[0].style.width=K(c),t[0].style.width=K(c),t[0].style[f]=d?b+"px":"0px");r.children("colgroup").insertBefore(r.children("thead"));m.trigger("scroll");!a.bSorted&&!a.bFiltered||a._drawHold||(h.scrollTop=0)}}function Z(a,b,c){for(var d=0,f=0,e=b.length,g,h;f<e;){g=b[f].firstChild;
|
||||||
|
for(h=c?c[f].firstChild:null;g;)1===g.nodeType&&(c?a(g,h,d):a(g,d),d++),g=g.nextSibling,h=c?h.nextSibling:null;f++}}function Xa(a){var b=a.nTable,c=a.aoColumns,d=a.oScroll,f=d.sY,e=d.sX,g=d.sXInner,h=c.length,l=Fa(a,"bVisible"),n=k("th",a.nTHead),m=b.getAttribute("width"),p=b.parentNode,t=!1,v,x=a.oBrowser;d=x.bScrollOversize;(v=b.style.width)&&-1!==v.indexOf("%")&&(m=v);for(v=0;v<l.length;v++){var r=c[l[v]];null!==r.sWidth&&(r.sWidth=Zb(r.sWidthOrig,p),t=!0)}if(d||!t&&!e&&!f&&h==na(a)&&h==n.length)for(v=
|
||||||
|
0;v<h;v++)l=sa(a,v),null!==l&&(c[l].sWidth=K(n.eq(v).width()));else{h=k(b).clone().css("visibility","hidden").removeAttr("id");h.find("tbody tr").remove();var A=k("<tr/>").appendTo(h.find("tbody"));h.find("thead, tfoot").remove();h.append(k(a.nTHead).clone()).append(k(a.nTFoot).clone());h.find("tfoot th, tfoot td").css("width","");n=Ka(a,h.find("thead")[0]);for(v=0;v<l.length;v++)r=c[l[v]],n[v].style.width=null!==r.sWidthOrig&&""!==r.sWidthOrig?K(r.sWidthOrig):"",r.sWidthOrig&&e&&k(n[v]).append(k("<div/>").css({width:r.sWidthOrig,
|
||||||
|
margin:0,padding:0,border:0,height:1}));if(a.aoData.length)for(v=0;v<l.length;v++)t=l[v],r=c[t],k($b(a,t)).clone(!1).append(r.sContentPadding).appendTo(A);k("[name]",h).removeAttr("name");r=k("<div/>").css(e||f?{position:"absolute",top:0,left:0,height:1,right:0,overflow:"hidden"}:{}).append(h).appendTo(p);e&&g?h.width(g):e?(h.css("width","auto"),h.removeAttr("width"),h.width()<p.clientWidth&&m&&h.width(p.clientWidth)):f?h.width(p.clientWidth):m&&h.width(m);for(v=f=0;v<l.length;v++)p=k(n[v]),g=p.outerWidth()-
|
||||||
|
p.width(),p=x.bBounding?Math.ceil(n[v].getBoundingClientRect().width):p.outerWidth(),f+=p,c[l[v]].sWidth=K(p-g);b.style.width=K(f);r.remove()}m&&(b.style.width=K(m));!m&&!e||a._reszEvt||(b=function(){k(y).on("resize.DT-"+a.sInstance,fb(function(){ra(a)}))},d?setTimeout(b,1E3):b(),a._reszEvt=!0)}function Zb(a,b){if(!a)return 0;a=k("<div/>").css("width",K(a)).appendTo(b||z.body);b=a[0].offsetWidth;a.remove();return b}function $b(a,b){var c=ac(a,b);if(0>c)return null;var d=a.aoData[c];return d.nTr?d.anCells[b]:
|
||||||
|
k("<td/>").html(S(a,c,b,"display"))[0]}function ac(a,b){for(var c,d=-1,f=-1,e=0,g=a.aoData.length;e<g;e++)c=S(a,e,b,"display")+"",c=c.replace(sc,""),c=c.replace(/ /g," "),c.length>d&&(d=c.length,f=e);return f}function K(a){return null===a?"0px":"number"==typeof a?0>a?"0px":a+"px":a.match(/\d$/)?a+"px":a}function pa(a){var b=[],c=a.aoColumns;var d=a.aaSortingFixed;var f=k.isPlainObject(d);var e=[];var g=function(m){m.length&&!Array.isArray(m[0])?e.push(m):k.merge(e,m)};Array.isArray(d)&&g(d);
|
||||||
|
f&&d.pre&&g(d.pre);g(a.aaSorting);f&&d.post&&g(d.post);for(a=0;a<e.length;a++){var h=e[a][0];g=c[h].aDataSort;d=0;for(f=g.length;d<f;d++){var l=g[d];var n=c[l].sType||"string";e[a]._idx===q&&(e[a]._idx=k.inArray(e[a][1],c[l].asSorting));b.push({src:h,col:l,dir:e[a][1],index:e[a]._idx,type:n,formatter:u.ext.type.order[n+"-pre"]})}}return b}function Gb(a){var b,c=[],d=u.ext.type.order,f=a.aoData,e=0,g=a.aiDisplayMaster;Ya(a);var h=pa(a);var l=0;for(b=h.length;l<b;l++){var n=h[l];n.formatter&&e++;bc(a,
|
||||||
|
n.col)}if("ssp"!=P(a)&&0!==h.length){l=0;for(b=g.length;l<b;l++)c[g[l]]=l;e===h.length?g.sort(function(m,p){var t,v=h.length,x=f[m]._aSortData,r=f[p]._aSortData;for(t=0;t<v;t++){var A=h[t];var E=x[A.col];var H=r[A.col];E=E<H?-1:E>H?1:0;if(0!==E)return"asc"===A.dir?E:-E}E=c[m];H=c[p];return E<H?-1:E>H?1:0}):g.sort(function(m,p){var t,v=h.length,x=f[m]._aSortData,r=f[p]._aSortData;for(t=0;t<v;t++){var A=h[t];var E=x[A.col];var H=r[A.col];A=d[A.type+"-"+A.dir]||d["string-"+A.dir];E=A(E,H);if(0!==E)return E}E=
|
||||||
|
c[m];H=c[p];return E<H?-1:E>H?1:0})}a.bSorted=!0}function cc(a){var b=a.aoColumns,c=pa(a);a=a.oLanguage.oAria;for(var d=0,f=b.length;d<f;d++){var e=b[d];var g=e.asSorting;var h=e.sTitle.replace(/<.*?>/g,"");var l=e.nTh;l.removeAttribute("aria-sort");e.bSortable&&(0<c.length&&c[0].col==d?(l.setAttribute("aria-sort","asc"==c[0].dir?"ascending":"descending"),e=g[c[0].index+1]||g[0]):e=g[0],h+="asc"===e?a.sSortAscending:a.sSortDescending);l.setAttribute("aria-label",h)}}function nb(a,b,c,d){var f=a.aaSorting,
|
||||||
|
e=a.aoColumns[b].asSorting,g=function(h,l){var n=h._idx;n===q&&(n=k.inArray(h[1],e));return n+1<e.length?n+1:l?null:0};"number"===typeof f[0]&&(f=a.aaSorting=[f]);c&&a.oFeatures.bSortMulti?(c=k.inArray(b,T(f,"0")),-1!==c?(b=g(f[c],!0),null===b&&1===f.length&&(b=0),null===b?f.splice(c,1):(f[c][1]=e[b],f[c]._idx=b)):(f.push([b,e[0],0]),f[f.length-1]._idx=0)):f.length&&f[0][0]==b?(b=g(f[0]),f.length=1,f[0][1]=e[b],f[0]._idx=b):(f.length=0,f.push([b,e[0]]),f[0]._idx=0);ja(a);"function"==typeof d&&d(a)}
|
||||||
|
function db(a,b,c,d){var f=a.aoColumns[c];ob(b,{},function(e){!1!==f.bSortable&&(a.oFeatures.bProcessing?(U(a,!0),setTimeout(function(){nb(a,c,e.shiftKey,d);"ssp"!==P(a)&&U(a,!1)},0)):nb(a,c,e.shiftKey,d))})}function Pa(a){var b=a.aLastSort,c=a.oClasses.sSortColumn,d=pa(a),f=a.oFeatures,e;if(f.bSort&&f.bSortClasses){f=0;for(e=b.length;f<e;f++){var g=b[f].src;k(T(a.aoData,"anCells",g)).removeClass(c+(2>f?f+1:3))}f=0;for(e=d.length;f<e;f++)g=d[f].src,k(T(a.aoData,"anCells",g)).addClass(c+(2>f?f+1:3))}a.aLastSort=
|
||||||
|
d}function bc(a,b){var c=a.aoColumns[b],d=u.ext.order[c.sSortDataType],f;d&&(f=d.call(a.oInstance,a,b,ta(a,b)));for(var e,g=u.ext.type.order[c.sType+"-pre"],h=0,l=a.aoData.length;h<l;h++)if(c=a.aoData[h],c._aSortData||(c._aSortData=[]),!c._aSortData[b]||d)e=d?f[h]:S(a,h,b,"sort"),c._aSortData[b]=g?g(e):e}function Qa(a){if(a.oFeatures.bStateSave&&!a.bDestroying){var b={time:+new Date,start:a._iDisplayStart,length:a._iDisplayLength,order:k.extend(!0,[],a.aaSorting),search:Ub(a.oPreviousSearch),columns:k.map(a.aoColumns,
|
||||||
|
function(c,d){return{visible:c.bVisible,search:Ub(a.aoPreSearchCols[d])}})};I(a,"aoStateSaveParams","stateSaveParams",[a,b]);a.oSavedState=b;a.fnStateSaveCallback.call(a.oInstance,a,b)}}function dc(a,b,c){var d,f,e=a.aoColumns;b=function(h){if(h&&h.time){var l=I(a,"aoStateLoadParams","stateLoadParams",[a,h]);if(-1===k.inArray(!1,l)&&(l=a.iStateDuration,!(0<l&&h.time<+new Date-1E3*l||h.columns&&e.length!==h.columns.length))){a.oLoadedState=k.extend(!0,{},h);h.start!==q&&(a._iDisplayStart=h.start,a.iInitDisplayStart=
|
||||||
|
h.start);h.length!==q&&(a._iDisplayLength=h.length);h.order!==q&&(a.aaSorting=[],k.each(h.order,function(n,m){a.aaSorting.push(m[0]>=e.length?[0,m[1]]:m)}));h.search!==q&&k.extend(a.oPreviousSearch,Vb(h.search));if(h.columns)for(d=0,f=h.columns.length;d<f;d++)l=h.columns[d],l.visible!==q&&(e[d].bVisible=l.visible),l.search!==q&&k.extend(a.aoPreSearchCols[d],Vb(l.search));I(a,"aoStateLoaded","stateLoaded",[a,h])}}c()};if(a.oFeatures.bStateSave){var g=a.fnStateLoadCallback.call(a.oInstance,a,b);g!==
|
||||||
|
q&&b(g)}else c()}function Ra(a){var b=u.settings;a=k.inArray(a,T(b,"nTable"));return-1!==a?b[a]:null}function aa(a,b,c,d){c="DataTables warning: "+(a?"table id="+a.sTableId+" - ":"")+c;d&&(c+=". For more information about this error, please see http://datatables.net/tn/"+d);if(b)y.console&&console.log&&console.log(c);else if(b=u.ext,b=b.sErrMode||b.errMode,a&&I(a,null,"error",[a,d,c]),"alert"==b)alert(c);else{if("throw"==b)throw Error(c);"function"==typeof b&&b(a,d,c)}}function V(a,b,c,d){Array.isArray(c)?
|
||||||
|
k.each(c,function(f,e){Array.isArray(e)?V(a,b,e[0],e[1]):V(a,b,e)}):(d===q&&(d=c),b[c]!==q&&(a[d]=b[c]))}function pb(a,b,c){var d;for(d in b)if(b.hasOwnProperty(d)){var f=b[d];k.isPlainObject(f)?(k.isPlainObject(a[d])||(a[d]={}),k.extend(!0,a[d],f)):c&&"data"!==d&&"aaData"!==d&&Array.isArray(f)?a[d]=f.slice():a[d]=f}return a}function ob(a,b,c){k(a).on("click.DT",b,function(d){k(a).trigger("blur");c(d)}).on("keypress.DT",b,function(d){13===d.which&&(d.preventDefault(),c(d))}).on("selectstart.DT",function(){return!1})}
|
||||||
|
function Q(a,b,c,d){c&&a[b].push({fn:c,sName:d})}function I(a,b,c,d){var f=[];b&&(f=k.map(a[b].slice().reverse(),function(e,g){return e.fn.apply(a.oInstance,d)}));null!==c&&(b=k.Event(c+".dt"),k(a.nTable).trigger(b,d),f.push(b.result));return f}function jb(a){var b=a._iDisplayStart,c=a.fnDisplayEnd(),d=a._iDisplayLength;b>=c&&(b=c-d);b-=b%d;if(-1===d||0>b)b=0;a._iDisplayStart=b}function eb(a,b){a=a.renderer;var c=u.ext.renderer[b];return k.isPlainObject(a)&&a[b]?c[a[b]]||c._:"string"===typeof a?c[a]||
|
||||||
|
c._:c._}function P(a){return a.oFeatures.bServerSide?"ssp":a.ajax||a.sAjaxSource?"ajax":"dom"}function Ba(a,b){var c=ec.numbers_length,d=Math.floor(c/2);b<=c?a=qa(0,b):a<=d?(a=qa(0,c-2),a.push("ellipsis"),a.push(b-1)):(a>=b-1-d?a=qa(b-(c-2),b):(a=qa(a-d+2,a+d-1),a.push("ellipsis"),a.push(b-1)),a.splice(0,0,"ellipsis"),a.splice(0,0,0));a.DT_el="span";return a}function Va(a){k.each({num:function(b){return Sa(b,a)},"num-fmt":function(b){return Sa(b,a,qb)},"html-num":function(b){return Sa(b,a,Ta)},"html-num-fmt":function(b){return Sa(b,
|
||||||
|
a,Ta,qb)}},function(b,c){L.type.order[b+a+"-pre"]=c;b.match(/^html\-/)&&(L.type.search[b+a]=L.type.search.html)})}function fc(a){return function(){var b=[Ra(this[u.ext.iApiIndex])].concat(Array.prototype.slice.call(arguments));return u.ext.internal[a].apply(this,b)}}var u=function(a){this.$=function(e,g){return this.api(!0).$(e,g)};this._=function(e,g){return this.api(!0).rows(e,g).data()};this.api=function(e){return e?new D(Ra(this[L.iApiIndex])):new D(this)};this.fnAddData=function(e,g){var h=this.api(!0);
|
||||||
|
e=Array.isArray(e)&&(Array.isArray(e[0])||k.isPlainObject(e[0]))?h.rows.add(e):h.row.add(e);(g===q||g)&&h.draw();return e.flatten().toArray()};this.fnAdjustColumnSizing=function(e){var g=this.api(!0).columns.adjust(),h=g.settings()[0],l=h.oScroll;e===q||e?g.draw(!1):(""!==l.sX||""!==l.sY)&&Ea(h)};this.fnClearTable=function(e){var g=this.api(!0).clear();(e===q||e)&&g.draw()};this.fnClose=function(e){this.api(!0).row(e).child.hide()};this.fnDeleteRow=function(e,g,h){var l=this.api(!0);e=l.rows(e);var n=
|
||||||
|
e.settings()[0],m=n.aoData[e[0][0]];e.remove();g&&g.call(this,n,m);(h===q||h)&&l.draw();return m};this.fnDestroy=function(e){this.api(!0).destroy(e)};this.fnDraw=function(e){this.api(!0).draw(e)};this.fnFilter=function(e,g,h,l,n,m){n=this.api(!0);null===g||g===q?n.search(e,h,l,m):n.column(g).search(e,h,l,m);n.draw()};this.fnGetData=function(e,g){var h=this.api(!0);if(e!==q){var l=e.nodeName?e.nodeName.toLowerCase():"";return g!==q||"td"==l||"th"==l?h.cell(e,g).data():h.row(e).data()||null}return h.data().toArray()};
|
||||||
|
this.fnGetNodes=function(e){var g=this.api(!0);return e!==q?g.row(e).node():g.rows().nodes().flatten().toArray()};this.fnGetPosition=function(e){var g=this.api(!0),h=e.nodeName.toUpperCase();return"TR"==h?g.row(e).index():"TD"==h||"TH"==h?(e=g.cell(e).index(),[e.row,e.columnVisible,e.column]):null};this.fnIsOpen=function(e){return this.api(!0).row(e).child.isShown()};this.fnOpen=function(e,g,h){return this.api(!0).row(e).child(g,h).show().child()[0]};this.fnPageChange=function(e,g){e=this.api(!0).page(e);
|
||||||
|
(g===q||g)&&e.draw(!1)};this.fnSetColumnVis=function(e,g,h){e=this.api(!0).column(e).visible(g);(h===q||h)&&e.columns.adjust().draw()};this.fnSettings=function(){return Ra(this[L.iApiIndex])};this.fnSort=function(e){this.api(!0).order(e).draw()};this.fnSortListener=function(e,g,h){this.api(!0).order.listener(e,g,h)};this.fnUpdate=function(e,g,h,l,n){var m=this.api(!0);h===q||null===h?m.row(g).data(e):m.cell(g,h).data(e);(n===q||n)&&m.columns.adjust();(l===q||l)&&m.draw();return 0};this.fnVersionCheck=
|
||||||
|
L.fnVersionCheck;var b=this,c=a===q,d=this.length;c&&(a={});this.oApi=this.internal=L.internal;for(var f in u.ext.internal)f&&(this[f]=fc(f));this.each(function(){var e={},g=1<d?pb(e,a,!0):a,h=0,l;e=this.getAttribute("id");var n=!1,m=u.defaults,p=k(this);if("table"!=this.nodeName.toLowerCase())aa(null,0,"Non-table node initialisation ("+this.nodeName+")",2);else{yb(m);zb(m.column);O(m,m,!0);O(m.column,m.column,!0);O(m,k.extend(g,p.data()),!0);var t=u.settings;h=0;for(l=t.length;h<l;h++){var v=t[h];
|
||||||
|
if(v.nTable==this||v.nTHead&&v.nTHead.parentNode==this||v.nTFoot&&v.nTFoot.parentNode==this){var x=g.bRetrieve!==q?g.bRetrieve:m.bRetrieve;if(c||x)return v.oInstance;if(g.bDestroy!==q?g.bDestroy:m.bDestroy){v.oInstance.fnDestroy();break}else{aa(v,0,"Cannot reinitialise DataTable",3);return}}if(v.sTableId==this.id){t.splice(h,1);break}}if(null===e||""===e)this.id=e="DataTables_Table_"+u.ext._unique++;var r=k.extend(!0,{},u.models.oSettings,{sDestroyWidth:p[0].style.width,sInstance:e,sTableId:e});r.nTable=
|
||||||
|
this;r.oApi=b.internal;r.oInit=g;t.push(r);r.oInstance=1===b.length?b:p.dataTable();yb(g);ma(g.oLanguage);g.aLengthMenu&&!g.iDisplayLength&&(g.iDisplayLength=Array.isArray(g.aLengthMenu[0])?g.aLengthMenu[0][0]:g.aLengthMenu[0]);g=pb(k.extend(!0,{},m),g);V(r.oFeatures,g,"bPaginate bLengthChange bFilter bSort bSortMulti bInfo bProcessing bAutoWidth bSortClasses bServerSide bDeferRender".split(" "));V(r,g,["asStripeClasses","ajax","fnServerData","fnFormatNumber","sServerMethod","aaSorting","aaSortingFixed",
|
||||||
|
"aLengthMenu","sPaginationType","sAjaxSource","sAjaxDataProp","iStateDuration","sDom","bSortCellsTop","iTabIndex","fnStateLoadCallback","fnStateSaveCallback","renderer","searchDelay","rowId",["iCookieDuration","iStateDuration"],["oSearch","oPreviousSearch"],["aoSearchCols","aoPreSearchCols"],["iDisplayLength","_iDisplayLength"]]);V(r.oScroll,g,[["sScrollX","sX"],["sScrollXInner","sXInner"],["sScrollY","sY"],["bScrollCollapse","bCollapse"]]);V(r.oLanguage,g,"fnInfoCallback");Q(r,"aoDrawCallback",g.fnDrawCallback,
|
||||||
|
"user");Q(r,"aoServerParams",g.fnServerParams,"user");Q(r,"aoStateSaveParams",g.fnStateSaveParams,"user");Q(r,"aoStateLoadParams",g.fnStateLoadParams,"user");Q(r,"aoStateLoaded",g.fnStateLoaded,"user");Q(r,"aoRowCallback",g.fnRowCallback,"user");Q(r,"aoRowCreatedCallback",g.fnCreatedRow,"user");Q(r,"aoHeaderCallback",g.fnHeaderCallback,"user");Q(r,"aoFooterCallback",g.fnFooterCallback,"user");Q(r,"aoInitComplete",g.fnInitComplete,"user");Q(r,"aoPreDrawCallback",g.fnPreDrawCallback,"user");r.rowIdFn=
|
||||||
|
ia(g.rowId);Ab(r);var A=r.oClasses;k.extend(A,u.ext.classes,g.oClasses);p.addClass(A.sTable);r.iInitDisplayStart===q&&(r.iInitDisplayStart=g.iDisplayStart,r._iDisplayStart=g.iDisplayStart);null!==g.iDeferLoading&&(r.bDeferLoading=!0,e=Array.isArray(g.iDeferLoading),r._iRecordsDisplay=e?g.iDeferLoading[0]:g.iDeferLoading,r._iRecordsTotal=e?g.iDeferLoading[1]:g.iDeferLoading);var E=r.oLanguage;k.extend(!0,E,g.oLanguage);E.sUrl&&(k.ajax({dataType:"json",url:E.sUrl,success:function(C){ma(C);O(m.oLanguage,
|
||||||
|
C);k.extend(!0,E,C);za(r)},error:function(){za(r)}}),n=!0);null===g.asStripeClasses&&(r.asStripeClasses=[A.sStripeOdd,A.sStripeEven]);e=r.asStripeClasses;var H=p.children("tbody").find("tr").eq(0);-1!==k.inArray(!0,k.map(e,function(C,B){return H.hasClass(C)}))&&(k("tbody tr",this).removeClass(e.join(" ")),r.asDestroyStripes=e.slice());e=[];t=this.getElementsByTagName("thead");0!==t.length&&(wa(r.aoHeader,t[0]),e=Ka(r));if(null===g.aoColumns)for(t=[],h=0,l=e.length;h<l;h++)t.push(null);else t=g.aoColumns;
|
||||||
|
h=0;for(l=t.length;h<l;h++)Wa(r,e?e[h]:null);Cb(r,g.aoColumnDefs,t,function(C,B){Da(r,C,B)});if(H.length){var W=function(C,B){return null!==C.getAttribute("data-"+B)?B:null};k(H[0]).children("th, td").each(function(C,B){var ba=r.aoColumns[C];if(ba.mData===C){var X=W(B,"sort")||W(B,"order");B=W(B,"filter")||W(B,"search");if(null!==X||null!==B)ba.mData={_:C+".display",sort:null!==X?C+".@data-"+X:q,type:null!==X?C+".@data-"+X:q,filter:null!==B?C+".@data-"+B:q},Da(r,C)}})}var M=r.oFeatures;e=function(){if(g.aaSorting===
|
||||||
|
q){var C=r.aaSorting;h=0;for(l=C.length;h<l;h++)C[h][1]=r.aoColumns[h].asSorting[0]}Pa(r);M.bSort&&Q(r,"aoDrawCallback",function(){if(r.bSorted){var ba=pa(r),X={};k.each(ba,function(lb,Aa){X[Aa.src]=Aa.dir});I(r,null,"order",[r,ba,X]);cc(r)}});Q(r,"aoDrawCallback",function(){(r.bSorted||"ssp"===P(r)||M.bDeferRender)&&Pa(r)},"sc");C=p.children("caption").each(function(){this._captionSide=k(this).css("caption-side")});var B=p.children("thead");0===B.length&&(B=k("<thead/>").appendTo(p));r.nTHead=B[0];
|
||||||
|
B=p.children("tbody");0===B.length&&(B=k("<tbody/>").appendTo(p));r.nTBody=B[0];B=p.children("tfoot");0===B.length&&0<C.length&&(""!==r.oScroll.sX||""!==r.oScroll.sY)&&(B=k("<tfoot/>").appendTo(p));0===B.length||0===B.children().length?p.addClass(A.sNoFooter):0<B.length&&(r.nTFoot=B[0],wa(r.aoFooter,r.nTFoot));if(g.aaData)for(h=0;h<g.aaData.length;h++)ea(r,g.aaData[h]);else(r.bDeferLoading||"dom"==P(r))&&Ga(r,k(r.nTBody).children("tr"));r.aiDisplay=r.aiDisplayMaster.slice();r.bInitialised=!0;!1===
|
||||||
|
n&&za(r)};g.bStateSave?(M.bStateSave=!0,Q(r,"aoDrawCallback",Qa,"state_save"),dc(r,g,e)):e()}});b=null;return this},L,w,J,rb={},gc=/[\r\n\u2028]/g,Ta=/<.*?>/g,tc=/^\d{2,4}[\.\/\-]\d{1,2}[\.\/\-]\d{1,2}([T ]{1}\d{1,2}[:\.]\d{2}([\.:]\d{2})?)?$/,uc=/(\/|\.|\*|\+|\?|\||\(|\)|\[|\]|\{|\}|\\|\$|\^|\-)/g,qb=/['\u00A0,$£€¥%\u2009\u202F\u20BD\u20a9\u20BArfkɃΞ]/gi,ca=function(a){return a&&!0!==a&&"-"!==a?!1:!0},hc=function(a){var b=parseInt(a,10);return!isNaN(b)&&isFinite(a)?b:null},ic=function(a,b){rb[b]||
|
||||||
|
(rb[b]=new RegExp(hb(b),"g"));return"string"===typeof a&&"."!==b?a.replace(/\./g,"").replace(rb[b],"."):a},sb=function(a,b,c){var d="string"===typeof a;if(ca(a))return!0;b&&d&&(a=ic(a,b));c&&d&&(a=a.replace(qb,""));return!isNaN(parseFloat(a))&&isFinite(a)},jc=function(a,b,c){return ca(a)?!0:ca(a)||"string"===typeof a?sb(a.replace(Ta,""),b,c)?!0:null:null},T=function(a,b,c){var d=[],f=0,e=a.length;if(c!==q)for(;f<e;f++)a[f]&&a[f][b]&&d.push(a[f][b][c]);else for(;f<e;f++)a[f]&&d.push(a[f][b]);return d},
|
||||||
|
Ca=function(a,b,c,d){var f=[],e=0,g=b.length;if(d!==q)for(;e<g;e++)a[b[e]][c]&&f.push(a[b[e]][c][d]);else for(;e<g;e++)f.push(a[b[e]][c]);return f},qa=function(a,b){var c=[];if(b===q){b=0;var d=a}else d=b,b=a;for(a=b;a<d;a++)c.push(a);return c},kc=function(a){for(var b=[],c=0,d=a.length;c<d;c++)a[c]&&b.push(a[c]);return b},Ja=function(a){a:{if(!(2>a.length)){var b=a.slice().sort();for(var c=b[0],d=1,f=b.length;d<f;d++){if(b[d]===c){b=!1;break a}c=b[d]}}b=!0}if(b)return a.slice();b=[];f=a.length;var e,
|
||||||
|
g=0;d=0;a:for(;d<f;d++){c=a[d];for(e=0;e<g;e++)if(b[e]===c)continue a;b.push(c);g++}return b},lc=function(a,b){if(Array.isArray(b))for(var c=0;c<b.length;c++)lc(a,b[c]);else a.push(b);return a};Array.isArray||(Array.isArray=function(a){return"[object Array]"===Object.prototype.toString.call(a)});String.prototype.trim||(String.prototype.trim=function(){return this.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,"")});u.util={throttle:function(a,b){var c=b!==q?b:200,d,f;return function(){var e=this,g=
|
||||||
|
+new Date,h=arguments;d&&g<d+c?(clearTimeout(f),f=setTimeout(function(){d=q;a.apply(e,h)},c)):(d=g,a.apply(e,h))}},escapeRegex:function(a){return a.replace(uc,"\\$1")}};var R=function(a,b,c){a[b]!==q&&(a[c]=a[b])},ua=/\[.*?\]$/,oa=/\(\)$/,hb=u.util.escapeRegex,Oa=k("<div>")[0],rc=Oa.textContent!==q,sc=/<.*?>/g,fb=u.util.throttle,mc=[],N=Array.prototype,vc=function(a){var b,c=u.settings,d=k.map(c,function(e,g){return e.nTable});if(a){if(a.nTable&&a.oApi)return[a];if(a.nodeName&&"table"===a.nodeName.toLowerCase()){var f=
|
||||||
|
k.inArray(a,d);return-1!==f?[c[f]]:null}if(a&&"function"===typeof a.settings)return a.settings().toArray();"string"===typeof a?b=k(a):a instanceof k&&(b=a)}else return[];if(b)return b.map(function(e){f=k.inArray(this,d);return-1!==f?c[f]:null}).toArray()};var D=function(a,b){if(!(this instanceof D))return new D(a,b);var c=[],d=function(g){(g=vc(g))&&c.push.apply(c,g)};if(Array.isArray(a))for(var f=0,e=a.length;f<e;f++)d(a[f]);else d(a);this.context=Ja(c);b&&k.merge(this,b);this.selector={rows:null,
|
||||||
|
cols:null,opts:null};D.extend(this,this,mc)};u.Api=D;k.extend(D.prototype,{any:function(){return 0!==this.count()},concat:N.concat,context:[],count:function(){return this.flatten().length},each:function(a){for(var b=0,c=this.length;b<c;b++)a.call(this,this[b],b,this);return this},eq:function(a){var b=this.context;return b.length>a?new D(b[a],this[a]):null},filter:function(a){var b=[];if(N.filter)b=N.filter.call(this,a,this);else for(var c=0,d=this.length;c<d;c++)a.call(this,this[c],c,this)&&b.push(this[c]);
|
||||||
|
return new D(this.context,b)},flatten:function(){var a=[];return new D(this.context,a.concat.apply(a,this.toArray()))},join:N.join,indexOf:N.indexOf||function(a,b){b=b||0;for(var c=this.length;b<c;b++)if(this[b]===a)return b;return-1},iterator:function(a,b,c,d){var f=[],e,g,h=this.context,l,n=this.selector;"string"===typeof a&&(d=c,c=b,b=a,a=!1);var m=0;for(e=h.length;m<e;m++){var p=new D(h[m]);if("table"===b){var t=c.call(p,h[m],m);t!==q&&f.push(t)}else if("columns"===b||"rows"===b)t=c.call(p,h[m],
|
||||||
|
this[m],m),t!==q&&f.push(t);else if("column"===b||"column-rows"===b||"row"===b||"cell"===b){var v=this[m];"column-rows"===b&&(l=Ua(h[m],n.opts));var x=0;for(g=v.length;x<g;x++)t=v[x],t="cell"===b?c.call(p,h[m],t.row,t.column,m,x):c.call(p,h[m],t,m,x,l),t!==q&&f.push(t)}}return f.length||d?(a=new D(h,a?f.concat.apply([],f):f),b=a.selector,b.rows=n.rows,b.cols=n.cols,b.opts=n.opts,a):this},lastIndexOf:N.lastIndexOf||function(a,b){return this.indexOf.apply(this.toArray.reverse(),arguments)},length:0,
|
||||||
|
map:function(a){var b=[];if(N.map)b=N.map.call(this,a,this);else for(var c=0,d=this.length;c<d;c++)b.push(a.call(this,this[c],c));return new D(this.context,b)},pluck:function(a){return this.map(function(b){return b[a]})},pop:N.pop,push:N.push,reduce:N.reduce||function(a,b){return Bb(this,a,b,0,this.length,1)},reduceRight:N.reduceRight||function(a,b){return Bb(this,a,b,this.length-1,-1,-1)},reverse:N.reverse,selector:null,shift:N.shift,slice:function(){return new D(this.context,this)},sort:N.sort,
|
||||||
|
splice:N.splice,toArray:function(){return N.slice.call(this)},to$:function(){return k(this)},toJQuery:function(){return k(this)},unique:function(){return new D(this.context,Ja(this))},unshift:N.unshift});D.extend=function(a,b,c){if(c.length&&b&&(b instanceof D||b.__dt_wrapper)){var d,f=function(h,l,n){return function(){var m=l.apply(h,arguments);D.extend(m,m,n.methodExt);return m}};var e=0;for(d=c.length;e<d;e++){var g=c[e];b[g.name]="function"===g.type?f(a,g.val,g):"object"===g.type?{}:g.val;b[g.name].__dt_wrapper=
|
||||||
|
!0;D.extend(a,b[g.name],g.propExt)}}};D.register=w=function(a,b){if(Array.isArray(a))for(var c=0,d=a.length;c<d;c++)D.register(a[c],b);else{d=a.split(".");var f=mc,e;a=0;for(c=d.length;a<c;a++){var g=(e=-1!==d[a].indexOf("()"))?d[a].replace("()",""):d[a];a:{var h=0;for(var l=f.length;h<l;h++)if(f[h].name===g){h=f[h];break a}h=null}h||(h={name:g,val:{},methodExt:[],propExt:[],type:"object"},f.push(h));a===c-1?(h.val=b,h.type="function"===typeof b?"function":k.isPlainObject(b)?"object":"other"):f=e?
|
||||||
|
h.methodExt:h.propExt}}};D.registerPlural=J=function(a,b,c){D.register(a,c);D.register(b,function(){var d=c.apply(this,arguments);return d===this?this:d instanceof D?d.length?Array.isArray(d[0])?new D(d.context,d[0]):d[0]:q:d})};var nc=function(a,b){if(Array.isArray(a))return k.map(a,function(d){return nc(d,b)});if("number"===typeof a)return[b[a]];var c=k.map(b,function(d,f){return d.nTable});return k(c).filter(a).map(function(d){d=k.inArray(this,c);return b[d]}).toArray()};w("tables()",function(a){return a!==
|
||||||
|
q&&null!==a?new D(nc(a,this.context)):this});w("table()",function(a){a=this.tables(a);var b=a.context;return b.length?new D(b[0]):a});J("tables().nodes()","table().node()",function(){return this.iterator("table",function(a){return a.nTable},1)});J("tables().body()","table().body()",function(){return this.iterator("table",function(a){return a.nTBody},1)});J("tables().header()","table().header()",function(){return this.iterator("table",function(a){return a.nTHead},1)});J("tables().footer()","table().footer()",
|
||||||
|
function(){return this.iterator("table",function(a){return a.nTFoot},1)});J("tables().containers()","table().container()",function(){return this.iterator("table",function(a){return a.nTableWrapper},1)});w("draw()",function(a){return this.iterator("table",function(b){"page"===a?fa(b):("string"===typeof a&&(a="full-hold"===a?!1:!0),ja(b,!1===a))})});w("page()",function(a){return a===q?this.page.info().page:this.iterator("table",function(b){kb(b,a)})});w("page.info()",function(a){if(0===this.context.length)return q;
|
||||||
|
a=this.context[0];var b=a._iDisplayStart,c=a.oFeatures.bPaginate?a._iDisplayLength:-1,d=a.fnRecordsDisplay(),f=-1===c;return{page:f?0:Math.floor(b/c),pages:f?1:Math.ceil(d/c),start:b,end:a.fnDisplayEnd(),length:c,recordsTotal:a.fnRecordsTotal(),recordsDisplay:d,serverSide:"ssp"===P(a)}});w("page.len()",function(a){return a===q?0!==this.context.length?this.context[0]._iDisplayLength:q:this.iterator("table",function(b){ib(b,a)})});var oc=function(a,b,c){if(c){var d=new D(a);d.one("draw",function(){c(d.ajax.json())})}if("ssp"==
|
||||||
|
P(a))ja(a,b);else{U(a,!0);var f=a.jqXHR;f&&4!==f.readyState&&f.abort();La(a,[],function(e){Ha(a);e=Ma(a,e);for(var g=0,h=e.length;g<h;g++)ea(a,e[g]);ja(a,b);U(a,!1)})}};w("ajax.json()",function(){var a=this.context;if(0<a.length)return a[0].json});w("ajax.params()",function(){var a=this.context;if(0<a.length)return a[0].oAjaxData});w("ajax.reload()",function(a,b){return this.iterator("table",function(c){oc(c,!1===b,a)})});w("ajax.url()",function(a){var b=this.context;if(a===q){if(0===b.length)return q;
|
||||||
|
b=b[0];return b.ajax?k.isPlainObject(b.ajax)?b.ajax.url:b.ajax:b.sAjaxSource}return this.iterator("table",function(c){k.isPlainObject(c.ajax)?c.ajax.url=a:c.ajax=a})});w("ajax.url().load()",function(a,b){return this.iterator("table",function(c){oc(c,!1===b,a)})});var tb=function(a,b,c,d,f){var e=[],g,h,l;var n=typeof b;b&&"string"!==n&&"function"!==n&&b.length!==q||(b=[b]);n=0;for(h=b.length;n<h;n++){var m=b[n]&&b[n].split&&!b[n].match(/[\[\(:]/)?b[n].split(","):[b[n]];var p=0;for(l=m.length;p<l;p++)(g=
|
||||||
|
c("string"===typeof m[p]?m[p].trim():m[p]))&&g.length&&(e=e.concat(g))}a=L.selector[a];if(a.length)for(n=0,h=a.length;n<h;n++)e=a[n](d,f,e);return Ja(e)},ub=function(a){a||(a={});a.filter&&a.search===q&&(a.search=a.filter);return k.extend({search:"none",order:"current",page:"all"},a)},vb=function(a){for(var b=0,c=a.length;b<c;b++)if(0<a[b].length)return a[0]=a[b],a[0].length=1,a.length=1,a.context=[a.context[b]],a;a.length=0;return a},Ua=function(a,b){var c=[],d=a.aiDisplay;var f=a.aiDisplayMaster;
|
||||||
|
var e=b.search;var g=b.order;b=b.page;if("ssp"==P(a))return"removed"===e?[]:qa(0,f.length);if("current"==b)for(g=a._iDisplayStart,a=a.fnDisplayEnd();g<a;g++)c.push(d[g]);else if("current"==g||"applied"==g)if("none"==e)c=f.slice();else if("applied"==e)c=d.slice();else{if("removed"==e){var h={};g=0;for(a=d.length;g<a;g++)h[d[g]]=null;c=k.map(f,function(l){return h.hasOwnProperty(l)?null:l})}}else if("index"==g||"original"==g)for(g=0,a=a.aoData.length;g<a;g++)"none"==e?c.push(g):(f=k.inArray(g,d),(-1===
|
||||||
|
f&&"removed"==e||0<=f&&"applied"==e)&&c.push(g));return c},wc=function(a,b,c){var d;return tb("row",b,function(f){var e=hc(f),g=a.aoData;if(null!==e&&!c)return[e];d||(d=Ua(a,c));if(null!==e&&-1!==k.inArray(e,d))return[e];if(null===f||f===q||""===f)return d;if("function"===typeof f)return k.map(d,function(l){var n=g[l];return f(l,n._aData,n.nTr)?l:null});if(f.nodeName){e=f._DT_RowIndex;var h=f._DT_CellIndex;if(e!==q)return g[e]&&g[e].nTr===f?[e]:[];if(h)return g[h.row]&&g[h.row].nTr===f.parentNode?
|
||||||
|
[h.row]:[];e=k(f).closest("*[data-dt-row]");return e.length?[e.data("dt-row")]:[]}if("string"===typeof f&&"#"===f.charAt(0)&&(e=a.aIds[f.replace(/^#/,"")],e!==q))return[e.idx];e=kc(Ca(a.aoData,d,"nTr"));return k(e).filter(f).map(function(){return this._DT_RowIndex}).toArray()},a,c)};w("rows()",function(a,b){a===q?a="":k.isPlainObject(a)&&(b=a,a="");b=ub(b);var c=this.iterator("table",function(d){return wc(d,a,b)},1);c.selector.rows=a;c.selector.opts=b;return c});w("rows().nodes()",function(){return this.iterator("row",
|
||||||
|
function(a,b){return a.aoData[b].nTr||q},1)});w("rows().data()",function(){return this.iterator(!0,"rows",function(a,b){return Ca(a.aoData,b,"_aData")},1)});J("rows().cache()","row().cache()",function(a){return this.iterator("row",function(b,c){b=b.aoData[c];return"search"===a?b._aFilterData:b._aSortData},1)});J("rows().invalidate()","row().invalidate()",function(a){return this.iterator("row",function(b,c){va(b,c,a)})});J("rows().indexes()","row().index()",function(){return this.iterator("row",function(a,
|
||||||
|
b){return b},1)});J("rows().ids()","row().id()",function(a){for(var b=[],c=this.context,d=0,f=c.length;d<f;d++)for(var e=0,g=this[d].length;e<g;e++){var h=c[d].rowIdFn(c[d].aoData[this[d][e]]._aData);b.push((!0===a?"#":"")+h)}return new D(c,b)});J("rows().remove()","row().remove()",function(){var a=this;this.iterator("row",function(b,c,d){var f=b.aoData,e=f[c],g,h;f.splice(c,1);var l=0;for(g=f.length;l<g;l++){var n=f[l];var m=n.anCells;null!==n.nTr&&(n.nTr._DT_RowIndex=l);if(null!==m)for(n=0,h=m.length;n<
|
||||||
|
h;n++)m[n]._DT_CellIndex.row=l}Ia(b.aiDisplayMaster,c);Ia(b.aiDisplay,c);Ia(a[d],c,!1);0<b._iRecordsDisplay&&b._iRecordsDisplay--;jb(b);c=b.rowIdFn(e._aData);c!==q&&delete b.aIds[c]});this.iterator("table",function(b){for(var c=0,d=b.aoData.length;c<d;c++)b.aoData[c].idx=c});return this});w("rows.add()",function(a){var b=this.iterator("table",function(d){var f,e=[];var g=0;for(f=a.length;g<f;g++){var h=a[g];h.nodeName&&"TR"===h.nodeName.toUpperCase()?e.push(Ga(d,h)[0]):e.push(ea(d,h))}return e},1),
|
||||||
|
c=this.rows(-1);c.pop();k.merge(c,b);return c});w("row()",function(a,b){return vb(this.rows(a,b))});w("row().data()",function(a){var b=this.context;if(a===q)return b.length&&this.length?b[0].aoData[this[0]]._aData:q;var c=b[0].aoData[this[0]];c._aData=a;Array.isArray(a)&&c.nTr&&c.nTr.id&&da(b[0].rowId)(a,c.nTr.id);va(b[0],this[0],"data");return this});w("row().node()",function(){var a=this.context;return a.length&&this.length?a[0].aoData[this[0]].nTr||null:null});w("row.add()",function(a){a instanceof
|
||||||
|
k&&a.length&&(a=a[0]);var b=this.iterator("table",function(c){return a.nodeName&&"TR"===a.nodeName.toUpperCase()?Ga(c,a)[0]:ea(c,a)});return this.row(b[0])});var xc=function(a,b,c,d){var f=[],e=function(g,h){if(Array.isArray(g)||g instanceof k)for(var l=0,n=g.length;l<n;l++)e(g[l],h);else g.nodeName&&"tr"===g.nodeName.toLowerCase()?f.push(g):(l=k("<tr><td></td></tr>").addClass(h),k("td",l).addClass(h).html(g)[0].colSpan=na(a),f.push(l[0]))};e(c,d);b._details&&b._details.detach();b._details=k(f);b._detailsShow&&
|
||||||
|
b._details.insertAfter(b.nTr)},wb=function(a,b){var c=a.context;c.length&&(a=c[0].aoData[b!==q?b:a[0]])&&a._details&&(a._details.remove(),a._detailsShow=q,a._details=q)},pc=function(a,b){var c=a.context;c.length&&a.length&&(a=c[0].aoData[a[0]],a._details&&((a._detailsShow=b)?a._details.insertAfter(a.nTr):a._details.detach(),yc(c[0])))},yc=function(a){var b=new D(a),c=a.aoData;b.off("draw.dt.DT_details column-visibility.dt.DT_details destroy.dt.DT_details");0<T(c,"_details").length&&(b.on("draw.dt.DT_details",
|
||||||
|
function(d,f){a===f&&b.rows({page:"current"}).eq(0).each(function(e){e=c[e];e._detailsShow&&e._details.insertAfter(e.nTr)})}),b.on("column-visibility.dt.DT_details",function(d,f,e,g){if(a===f)for(f=na(f),e=0,g=c.length;e<g;e++)d=c[e],d._details&&d._details.children("td[colspan]").attr("colspan",f)}),b.on("destroy.dt.DT_details",function(d,f){if(a===f)for(d=0,f=c.length;d<f;d++)c[d]._details&&wb(b,d)}))};w("row().child()",function(a,b){var c=this.context;if(a===q)return c.length&&this.length?c[0].aoData[this[0]]._details:
|
||||||
|
q;!0===a?this.child.show():!1===a?wb(this):c.length&&this.length&&xc(c[0],c[0].aoData[this[0]],a,b);return this});w(["row().child.show()","row().child().show()"],function(a){pc(this,!0);return this});w(["row().child.hide()","row().child().hide()"],function(){pc(this,!1);return this});w(["row().child.remove()","row().child().remove()"],function(){wb(this);return this});w("row().child.isShown()",function(){var a=this.context;return a.length&&this.length?a[0].aoData[this[0]]._detailsShow||!1:!1});var zc=
|
||||||
|
/^([^:]+):(name|visIdx|visible)$/,qc=function(a,b,c,d,f){c=[];d=0;for(var e=f.length;d<e;d++)c.push(S(a,f[d],b));return c},Ac=function(a,b,c){var d=a.aoColumns,f=T(d,"sName"),e=T(d,"nTh");return tb("column",b,function(g){var h=hc(g);if(""===g)return qa(d.length);if(null!==h)return[0<=h?h:d.length+h];if("function"===typeof g){var l=Ua(a,c);return k.map(d,function(p,t){return g(t,qc(a,t,0,0,l),e[t])?t:null})}var n="string"===typeof g?g.match(zc):"";if(n)switch(n[2]){case "visIdx":case "visible":h=parseInt(n[1],
|
||||||
|
10);if(0>h){var m=k.map(d,function(p,t){return p.bVisible?t:null});return[m[m.length+h]]}return[sa(a,h)];case "name":return k.map(f,function(p,t){return p===n[1]?t:null});default:return[]}if(g.nodeName&&g._DT_CellIndex)return[g._DT_CellIndex.column];h=k(e).filter(g).map(function(){return k.inArray(this,e)}).toArray();if(h.length||!g.nodeName)return h;h=k(g).closest("*[data-dt-column]");return h.length?[h.data("dt-column")]:[]},a,c)};w("columns()",function(a,b){a===q?a="":k.isPlainObject(a)&&(b=a,
|
||||||
|
a="");b=ub(b);var c=this.iterator("table",function(d){return Ac(d,a,b)},1);c.selector.cols=a;c.selector.opts=b;return c});J("columns().header()","column().header()",function(a,b){return this.iterator("column",function(c,d){return c.aoColumns[d].nTh},1)});J("columns().footer()","column().footer()",function(a,b){return this.iterator("column",function(c,d){return c.aoColumns[d].nTf},1)});J("columns().data()","column().data()",function(){return this.iterator("column-rows",qc,1)});J("columns().dataSrc()",
|
||||||
|
"column().dataSrc()",function(){return this.iterator("column",function(a,b){return a.aoColumns[b].mData},1)});J("columns().cache()","column().cache()",function(a){return this.iterator("column-rows",function(b,c,d,f,e){return Ca(b.aoData,e,"search"===a?"_aFilterData":"_aSortData",c)},1)});J("columns().nodes()","column().nodes()",function(){return this.iterator("column-rows",function(a,b,c,d,f){return Ca(a.aoData,f,"anCells",b)},1)});J("columns().visible()","column().visible()",function(a,b){var c=
|
||||||
|
this,d=this.iterator("column",function(f,e){if(a===q)return f.aoColumns[e].bVisible;var g=f.aoColumns,h=g[e],l=f.aoData,n;if(a!==q&&h.bVisible!==a){if(a){var m=k.inArray(!0,T(g,"bVisible"),e+1);g=0;for(n=l.length;g<n;g++){var p=l[g].nTr;f=l[g].anCells;p&&p.insertBefore(f[e],f[m]||null)}}else k(T(f.aoData,"anCells",e)).detach();h.bVisible=a}});a!==q&&this.iterator("table",function(f){xa(f,f.aoHeader);xa(f,f.aoFooter);f.aiDisplay.length||k(f.nTBody).find("td[colspan]").attr("colspan",na(f));Qa(f);c.iterator("column",
|
||||||
|
function(e,g){I(e,null,"column-visibility",[e,g,a,b])});(b===q||b)&&c.columns.adjust()});return d});J("columns().indexes()","column().index()",function(a){return this.iterator("column",function(b,c){return"visible"===a?ta(b,c):c},1)});w("columns.adjust()",function(){return this.iterator("table",function(a){ra(a)},1)});w("column.index()",function(a,b){if(0!==this.context.length){var c=this.context[0];if("fromVisible"===a||"toData"===a)return sa(c,b);if("fromData"===a||"toVisible"===a)return ta(c,b)}});
|
||||||
|
w("column()",function(a,b){return vb(this.columns(a,b))});var Bc=function(a,b,c){var d=a.aoData,f=Ua(a,c),e=kc(Ca(d,f,"anCells")),g=k(lc([],e)),h,l=a.aoColumns.length,n,m,p,t,v,x;return tb("cell",b,function(r){var A="function"===typeof r;if(null===r||r===q||A){n=[];m=0;for(p=f.length;m<p;m++)for(h=f[m],t=0;t<l;t++)v={row:h,column:t},A?(x=d[h],r(v,S(a,h,t),x.anCells?x.anCells[t]:null)&&n.push(v)):n.push(v);return n}if(k.isPlainObject(r))return r.column!==q&&r.row!==q&&-1!==k.inArray(r.row,f)?[r]:[];
|
||||||
|
A=g.filter(r).map(function(E,H){return{row:H._DT_CellIndex.row,column:H._DT_CellIndex.column}}).toArray();if(A.length||!r.nodeName)return A;x=k(r).closest("*[data-dt-row]");return x.length?[{row:x.data("dt-row"),column:x.data("dt-column")}]:[]},a,c)};w("cells()",function(a,b,c){k.isPlainObject(a)&&(a.row===q?(c=a,a=null):(c=b,b=null));k.isPlainObject(b)&&(c=b,b=null);if(null===b||b===q)return this.iterator("table",function(m){return Bc(m,a,ub(c))});var d=c?{page:c.page,order:c.order,search:c.search}:
|
||||||
|
{},f=this.columns(b,d),e=this.rows(a,d),g,h,l,n;d=this.iterator("table",function(m,p){m=[];g=0;for(h=e[p].length;g<h;g++)for(l=0,n=f[p].length;l<n;l++)m.push({row:e[p][g],column:f[p][l]});return m},1);d=c&&c.selected?this.cells(d,c):d;k.extend(d.selector,{cols:b,rows:a,opts:c});return d});J("cells().nodes()","cell().node()",function(){return this.iterator("cell",function(a,b,c){return(a=a.aoData[b])&&a.anCells?a.anCells[c]:q},1)});w("cells().data()",function(){return this.iterator("cell",function(a,
|
||||||
|
b,c){return S(a,b,c)},1)});J("cells().cache()","cell().cache()",function(a){a="search"===a?"_aFilterData":"_aSortData";return this.iterator("cell",function(b,c,d){return b.aoData[c][a][d]},1)});J("cells().render()","cell().render()",function(a){return this.iterator("cell",function(b,c,d){return S(b,c,d,a)},1)});J("cells().indexes()","cell().index()",function(){return this.iterator("cell",function(a,b,c){return{row:b,column:c,columnVisible:ta(a,c)}},1)});J("cells().invalidate()","cell().invalidate()",
|
||||||
|
function(a){return this.iterator("cell",function(b,c,d){va(b,c,a,d)})});w("cell()",function(a,b,c){return vb(this.cells(a,b,c))});w("cell().data()",function(a){var b=this.context,c=this[0];if(a===q)return b.length&&c.length?S(b[0],c[0].row,c[0].column):q;Db(b[0],c[0].row,c[0].column,a);va(b[0],c[0].row,"data",c[0].column);return this});w("order()",function(a,b){var c=this.context;if(a===q)return 0!==c.length?c[0].aaSorting:q;"number"===typeof a?a=[[a,b]]:a.length&&!Array.isArray(a[0])&&(a=Array.prototype.slice.call(arguments));
|
||||||
|
return this.iterator("table",function(d){d.aaSorting=a.slice()})});w("order.listener()",function(a,b,c){return this.iterator("table",function(d){db(d,a,b,c)})});w("order.fixed()",function(a){if(!a){var b=this.context;b=b.length?b[0].aaSortingFixed:q;return Array.isArray(b)?{pre:b}:b}return this.iterator("table",function(c){c.aaSortingFixed=k.extend(!0,{},a)})});w(["columns().order()","column().order()"],function(a){var b=this;return this.iterator("table",function(c,d){var f=[];k.each(b[d],function(e,
|
||||||
|
g){f.push([g,a])});c.aaSorting=f})});w("search()",function(a,b,c,d){var f=this.context;return a===q?0!==f.length?f[0].oPreviousSearch.sSearch:q:this.iterator("table",function(e){e.oFeatures.bFilter&&ya(e,k.extend({},e.oPreviousSearch,{sSearch:a+"",bRegex:null===b?!1:b,bSmart:null===c?!0:c,bCaseInsensitive:null===d?!0:d}),1)})});J("columns().search()","column().search()",function(a,b,c,d){return this.iterator("column",function(f,e){var g=f.aoPreSearchCols;if(a===q)return g[e].sSearch;f.oFeatures.bFilter&&
|
||||||
|
(k.extend(g[e],{sSearch:a+"",bRegex:null===b?!1:b,bSmart:null===c?!0:c,bCaseInsensitive:null===d?!0:d}),ya(f,f.oPreviousSearch,1))})});w("state()",function(){return this.context.length?this.context[0].oSavedState:null});w("state.clear()",function(){return this.iterator("table",function(a){a.fnStateSaveCallback.call(a.oInstance,a,{})})});w("state.loaded()",function(){return this.context.length?this.context[0].oLoadedState:null});w("state.save()",function(){return this.iterator("table",function(a){Qa(a)})});
|
||||||
|
u.versionCheck=u.fnVersionCheck=function(a){var b=u.version.split(".");a=a.split(".");for(var c,d,f=0,e=a.length;f<e;f++)if(c=parseInt(b[f],10)||0,d=parseInt(a[f],10)||0,c!==d)return c>d;return!0};u.isDataTable=u.fnIsDataTable=function(a){var b=k(a).get(0),c=!1;if(a instanceof u.Api)return!0;k.each(u.settings,function(d,f){d=f.nScrollHead?k("table",f.nScrollHead)[0]:null;var e=f.nScrollFoot?k("table",f.nScrollFoot)[0]:null;if(f.nTable===b||d===b||e===b)c=!0});return c};u.tables=u.fnTables=function(a){var b=
|
||||||
|
!1;k.isPlainObject(a)&&(b=a.api,a=a.visible);var c=k.map(u.settings,function(d){if(!a||a&&k(d.nTable).is(":visible"))return d.nTable});return b?new D(c):c};u.camelToHungarian=O;w("$()",function(a,b){b=this.rows(b).nodes();b=k(b);return k([].concat(b.filter(a).toArray(),b.find(a).toArray()))});k.each(["on","one","off"],function(a,b){w(b+"()",function(){var c=Array.prototype.slice.call(arguments);c[0]=k.map(c[0].split(/\s/),function(f){return f.match(/\.dt\b/)?f:f+".dt"}).join(" ");var d=k(this.tables().nodes());
|
||||||
|
d[b].apply(d,c);return this})});w("clear()",function(){return this.iterator("table",function(a){Ha(a)})});w("settings()",function(){return new D(this.context,this.context)});w("init()",function(){var a=this.context;return a.length?a[0].oInit:null});w("data()",function(){return this.iterator("table",function(a){return T(a.aoData,"_aData")}).flatten()});w("destroy()",function(a){a=a||!1;return this.iterator("table",function(b){var c=b.nTableWrapper.parentNode,d=b.oClasses,f=b.nTable,e=b.nTBody,g=b.nTHead,
|
||||||
|
h=b.nTFoot,l=k(f);e=k(e);var n=k(b.nTableWrapper),m=k.map(b.aoData,function(t){return t.nTr}),p;b.bDestroying=!0;I(b,"aoDestroyCallback","destroy",[b]);a||(new D(b)).columns().visible(!0);n.off(".DT").find(":not(tbody *)").off(".DT");k(y).off(".DT-"+b.sInstance);f!=g.parentNode&&(l.children("thead").detach(),l.append(g));h&&f!=h.parentNode&&(l.children("tfoot").detach(),l.append(h));b.aaSorting=[];b.aaSortingFixed=[];Pa(b);k(m).removeClass(b.asStripeClasses.join(" "));k("th, td",g).removeClass(d.sSortable+
|
||||||
|
" "+d.sSortableAsc+" "+d.sSortableDesc+" "+d.sSortableNone);e.children().detach();e.append(m);g=a?"remove":"detach";l[g]();n[g]();!a&&c&&(c.insertBefore(f,b.nTableReinsertBefore),l.css("width",b.sDestroyWidth).removeClass(d.sTable),(p=b.asDestroyStripes.length)&&e.children().each(function(t){k(this).addClass(b.asDestroyStripes[t%p])}));c=k.inArray(b,u.settings);-1!==c&&u.settings.splice(c,1)})});k.each(["column","row","cell"],function(a,b){w(b+"s().every()",function(c){var d=this.selector.opts,f=
|
||||||
|
this;return this.iterator(b,function(e,g,h,l,n){c.call(f[b](g,"cell"===b?h:d,"cell"===b?d:q),g,h,l,n)})})});w("i18n()",function(a,b,c){var d=this.context[0];a=ia(a)(d.oLanguage);a===q&&(a=b);c!==q&&k.isPlainObject(a)&&(a=a[c]!==q?a[c]:a._);return a.replace("%d",c)});u.version="1.10.22";u.settings=[];u.models={};u.models.oSearch={bCaseInsensitive:!0,sSearch:"",bRegex:!1,bSmart:!0};u.models.oRow={nTr:null,anCells:null,_aData:[],_aSortData:null,_aFilterData:null,_sFilterRow:null,_sRowStripe:"",src:null,
|
||||||
|
idx:-1};u.models.oColumn={idx:null,aDataSort:null,asSorting:null,bSearchable:null,bSortable:null,bVisible:null,_sManualType:null,_bAttrSrc:!1,fnCreatedCell:null,fnGetData:null,fnSetData:null,mData:null,mRender:null,nTh:null,nTf:null,sClass:null,sContentPadding:null,sDefaultContent:null,sName:null,sSortDataType:"std",sSortingClass:null,sSortingClassJUI:null,sTitle:null,sType:null,sWidth:null,sWidthOrig:null};u.defaults={aaData:null,aaSorting:[[0,"asc"]],aaSortingFixed:[],ajax:null,aLengthMenu:[10,
|
||||||
|
25,50,100],aoColumns:null,aoColumnDefs:null,aoSearchCols:[],asStripeClasses:null,bAutoWidth:!0,bDeferRender:!1,bDestroy:!1,bFilter:!0,bInfo:!0,bLengthChange:!0,bPaginate:!0,bProcessing:!1,bRetrieve:!1,bScrollCollapse:!1,bServerSide:!1,bSort:!0,bSortMulti:!0,bSortCellsTop:!1,bSortClasses:!0,bStateSave:!1,fnCreatedRow:null,fnDrawCallback:null,fnFooterCallback:null,fnFormatNumber:function(a){return a.toString().replace(/\B(?=(\d{3})+(?!\d))/g,this.oLanguage.sThousands)},fnHeaderCallback:null,fnInfoCallback:null,
|
||||||
|
fnInitComplete:null,fnPreDrawCallback:null,fnRowCallback:null,fnServerData:null,fnServerParams:null,fnStateLoadCallback:function(a){try{return JSON.parse((-1===a.iStateDuration?sessionStorage:localStorage).getItem("DataTables_"+a.sInstance+"_"+location.pathname))}catch(b){return{}}},fnStateLoadParams:null,fnStateLoaded:null,fnStateSaveCallback:function(a,b){try{(-1===a.iStateDuration?sessionStorage:localStorage).setItem("DataTables_"+a.sInstance+"_"+location.pathname,JSON.stringify(b))}catch(c){}},
|
||||||
|
fnStateSaveParams:null,iStateDuration:7200,iDeferLoading:null,iDisplayLength:10,iDisplayStart:0,iTabIndex:0,oClasses:{},oLanguage:{oAria:{sSortAscending:": activate to sort column ascending",sSortDescending:": activate to sort column descending"},oPaginate:{sFirst:"First",sLast:"Last",sNext:"Next",sPrevious:"Previous"},sEmptyTable:"No data available in table",sInfo:"Showing _START_ to _END_ of _TOTAL_ entries",sInfoEmpty:"Showing 0 to 0 of 0 entries",sInfoFiltered:"(filtered from _MAX_ total entries)",
|
||||||
|
sInfoPostFix:"",sDecimal:"",sThousands:",",sLengthMenu:"Show _MENU_ entries",sLoadingRecords:"Loading...",sProcessing:"Processing...",sSearch:"Search:",sSearchPlaceholder:"",sUrl:"",sZeroRecords:"No matching records found"},oSearch:k.extend({},u.models.oSearch),sAjaxDataProp:"data",sAjaxSource:null,sDom:"lfrtip",searchDelay:null,sPaginationType:"simple_numbers",sScrollX:"",sScrollXInner:"",sScrollY:"",sServerMethod:"GET",renderer:null,rowId:"DT_RowId"};G(u.defaults);u.defaults.column={aDataSort:null,
|
||||||
|
iDataSort:-1,asSorting:["asc","desc"],bSearchable:!0,bSortable:!0,bVisible:!0,fnCreatedCell:null,mData:null,mRender:null,sCellType:"td",sClass:"",sContentPadding:"",sDefaultContent:null,sName:"",sSortDataType:"std",sTitle:null,sType:null,sWidth:null};G(u.defaults.column);u.models.oSettings={oFeatures:{bAutoWidth:null,bDeferRender:null,bFilter:null,bInfo:null,bLengthChange:null,bPaginate:null,bProcessing:null,bServerSide:null,bSort:null,bSortMulti:null,bSortClasses:null,bStateSave:null},oScroll:{bCollapse:null,
|
||||||
|
iBarWidth:0,sX:null,sXInner:null,sY:null},oLanguage:{fnInfoCallback:null},oBrowser:{bScrollOversize:!1,bScrollbarLeft:!1,bBounding:!1,barWidth:0},ajax:null,aanFeatures:[],aoData:[],aiDisplay:[],aiDisplayMaster:[],aIds:{},aoColumns:[],aoHeader:[],aoFooter:[],oPreviousSearch:{},aoPreSearchCols:[],aaSorting:null,aaSortingFixed:[],asStripeClasses:null,asDestroyStripes:[],sDestroyWidth:0,aoRowCallback:[],aoHeaderCallback:[],aoFooterCallback:[],aoDrawCallback:[],aoRowCreatedCallback:[],aoPreDrawCallback:[],
|
||||||
|
aoInitComplete:[],aoStateSaveParams:[],aoStateLoadParams:[],aoStateLoaded:[],sTableId:"",nTable:null,nTHead:null,nTFoot:null,nTBody:null,nTableWrapper:null,bDeferLoading:!1,bInitialised:!1,aoOpenRows:[],sDom:null,searchDelay:null,sPaginationType:"two_button",iStateDuration:0,aoStateSave:[],aoStateLoad:[],oSavedState:null,oLoadedState:null,sAjaxSource:null,sAjaxDataProp:null,bAjaxDataGet:!0,jqXHR:null,json:q,oAjaxData:q,fnServerData:null,aoServerParams:[],sServerMethod:null,fnFormatNumber:null,aLengthMenu:null,
|
||||||
|
iDraw:0,bDrawing:!1,iDrawError:-1,_iDisplayLength:10,_iDisplayStart:0,_iRecordsTotal:0,_iRecordsDisplay:0,oClasses:{},bFiltered:!1,bSorted:!1,bSortCellsTop:null,oInit:null,aoDestroyCallback:[],fnRecordsTotal:function(){return"ssp"==P(this)?1*this._iRecordsTotal:this.aiDisplayMaster.length},fnRecordsDisplay:function(){return"ssp"==P(this)?1*this._iRecordsDisplay:this.aiDisplay.length},fnDisplayEnd:function(){var a=this._iDisplayLength,b=this._iDisplayStart,c=b+a,d=this.aiDisplay.length,f=this.oFeatures,
|
||||||
|
e=f.bPaginate;return f.bServerSide?!1===e||-1===a?b+d:Math.min(b+a,this._iRecordsDisplay):!e||c>d||-1===a?d:c},oInstance:null,sInstance:null,iTabIndex:0,nScrollHead:null,nScrollFoot:null,aLastSort:[],oPlugins:{},rowIdFn:null,rowId:null};u.ext=L={buttons:{},classes:{},builder:"bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2",errMode:"alert",feature:[],search:[],selector:{cell:[],column:[],row:[]},internal:{},legacy:{ajax:null},pager:{},renderer:{pageButton:{},header:{}},order:{},type:{detect:[],search:{},order:{}},_unique:0,fnVersionCheck:u.fnVersionCheck,
|
||||||
|
iApiIndex:0,oJUIClasses:{},sVersion:u.version};k.extend(L,{afnFiltering:L.search,aTypes:L.type.detect,ofnSearch:L.type.search,oSort:L.type.order,afnSortData:L.order,aoFeatures:L.feature,oApi:L.internal,oStdClasses:L.classes,oPagination:L.pager});k.extend(u.ext.classes,{sTable:"dataTable",sNoFooter:"no-footer",sPageButton:"paginate_button",sPageButtonActive:"current",sPageButtonDisabled:"disabled",sStripeOdd:"odd",sStripeEven:"even",sRowEmpty:"dataTables_empty",sWrapper:"dataTables_wrapper",sFilter:"dataTables_filter",
|
||||||
|
sInfo:"dataTables_info",sPaging:"dataTables_paginate paging_",sLength:"dataTables_length",sProcessing:"dataTables_processing",sSortAsc:"sorting_asc",sSortDesc:"sorting_desc",sSortable:"sorting",sSortableAsc:"sorting_asc_disabled",sSortableDesc:"sorting_desc_disabled",sSortableNone:"sorting_disabled",sSortColumn:"sorting_",sFilterInput:"",sLengthSelect:"",sScrollWrapper:"dataTables_scroll",sScrollHead:"dataTables_scrollHead",sScrollHeadInner:"dataTables_scrollHeadInner",sScrollBody:"dataTables_scrollBody",
|
||||||
|
sScrollFoot:"dataTables_scrollFoot",sScrollFootInner:"dataTables_scrollFootInner",sHeaderTH:"",sFooterTH:"",sSortJUIAsc:"",sSortJUIDesc:"",sSortJUI:"",sSortJUIAscAllowed:"",sSortJUIDescAllowed:"",sSortJUIWrapper:"",sSortIcon:"",sJUIHeader:"",sJUIFooter:""});var ec=u.ext.pager;k.extend(ec,{simple:function(a,b){return["previous","next"]},full:function(a,b){return["first","previous","next","last"]},numbers:function(a,b){return[Ba(a,b)]},simple_numbers:function(a,b){return["previous",Ba(a,b),"next"]},
|
||||||
|
full_numbers:function(a,b){return["first","previous",Ba(a,b),"next","last"]},first_last_numbers:function(a,b){return["first",Ba(a,b),"last"]},_numbers:Ba,numbers_length:7});k.extend(!0,u.ext.renderer,{pageButton:{_:function(a,b,c,d,f,e){var g=a.oClasses,h=a.oLanguage.oPaginate,l=a.oLanguage.oAria.paginate||{},n,m,p=0,t=function(x,r){var A,E=g.sPageButtonDisabled,H=function(B){kb(a,B.data.action,!0)};var W=0;for(A=r.length;W<A;W++){var M=r[W];if(Array.isArray(M)){var C=k("<"+(M.DT_el||"div")+"/>").appendTo(x);
|
||||||
|
t(C,M)}else{n=null;m=M;C=a.iTabIndex;switch(M){case "ellipsis":x.append('<span class="ellipsis">…</span>');break;case "first":n=h.sFirst;0===f&&(C=-1,m+=" "+E);break;case "previous":n=h.sPrevious;0===f&&(C=-1,m+=" "+E);break;case "next":n=h.sNext;if(0===e||f===e-1)C=-1,m+=" "+E;break;case "last":n=h.sLast;if(0===e||f===e-1)C=-1,m+=" "+E;break;default:n=a.fnFormatNumber(M+1),m=f===M?g.sPageButtonActive:""}null!==n&&(C=k("<a>",{"class":g.sPageButton+" "+m,"aria-controls":a.sTableId,"aria-label":l[M],
|
||||||
|
"data-dt-idx":p,tabindex:C,id:0===c&&"string"===typeof M?a.sTableId+"_"+M:null}).html(n).appendTo(x),ob(C,{action:M},H),p++)}}};try{var v=k(b).find(z.activeElement).data("dt-idx")}catch(x){}t(k(b).empty(),d);v!==q&&k(b).find("[data-dt-idx="+v+"]").trigger("focus")}}});k.extend(u.ext.type.detect,[function(a,b){b=b.oLanguage.sDecimal;return sb(a,b)?"num"+b:null},function(a,b){if(a&&!(a instanceof Date)&&!tc.test(a))return null;b=Date.parse(a);return null!==b&&!isNaN(b)||ca(a)?"date":null},function(a,
|
||||||
|
b){b=b.oLanguage.sDecimal;return sb(a,b,!0)?"num-fmt"+b:null},function(a,b){b=b.oLanguage.sDecimal;return jc(a,b)?"html-num"+b:null},function(a,b){b=b.oLanguage.sDecimal;return jc(a,b,!0)?"html-num-fmt"+b:null},function(a,b){return ca(a)||"string"===typeof a&&-1!==a.indexOf("<")?"html":null}]);k.extend(u.ext.type.search,{html:function(a){return ca(a)?a:"string"===typeof a?a.replace(gc," ").replace(Ta,""):""},string:function(a){return ca(a)?a:"string"===typeof a?a.replace(gc," "):a}});var Sa=function(a,
|
||||||
|
b,c,d){if(0!==a&&(!a||"-"===a))return-Infinity;b&&(a=ic(a,b));a.replace&&(c&&(a=a.replace(c,"")),d&&(a=a.replace(d,"")));return 1*a};k.extend(L.type.order,{"date-pre":function(a){a=Date.parse(a);return isNaN(a)?-Infinity:a},"html-pre":function(a){return ca(a)?"":a.replace?a.replace(/<.*?>/g,"").toLowerCase():a+""},"string-pre":function(a){return ca(a)?"":"string"===typeof a?a.toLowerCase():a.toString?a.toString():""},"string-asc":function(a,b){return a<b?-1:a>b?1:0},"string-desc":function(a,b){return a<
|
||||||
|
b?1:a>b?-1:0}});Va("");k.extend(!0,u.ext.renderer,{header:{_:function(a,b,c,d){k(a.nTable).on("order.dt.DT",function(f,e,g,h){a===e&&(f=c.idx,b.removeClass(c.sSortingClass+" "+d.sSortAsc+" "+d.sSortDesc).addClass("asc"==h[f]?d.sSortAsc:"desc"==h[f]?d.sSortDesc:c.sSortingClass))})},jqueryui:function(a,b,c,d){k("<div/>").addClass(d.sSortJUIWrapper).append(b.contents()).append(k("<span/>").addClass(d.sSortIcon+" "+c.sSortingClassJUI)).appendTo(b);k(a.nTable).on("order.dt.DT",function(f,e,g,h){a===e&&
|
||||||
|
(f=c.idx,b.removeClass(d.sSortAsc+" "+d.sSortDesc).addClass("asc"==h[f]?d.sSortAsc:"desc"==h[f]?d.sSortDesc:c.sSortingClass),b.find("span."+d.sSortIcon).removeClass(d.sSortJUIAsc+" "+d.sSortJUIDesc+" "+d.sSortJUI+" "+d.sSortJUIAscAllowed+" "+d.sSortJUIDescAllowed).addClass("asc"==h[f]?d.sSortJUIAsc:"desc"==h[f]?d.sSortJUIDesc:c.sSortingClassJUI))})}}});var xb=function(a){return"string"===typeof a?a.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,"""):a};u.render=
|
||||||
|
{number:function(a,b,c,d,f){return{display:function(e){if("number"!==typeof e&&"string"!==typeof e)return e;var g=0>e?"-":"",h=parseFloat(e);if(isNaN(h))return xb(e);h=h.toFixed(c);e=Math.abs(h);h=parseInt(e,10);e=c?b+(e-h).toFixed(c).substring(2):"";return g+(d||"")+h.toString().replace(/\B(?=(\d{3})+(?!\d))/g,a)+e+(f||"")}}},text:function(){return{display:xb,filter:xb}}};k.extend(u.ext.internal,{_fnExternApiFunc:fc,_fnBuildAjax:La,_fnAjaxUpdate:Fb,_fnAjaxParameters:Ob,_fnAjaxUpdateDraw:Pb,_fnAjaxDataSrc:Ma,
|
||||||
|
_fnAddColumn:Wa,_fnColumnOptions:Da,_fnAdjustColumnSizing:ra,_fnVisibleToColumnIndex:sa,_fnColumnIndexToVisible:ta,_fnVisbleColumns:na,_fnGetColumns:Fa,_fnColumnTypes:Ya,_fnApplyColumnDefs:Cb,_fnHungarianMap:G,_fnCamelToHungarian:O,_fnLanguageCompat:ma,_fnBrowserDetect:Ab,_fnAddData:ea,_fnAddTr:Ga,_fnNodeToDataIndex:function(a,b){return b._DT_RowIndex!==q?b._DT_RowIndex:null},_fnNodeToColumnIndex:function(a,b,c){return k.inArray(c,a.aoData[b].anCells)},_fnGetCellData:S,_fnSetCellData:Db,_fnSplitObjNotation:ab,
|
||||||
|
_fnGetObjectDataFn:ia,_fnSetObjectDataFn:da,_fnGetDataMaster:bb,_fnClearTable:Ha,_fnDeleteIndex:Ia,_fnInvalidate:va,_fnGetRowElements:$a,_fnCreateTr:Za,_fnBuildHead:Eb,_fnDrawHead:xa,_fnDraw:fa,_fnReDraw:ja,_fnAddOptionsHtml:Hb,_fnDetectHeader:wa,_fnGetUniqueThs:Ka,_fnFeatureHtmlFilter:Jb,_fnFilterComplete:ya,_fnFilterCustom:Sb,_fnFilterColumn:Rb,_fnFilter:Qb,_fnFilterCreateSearch:gb,_fnEscapeRegex:hb,_fnFilterData:Tb,_fnFeatureHtmlInfo:Mb,_fnUpdateInfo:Wb,_fnInfoMacros:Xb,_fnInitialise:za,_fnInitComplete:Na,
|
||||||
|
_fnLengthChange:ib,_fnFeatureHtmlLength:Ib,_fnFeatureHtmlPaginate:Nb,_fnPageChange:kb,_fnFeatureHtmlProcessing:Kb,_fnProcessingDisplay:U,_fnFeatureHtmlTable:Lb,_fnScrollDraw:Ea,_fnApplyToChildren:Z,_fnCalculateColumnWidths:Xa,_fnThrottle:fb,_fnConvertToWidth:Zb,_fnGetWidestNode:$b,_fnGetMaxLenString:ac,_fnStringToCss:K,_fnSortFlatten:pa,_fnSort:Gb,_fnSortAria:cc,_fnSortListener:nb,_fnSortAttachListener:db,_fnSortingClasses:Pa,_fnSortData:bc,_fnSaveState:Qa,_fnLoadState:dc,_fnSettingsFromNode:Ra,_fnLog:aa,
|
||||||
|
_fnMap:V,_fnBindAction:ob,_fnCallbackReg:Q,_fnCallbackFire:I,_fnLengthOverflow:jb,_fnRenderer:eb,_fnDataSource:P,_fnRowAttributes:cb,_fnExtend:pb,_fnCalculateEnd:function(){}});k.fn.dataTable=u;u.$=k;k.fn.dataTableSettings=u.settings;k.fn.dataTableExt=u.ext;k.fn.DataTable=function(a){return k(this).dataTable(a).api()};k.each(u,function(a,b){k.fn.DataTable[a]=b});return k.fn.dataTable});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
DataTables Bootstrap 4 integration
|
||||||
|
©2011-2017 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.findInternal=function(a,b,c){a instanceof String&&(a=String(a));for(var e=a.length,d=0;d<e;d++){var f=a[d];if(b.call(c,f,d,a))return{i:d,v:f}}return{i:-1,v:void 0}};$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;$jscomp.ISOLATE_POLYFILLS=!1;
|
||||||
|
$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(a,b,c){if(a==Array.prototype||a==Object.prototype)return a;a[b]=c.value;return a};$jscomp.getGlobal=function(a){a=["object"==typeof globalThis&&globalThis,a,"object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global];for(var b=0;b<a.length;++b){var c=a[b];if(c&&c.Math==Math)return c}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.IS_SYMBOL_NATIVE="function"===typeof Symbol&&"symbol"===typeof Symbol("x");$jscomp.TRUST_ES6_POLYFILLS=!$jscomp.ISOLATE_POLYFILLS||$jscomp.IS_SYMBOL_NATIVE;$jscomp.polyfills={};$jscomp.propertyToPolyfillSymbol={};$jscomp.POLYFILL_PREFIX="$jscp$";var $jscomp$lookupPolyfilledValue=function(a,b){var c=$jscomp.propertyToPolyfillSymbol[b];if(null==c)return a[b];c=a[c];return void 0!==c?c:a[b]};
|
||||||
|
$jscomp.polyfill=function(a,b,c,e){b&&($jscomp.ISOLATE_POLYFILLS?$jscomp.polyfillIsolated(a,b,c,e):$jscomp.polyfillUnisolated(a,b,c,e))};$jscomp.polyfillUnisolated=function(a,b,c,e){c=$jscomp.global;a=a.split(".");for(e=0;e<a.length-1;e++){var d=a[e];if(!(d in c))return;c=c[d]}a=a[a.length-1];e=c[a];b=b(e);b!=e&&null!=b&&$jscomp.defineProperty(c,a,{configurable:!0,writable:!0,value:b})};
|
||||||
|
$jscomp.polyfillIsolated=function(a,b,c,e){var d=a.split(".");a=1===d.length;e=d[0];e=!a&&e in $jscomp.polyfills?$jscomp.polyfills:$jscomp.global;for(var f=0;f<d.length-1;f++){var l=d[f];if(!(l in e))return;e=e[l]}d=d[d.length-1];c=$jscomp.IS_SYMBOL_NATIVE&&"es6"===c?e[d]:null;b=b(c);null!=b&&(a?$jscomp.defineProperty($jscomp.polyfills,d,{configurable:!0,writable:!0,value:b}):b!==c&&($jscomp.propertyToPolyfillSymbol[d]=$jscomp.IS_SYMBOL_NATIVE?$jscomp.global.Symbol(d):$jscomp.POLYFILL_PREFIX+d,d=
|
||||||
|
$jscomp.propertyToPolyfillSymbol[d],$jscomp.defineProperty(e,d,{configurable:!0,writable:!0,value:b})))};$jscomp.polyfill("Array.prototype.find",function(a){return a?a:function(b,c){return $jscomp.findInternal(this,b,c).v}},"es6","es3");
|
||||||
|
(function(a){"function"===typeof define&&define.amd?define(["jquery","datatables.net"],function(b){return a(b,window,document)}):"object"===typeof exports?module.exports=function(b,c){b||(b=window);c&&c.fn.dataTable||(c=require("datatables.net")(b,c).$);return a(c,b,b.document)}:a(jQuery,window,document)})(function(a,b,c,e){var d=a.fn.dataTable;a.extend(!0,d.defaults,{dom:"<'row'<'col-sm-12 col-md-6'l><'col-sm-12 col-md-6'f>><'row'<'col-sm-12'tr>><'row'<'col-sm-12 col-md-5'i><'col-sm-12 col-md-7'p>>",
|
||||||
|
renderer:"bootstrap"});a.extend(d.ext.classes,{sWrapper:"dataTables_wrapper dt-bootstrap4",sFilterInput:"form-control form-control-sm",sLengthSelect:"custom-select custom-select-sm form-control form-control-sm",sProcessing:"dataTables_processing card",sPageButton:"paginate_button page-item"});d.ext.renderer.pageButton.bootstrap=function(f,l,A,B,m,t){var u=new d.Api(f),C=f.oClasses,n=f.oLanguage.oPaginate,D=f.oLanguage.oAria.paginate||{},h,k,v=0,y=function(q,w){var x,E=function(p){p.preventDefault();
|
||||||
|
a(p.currentTarget).hasClass("disabled")||u.page()==p.data.action||u.page(p.data.action).draw("page")};var r=0;for(x=w.length;r<x;r++){var g=w[r];if(Array.isArray(g))y(q,g);else{k=h="";switch(g){case "ellipsis":h="…";k="disabled";break;case "first":h=n.sFirst;k=g+(0<m?"":" disabled");break;case "previous":h=n.sPrevious;k=g+(0<m?"":" disabled");break;case "next":h=n.sNext;k=g+(m<t-1?"":" disabled");break;case "last":h=n.sLast;k=g+(m<t-1?"":" disabled");break;default:h=g+1,k=m===g?"active":""}if(h){var F=
|
||||||
|
a("<li>",{"class":C.sPageButton+" "+k,id:0===A&&"string"===typeof g?f.sTableId+"_"+g:null}).append(a("<a>",{href:"#","aria-controls":f.sTableId,"aria-label":D[g],"data-dt-idx":v,tabindex:f.iTabIndex,"class":"page-link"}).html(h)).appendTo(q);f.oApi._fnBindAction(F,{action:g},E);v++}}}};try{var z=a(l).find(c.activeElement).data("dt-idx")}catch(q){}y(a(l).empty().html('<ul class="pagination"/>').children("ul"),B);z!==e&&a(l).find("[data-dt-idx="+z+"]").trigger("focus")};return d});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Copyright 2009-2020 SpryMedia Ltd.
|
||||||
|
|
||||||
|
This source file is free software, available under the following license:
|
||||||
|
MIT license - http://datatables.net/license/mit
|
||||||
|
|
||||||
|
This source file is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the license files for details.
|
||||||
|
|
||||||
|
For details please refer to: http://www.datatables.net
|
||||||
|
FixedHeader 3.1.7
|
||||||
|
©2009-2020 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.findInternal=function(c,d,f){c instanceof String&&(c=String(c));for(var h=c.length,g=0;g<h;g++){var m=c[g];if(d.call(f,m,g,c))return{i:g,v:m}}return{i:-1,v:void 0}};$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;
|
||||||
|
$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(c,d,f){c!=Array.prototype&&c!=Object.prototype&&(c[d]=f.value)};$jscomp.getGlobal=function(c){c=["object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global,c];for(var d=0;d<c.length;++d){var f=c[d];if(f&&f.Math==Math)return f}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.polyfill=function(c,d,f,h){if(d){f=$jscomp.global;c=c.split(".");for(h=0;h<c.length-1;h++){var g=c[h];g in f||(f[g]={});f=f[g]}c=c[c.length-1];h=f[c];d=d(h);d!=h&&null!=d&&$jscomp.defineProperty(f,c,{configurable:!0,writable:!0,value:d})}};$jscomp.polyfill("Array.prototype.find",function(c){return c?c:function(c,f){return $jscomp.findInternal(this,c,f).v}},"es6","es3");
|
||||||
|
(function(c){"function"===typeof define&&define.amd?define(["jquery","datatables.net"],function(d){return c(d,window,document)}):"object"===typeof exports?module.exports=function(d,f){d||(d=window);f&&f.fn.dataTable||(f=require("datatables.net")(d,f).$);return c(f,d,d.document)}:c(jQuery,window,document)})(function(c,d,f,h){var g=c.fn.dataTable,m=0,l=function(a,b){if(!(this instanceof l))throw"FixedHeader must be initialised with the 'new' keyword.";!0===b&&(b={});a=new g.Api(a);this.c=c.extend(!0,
|
||||||
|
{},l.defaults,b);this.s={dt:a,position:{theadTop:0,tbodyTop:0,tfootTop:0,tfootBottom:0,width:0,left:0,tfootHeight:0,theadHeight:0,windowHeight:c(d).height(),visible:!0},headerMode:null,footerMode:null,autoWidth:a.settings()[0].oFeatures.bAutoWidth,namespace:".dtfc"+m++,scrollLeft:{header:-1,footer:-1},enable:!0};this.dom={floatingHeader:null,thead:c(a.table().header()),tbody:c(a.table().body()),tfoot:c(a.table().footer()),header:{host:null,floating:null,placeholder:null},footer:{host:null,floating:null,
|
||||||
|
placeholder:null}};this.dom.header.host=this.dom.thead.parent();this.dom.footer.host=this.dom.tfoot.parent();a=a.settings()[0];if(a._fixedHeader)throw"FixedHeader already initialised on table "+a.nTable.id;a._fixedHeader=this;this._constructor()};c.extend(l.prototype,{destroy:function(){this.s.dt.off(".dtfc");c(d).off(this.s.namespace);this.c.header&&this._modeChange("in-place","header",!0);this.c.footer&&this.dom.tfoot.length&&this._modeChange("in-place","footer",!0)},enable:function(a,b){this.s.enable=
|
||||||
|
a;if(b||b===h)this._positions(),this._scroll(!0)},enabled:function(){return this.s.enable},headerOffset:function(a){a!==h&&(this.c.headerOffset=a,this.update());return this.c.headerOffset},footerOffset:function(a){a!==h&&(this.c.footerOffset=a,this.update());return this.c.footerOffset},update:function(){var a=this.s.dt.table().node();c(a).is(":visible")?this.enable(!0,!1):this.enable(!1,!1);this._positions();this._scroll(!0)},_constructor:function(){var a=this,b=this.s.dt;c(d).on("scroll"+this.s.namespace,
|
||||||
|
function(){a._scroll()}).on("resize"+this.s.namespace,g.util.throttle(function(){a.s.position.windowHeight=c(d).height();a.update()},50));var k=c(".fh-fixedHeader");!this.c.headerOffset&&k.length&&(this.c.headerOffset=k.outerHeight());k=c(".fh-fixedFooter");!this.c.footerOffset&&k.length&&(this.c.footerOffset=k.outerHeight());b.on("column-reorder.dt.dtfc column-visibility.dt.dtfc draw.dt.dtfc column-sizing.dt.dtfc responsive-display.dt.dtfc",function(){a.update()});b.on("destroy.dtfc",function(){a.destroy()});
|
||||||
|
this._positions();this._scroll()},_clone:function(a,b){var k=this.s.dt,e=this.dom[a],f="header"===a?this.dom.thead:this.dom.tfoot;!b&&e.floating?e.floating.removeClass("fixedHeader-floating fixedHeader-locked"):(e.floating&&(e.placeholder.remove(),this._unsize(a),e.floating.children().detach(),e.floating.remove()),e.floating=c(k.table().node().cloneNode(!1)).css("table-layout","fixed").attr("aria-hidden","true").removeAttr("id").append(f).appendTo("body"),e.placeholder=f.clone(!1),e.placeholder.find("*[id]").removeAttr("id"),
|
||||||
|
e.host.prepend(e.placeholder),this._matchWidths(e.placeholder,e.floating))},_matchWidths:function(a,b){var k=function(b){return c(b,a).map(function(){return c(this).width()}).toArray()},e=function(a,e){c(a,b).each(function(a){c(this).css({width:e[a],minWidth:e[a]})})},f=k("th");k=k("td");e("th",f);e("td",k)},_unsize:function(a){var b=this.dom[a].floating;b&&("footer"===a||"header"===a&&!this.s.autoWidth)?c("th, td",b).css({width:"",minWidth:""}):b&&"header"===a&&c("th, td",b).css("min-width","")},
|
||||||
|
_horizontal:function(a,b){var c=this.dom[a],e=this.s.position,f=this.s.scrollLeft;c.floating&&f[a]!==b&&(c.floating.css("left",e.left-b),f[a]=b)},_modeChange:function(a,b,k){var e=this.dom[b],d=this.s.position,g=function(a){e.floating.attr("style",function(b,c){return(c||"")+"width: "+a+"px !important;"})},h=this.dom["footer"===b?"tfoot":"thead"],l=c.contains(h[0],f.activeElement)?f.activeElement:null;l&&l.blur();"in-place"===a?(e.placeholder&&(e.placeholder.remove(),e.placeholder=null),this._unsize(b),
|
||||||
|
"header"===b?e.host.prepend(h):e.host.append(h),e.floating&&(e.floating.remove(),e.floating=null)):"in"===a?(this._clone(b,k),e.floating.addClass("fixedHeader-floating").css("header"===b?"top":"bottom",this.c[b+"Offset"]).css("left",d.left+"px"),g(d.width),"footer"===b&&e.floating.css("top","")):"below"===a?(this._clone(b,k),e.floating.addClass("fixedHeader-locked").css("top",d.tfootTop-d.theadHeight).css("left",d.left+"px"),g(d.width)):"above"===a&&(this._clone(b,k),e.floating.addClass("fixedHeader-locked").css("top",
|
||||||
|
d.tbodyTop).css("left",d.left+"px"),g(d.width));l&&l!==f.activeElement&&setTimeout(function(){l.focus()},10);this.s.scrollLeft.header=-1;this.s.scrollLeft.footer=-1;this.s[b+"Mode"]=a},_positions:function(){var a=this.s.dt.table(),b=this.s.position,f=this.dom;a=c(a.node());var e=a.children("thead"),d=a.children("tfoot");f=f.tbody;b.visible=a.is(":visible");b.width=a.outerWidth();b.left=a.offset().left;b.theadTop=e.offset().top;b.tbodyTop=f.offset().top;b.tbodyHeight=f.outerHeight();b.theadHeight=
|
||||||
|
b.tbodyTop-b.theadTop;d.length?(b.tfootTop=d.offset().top,b.tfootBottom=b.tfootTop+d.outerHeight(),b.tfootHeight=b.tfootBottom-b.tfootTop):(b.tfootTop=b.tbodyTop+f.outerHeight(),b.tfootBottom=b.tfootTop,b.tfootHeight=b.tfootTop)},_scroll:function(a){var b=c(f).scrollTop(),d=c(f).scrollLeft(),e=this.s.position;if(this.c.header){var g=this.s.enable?!e.visible||b<=e.theadTop-this.c.headerOffset?"in-place":b<=e.tfootTop-e.theadHeight-this.c.headerOffset?"in":"below":"in-place";(a||g!==this.s.headerMode)&&
|
||||||
|
this._modeChange(g,"header",a);this._horizontal("header",d)}this.c.footer&&this.dom.tfoot.length&&(b=this.s.enable?!e.visible||b+e.windowHeight>=e.tfootBottom+this.c.footerOffset?"in-place":e.windowHeight+b>e.tbodyTop+e.tfootHeight+this.c.footerOffset?"in":"above":"in-place",(a||b!==this.s.footerMode)&&this._modeChange(b,"footer",a),this._horizontal("footer",d))}});l.version="3.1.7";l.defaults={header:!0,footer:!1,headerOffset:0,footerOffset:0};c.fn.dataTable.FixedHeader=l;c.fn.DataTable.FixedHeader=
|
||||||
|
l;c(f).on("init.dt.dtfh",function(a,b,d){"dt"===a.namespace&&(a=b.oInit.fixedHeader,d=g.defaults.fixedHeader,!a&&!d||b._fixedHeader||(d=c.extend({},d,a),!1!==a&&new l(b,d)))});g.Api.register("fixedHeader()",function(){});g.Api.register("fixedHeader.adjust()",function(){return this.iterator("table",function(a){(a=a._fixedHeader)&&a.update()})});g.Api.register("fixedHeader.enable()",function(a){return this.iterator("table",function(b){b=b._fixedHeader;a=a!==h?a:!0;b&&a!==b.enabled()&&b.enable(a)})});
|
||||||
|
g.Api.register("fixedHeader.enabled()",function(){if(this.context.length){var a=this.content[0]._fixedHeader;if(a)return a.enabled()}return!1});g.Api.register("fixedHeader.disable()",function(){return this.iterator("table",function(a){(a=a._fixedHeader)&&a.enabled()&&a.enable(!1)})});c.each(["header","footer"],function(a,b){g.Api.register("fixedHeader."+b+"Offset()",function(a){var c=this.context;return a===h?c.length&&c[0]._fixedHeader?c[0]._fixedHeader[b+"Offset"]():h:this.iterator("table",function(c){if(c=
|
||||||
|
c._fixedHeader)c[b+"Offset"](a)})})});return l});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Bootstrap 4 styling wrapper for FixedHeader
|
||||||
|
©2018 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
(function(c){"function"===typeof define&&define.amd?define(["jquery","datatables.net-bs4","datatables.net-fixedheader"],function(a){return c(a,window,document)}):"object"===typeof exports?module.exports=function(a,b){a||(a=window);b&&b.fn.dataTable||(b=require("datatables.net-bs4")(a,b).$);b.fn.dataTable.FixedHeader||require("datatables.net-fixedheader")(a,b);return c(b,a,a.document)}:c(jQuery,window,document)})(function(c,a,b,d){return c.fn.dataTable});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Copyright 2014-2020 SpryMedia Ltd.
|
||||||
|
|
||||||
|
This source file is free software, available under the following license:
|
||||||
|
MIT license - http://datatables.net/license/mit
|
||||||
|
|
||||||
|
This source file is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the license files for details.
|
||||||
|
|
||||||
|
For details please refer to: http://www.datatables.net
|
||||||
|
Responsive 2.2.6
|
||||||
|
2014-2020 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.findInternal=function(b,k,m){b instanceof String&&(b=String(b));for(var n=b.length,p=0;p<n;p++){var y=b[p];if(k.call(m,y,p,b))return{i:p,v:y}}return{i:-1,v:void 0}};$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;$jscomp.ISOLATE_POLYFILLS=!1;
|
||||||
|
$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(b,k,m){if(b==Array.prototype||b==Object.prototype)return b;b[k]=m.value;return b};$jscomp.getGlobal=function(b){b=["object"==typeof globalThis&&globalThis,b,"object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global];for(var k=0;k<b.length;++k){var m=b[k];if(m&&m.Math==Math)return m}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.IS_SYMBOL_NATIVE="function"===typeof Symbol&&"symbol"===typeof Symbol("x");$jscomp.TRUST_ES6_POLYFILLS=!$jscomp.ISOLATE_POLYFILLS||$jscomp.IS_SYMBOL_NATIVE;$jscomp.polyfills={};$jscomp.propertyToPolyfillSymbol={};$jscomp.POLYFILL_PREFIX="$jscp$";var $jscomp$lookupPolyfilledValue=function(b,k){var m=$jscomp.propertyToPolyfillSymbol[k];if(null==m)return b[k];m=b[m];return void 0!==m?m:b[k]};
|
||||||
|
$jscomp.polyfill=function(b,k,m,n){k&&($jscomp.ISOLATE_POLYFILLS?$jscomp.polyfillIsolated(b,k,m,n):$jscomp.polyfillUnisolated(b,k,m,n))};$jscomp.polyfillUnisolated=function(b,k,m,n){m=$jscomp.global;b=b.split(".");for(n=0;n<b.length-1;n++){var p=b[n];if(!(p in m))return;m=m[p]}b=b[b.length-1];n=m[b];k=k(n);k!=n&&null!=k&&$jscomp.defineProperty(m,b,{configurable:!0,writable:!0,value:k})};
|
||||||
|
$jscomp.polyfillIsolated=function(b,k,m,n){var p=b.split(".");b=1===p.length;n=p[0];n=!b&&n in $jscomp.polyfills?$jscomp.polyfills:$jscomp.global;for(var y=0;y<p.length-1;y++){var z=p[y];if(!(z in n))return;n=n[z]}p=p[p.length-1];m=$jscomp.IS_SYMBOL_NATIVE&&"es6"===m?n[p]:null;k=k(m);null!=k&&(b?$jscomp.defineProperty($jscomp.polyfills,p,{configurable:!0,writable:!0,value:k}):k!==m&&($jscomp.propertyToPolyfillSymbol[p]=$jscomp.IS_SYMBOL_NATIVE?$jscomp.global.Symbol(p):$jscomp.POLYFILL_PREFIX+p,p=
|
||||||
|
$jscomp.propertyToPolyfillSymbol[p],$jscomp.defineProperty(n,p,{configurable:!0,writable:!0,value:k})))};$jscomp.polyfill("Array.prototype.find",function(b){return b?b:function(k,m){return $jscomp.findInternal(this,k,m).v}},"es6","es3");
|
||||||
|
(function(b){"function"===typeof define&&define.amd?define(["jquery","datatables.net"],function(k){return b(k,window,document)}):"object"===typeof exports?module.exports=function(k,m){k||(k=window);m&&m.fn.dataTable||(m=require("datatables.net")(k,m).$);return b(m,k,k.document)}:b(jQuery,window,document)})(function(b,k,m,n){function p(a,c,d){var f=c+"-"+d;if(A[f])return A[f];var g=[];a=a.cell(c,d).node().childNodes;c=0;for(d=a.length;c<d;c++)g.push(a[c]);return A[f]=g}function y(a,c,d){var f=c+"-"+
|
||||||
|
d;if(A[f]){a=a.cell(c,d).node();d=A[f][0].parentNode.childNodes;c=[];for(var g=0,l=d.length;g<l;g++)c.push(d[g]);d=0;for(g=c.length;d<g;d++)a.appendChild(c[d]);A[f]=n}}var z=b.fn.dataTable,u=function(a,c){if(!z.versionCheck||!z.versionCheck("1.10.10"))throw"DataTables Responsive requires DataTables 1.10.10 or newer";this.s={dt:new z.Api(a),columns:[],current:[]};this.s.dt.settings()[0].responsive||(c&&"string"===typeof c.details?c.details={type:c.details}:c&&!1===c.details?c.details={type:!1}:c&&
|
||||||
|
!0===c.details&&(c.details={type:"inline"}),this.c=b.extend(!0,{},u.defaults,z.defaults.responsive,c),a.responsive=this,this._constructor())};b.extend(u.prototype,{_constructor:function(){var a=this,c=this.s.dt,d=c.settings()[0],f=b(k).innerWidth();c.settings()[0]._responsive=this;b(k).on("resize.dtr orientationchange.dtr",z.util.throttle(function(){var g=b(k).innerWidth();g!==f&&(a._resize(),f=g)}));d.oApi._fnCallbackReg(d,"aoRowCreatedCallback",function(g,l,h){-1!==b.inArray(!1,a.s.current)&&b(">td, >th",
|
||||||
|
g).each(function(e){e=c.column.index("toData",e);!1===a.s.current[e]&&b(this).css("display","none")})});c.on("destroy.dtr",function(){c.off(".dtr");b(c.table().body()).off(".dtr");b(k).off("resize.dtr orientationchange.dtr");c.cells(".dtr-control").nodes().to$().removeClass("dtr-control");b.each(a.s.current,function(g,l){!1===l&&a._setColumnVis(g,!0)})});this.c.breakpoints.sort(function(g,l){return g.width<l.width?1:g.width>l.width?-1:0});this._classLogic();this._resizeAuto();d=this.c.details;!1!==
|
||||||
|
d.type&&(a._detailsInit(),c.on("column-visibility.dtr",function(){a._timer&&clearTimeout(a._timer);a._timer=setTimeout(function(){a._timer=null;a._classLogic();a._resizeAuto();a._resize(!0);a._redrawChildren()},100)}),c.on("draw.dtr",function(){a._redrawChildren()}),b(c.table().node()).addClass("dtr-"+d.type));c.on("column-reorder.dtr",function(g,l,h){a._classLogic();a._resizeAuto();a._resize(!0)});c.on("column-sizing.dtr",function(){a._resizeAuto();a._resize()});c.on("preXhr.dtr",function(){var g=
|
||||||
|
[];c.rows().every(function(){this.child.isShown()&&g.push(this.id(!0))});c.one("draw.dtr",function(){a._resizeAuto();a._resize();c.rows(g).every(function(){a._detailsDisplay(this,!1)})})});c.on("draw.dtr",function(){a._controlClass()}).on("init.dtr",function(g,l,h){"dt"===g.namespace&&(a._resizeAuto(),a._resize(),b.inArray(!1,a.s.current)&&c.columns.adjust())});this._resize()},_columnsVisiblity:function(a){var c=this.s.dt,d=this.s.columns,f,g=d.map(function(t,v){return{columnIdx:v,priority:t.priority}}).sort(function(t,
|
||||||
|
v){return t.priority!==v.priority?t.priority-v.priority:t.columnIdx-v.columnIdx}),l=b.map(d,function(t,v){return!1===c.column(v).visible()?"not-visible":t.auto&&null===t.minWidth?!1:!0===t.auto?"-":-1!==b.inArray(a,t.includeIn)}),h=0;var e=0;for(f=l.length;e<f;e++)!0===l[e]&&(h+=d[e].minWidth);e=c.settings()[0].oScroll;e=e.sY||e.sX?e.iBarWidth:0;h=c.table().container().offsetWidth-e-h;e=0;for(f=l.length;e<f;e++)d[e].control&&(h-=d[e].minWidth);var r=!1;e=0;for(f=g.length;e<f;e++){var q=g[e].columnIdx;
|
||||||
|
"-"===l[q]&&!d[q].control&&d[q].minWidth&&(r||0>h-d[q].minWidth?(r=!0,l[q]=!1):l[q]=!0,h-=d[q].minWidth)}g=!1;e=0;for(f=d.length;e<f;e++)if(!d[e].control&&!d[e].never&&!1===l[e]){g=!0;break}e=0;for(f=d.length;e<f;e++)d[e].control&&(l[e]=g),"not-visible"===l[e]&&(l[e]=!1);-1===b.inArray(!0,l)&&(l[0]=!0);return l},_classLogic:function(){var a=this,c=this.c.breakpoints,d=this.s.dt,f=d.columns().eq(0).map(function(h){var e=this.column(h),r=e.header().className;h=d.settings()[0].aoColumns[h].responsivePriority;
|
||||||
|
e=e.header().getAttribute("data-priority");h===n&&(h=e===n||null===e?1E4:1*e);return{className:r,includeIn:[],auto:!1,control:!1,never:r.match(/\bnever\b/)?!0:!1,priority:h}}),g=function(h,e){h=f[h].includeIn;-1===b.inArray(e,h)&&h.push(e)},l=function(h,e,r,q){if(!r)f[h].includeIn.push(e);else if("max-"===r)for(q=a._find(e).width,e=0,r=c.length;e<r;e++)c[e].width<=q&&g(h,c[e].name);else if("min-"===r)for(q=a._find(e).width,e=0,r=c.length;e<r;e++)c[e].width>=q&&g(h,c[e].name);else if("not-"===r)for(e=
|
||||||
|
0,r=c.length;e<r;e++)-1===c[e].name.indexOf(q)&&g(h,c[e].name)};f.each(function(h,e){for(var r=h.className.split(" "),q=!1,t=0,v=r.length;t<v;t++){var B=r[t].trim();if("all"===B){q=!0;h.includeIn=b.map(c,function(w){return w.name});return}if("none"===B||h.never){q=!0;return}if("control"===B||"dtr-control"===B){q=!0;h.control=!0;return}b.each(c,function(w,D){w=D.name.split("-");var x=B.match(new RegExp("(min\\-|max\\-|not\\-)?("+w[0]+")(\\-[_a-zA-Z0-9])?"));x&&(q=!0,x[2]===w[0]&&x[3]==="-"+w[1]?l(e,
|
||||||
|
D.name,x[1],x[2]+x[3]):x[2]!==w[0]||x[3]||l(e,D.name,x[1],x[2]))})}q||(h.auto=!0)});this.s.columns=f},_controlClass:function(){if("inline"===this.c.details.type){var a=this.s.dt,c=b.inArray(!0,this.s.current);a.cells(null,function(d){return d!==c},{page:"current"}).nodes().to$().filter(".dtr-control").removeClass("dtr-control");a.cells(null,c,{page:"current"}).nodes().to$().addClass("dtr-control")}},_detailsDisplay:function(a,c){var d=this,f=this.s.dt,g=this.c.details;if(g&&!1!==g.type){var l=g.display(a,
|
||||||
|
c,function(){return g.renderer(f,a[0],d._detailsObj(a[0]))});!0!==l&&!1!==l||b(f.table().node()).triggerHandler("responsive-display.dt",[f,a,l,c])}},_detailsInit:function(){var a=this,c=this.s.dt,d=this.c.details;"inline"===d.type&&(d.target="td.dtr-control, th.dtr-control");c.on("draw.dtr",function(){a._tabIndexes()});a._tabIndexes();b(c.table().body()).on("keyup.dtr","td, th",function(g){13===g.keyCode&&b(this).data("dtr-keyboard")&&b(this).click()});var f=d.target;d="string"===typeof f?f:"td, th";
|
||||||
|
if(f!==n||null!==f)b(c.table().body()).on("click.dtr mousedown.dtr mouseup.dtr",d,function(g){if(b(c.table().node()).hasClass("collapsed")&&-1!==b.inArray(b(this).closest("tr").get(0),c.rows().nodes().toArray())){if("number"===typeof f){var l=0>f?c.columns().eq(0).length+f:f;if(c.cell(this).index().column!==l)return}l=c.row(b(this).closest("tr"));"click"===g.type?a._detailsDisplay(l,!1):"mousedown"===g.type?b(this).css("outline","none"):"mouseup"===g.type&&b(this).trigger("blur").css("outline","")}})},
|
||||||
|
_detailsObj:function(a){var c=this,d=this.s.dt;return b.map(this.s.columns,function(f,g){if(!f.never&&!f.control)return f=d.settings()[0].aoColumns[g],{className:f.sClass,columnIndex:g,data:d.cell(a,g).render(c.c.orthogonal),hidden:d.column(g).visible()&&!c.s.current[g],rowIndex:a,title:null!==f.sTitle?f.sTitle:b(d.column(g).header()).text()}})},_find:function(a){for(var c=this.c.breakpoints,d=0,f=c.length;d<f;d++)if(c[d].name===a)return c[d]},_redrawChildren:function(){var a=this,c=this.s.dt;c.rows({page:"current"}).iterator("row",
|
||||||
|
function(d,f){c.row(f);a._detailsDisplay(c.row(f),!0)})},_resize:function(a){var c=this,d=this.s.dt,f=b(k).innerWidth(),g=this.c.breakpoints,l=g[0].name,h=this.s.columns,e,r=this.s.current.slice();for(e=g.length-1;0<=e;e--)if(f<=g[e].width){l=g[e].name;break}var q=this._columnsVisiblity(l);this.s.current=q;g=!1;e=0;for(f=h.length;e<f;e++)if(!1===q[e]&&!h[e].never&&!h[e].control&&!1===!d.column(e).visible()){g=!0;break}b(d.table().node()).toggleClass("collapsed",g);var t=!1,v=0;d.columns().eq(0).each(function(B,
|
||||||
|
w){!0===q[w]&&v++;if(a||q[w]!==r[w])t=!0,c._setColumnVis(B,q[w])});t&&(this._redrawChildren(),b(d.table().node()).trigger("responsive-resize.dt",[d,this.s.current]),0===d.page.info().recordsDisplay&&b("td",d.table().body()).eq(0).attr("colspan",v));c._controlClass()},_resizeAuto:function(){var a=this.s.dt,c=this.s.columns;if(this.c.auto&&-1!==b.inArray(!0,b.map(c,function(e){return e.auto}))){b.isEmptyObject(A)||b.each(A,function(e){e=e.split("-");y(a,1*e[0],1*e[1])});a.table().node();var d=a.table().node().cloneNode(!1),
|
||||||
|
f=b(a.table().header().cloneNode(!1)).appendTo(d),g=b(a.table().body()).clone(!1,!1).empty().appendTo(d);d.style.width="auto";var l=a.columns().header().filter(function(e){return a.column(e).visible()}).to$().clone(!1).css("display","table-cell").css("width","auto").css("min-width",0);b(g).append(b(a.rows({page:"current"}).nodes()).clone(!1)).find("th, td").css("display","");if(g=a.table().footer()){g=b(g.cloneNode(!1)).appendTo(d);var h=a.columns().footer().filter(function(e){return a.column(e).visible()}).to$().clone(!1).css("display",
|
||||||
|
"table-cell");b("<tr/>").append(h).appendTo(g)}b("<tr/>").append(l).appendTo(f);"inline"===this.c.details.type&&b(d).addClass("dtr-inline collapsed");b(d).find("[name]").removeAttr("name");b(d).css("position","relative");d=b("<div/>").css({width:1,height:1,overflow:"hidden",clear:"both"}).append(d);d.insertBefore(a.table().node());l.each(function(e){e=a.column.index("fromVisible",e);c[e].minWidth=this.offsetWidth||0});d.remove()}},_responsiveOnlyHidden:function(){var a=this.s.dt;return b.map(this.s.current,
|
||||||
|
function(c,d){return!1===a.column(d).visible()?!0:c})},_setColumnVis:function(a,c){var d=this.s.dt;c=c?"":"none";b(d.column(a).header()).css("display",c);b(d.column(a).footer()).css("display",c);d.column(a).nodes().to$().css("display",c);b.isEmptyObject(A)||d.cells(null,a).indexes().each(function(f){y(d,f.row,f.column)})},_tabIndexes:function(){var a=this.s.dt,c=a.cells({page:"current"}).nodes().to$(),d=a.settings()[0],f=this.c.details.target;c.filter("[data-dtr-keyboard]").removeData("[data-dtr-keyboard]");
|
||||||
|
"number"===typeof f?a.cells(null,f,{page:"current"}).nodes().to$().attr("tabIndex",d.iTabIndex).data("dtr-keyboard",1):("td:first-child, th:first-child"===f&&(f=">td:first-child, >th:first-child"),b(f,a.rows({page:"current"}).nodes()).attr("tabIndex",d.iTabIndex).data("dtr-keyboard",1))}});u.breakpoints=[{name:"desktop",width:Infinity},{name:"tablet-l",width:1024},{name:"tablet-p",width:768},{name:"mobile-l",width:480},{name:"mobile-p",width:320}];u.display={childRow:function(a,c,d){if(c){if(b(a.node()).hasClass("parent"))return a.child(d(),
|
||||||
|
"child").show(),!0}else{if(a.child.isShown())return a.child(!1),b(a.node()).removeClass("parent"),!1;a.child(d(),"child").show();b(a.node()).addClass("parent");return!0}},childRowImmediate:function(a,c,d){if(!c&&a.child.isShown()||!a.responsive.hasHidden())return a.child(!1),b(a.node()).removeClass("parent"),!1;a.child(d(),"child").show();b(a.node()).addClass("parent");return!0},modal:function(a){return function(c,d,f){if(d)b("div.dtr-modal-content").empty().append(f());else{var g=function(){l.remove();
|
||||||
|
b(m).off("keypress.dtr")},l=b('<div class="dtr-modal"/>').append(b('<div class="dtr-modal-display"/>').append(b('<div class="dtr-modal-content"/>').append(f())).append(b('<div class="dtr-modal-close">×</div>').click(function(){g()}))).append(b('<div class="dtr-modal-background"/>').click(function(){g()})).appendTo("body");b(m).on("keyup.dtr",function(h){27===h.keyCode&&(h.stopPropagation(),g())})}a&&a.header&&b("div.dtr-modal-content").prepend("<h2>"+a.header(c)+"</h2>")}}};var A={};u.renderer=
|
||||||
|
{listHiddenNodes:function(){return function(a,c,d){var f=b('<ul data-dtr-index="'+c+'" class="dtr-details"/>'),g=!1;b.each(d,function(l,h){h.hidden&&(b("<li "+(h.className?'class="'+h.className+'"':"")+' data-dtr-index="'+h.columnIndex+'" data-dt-row="'+h.rowIndex+'" data-dt-column="'+h.columnIndex+'"><span class="dtr-title">'+h.title+"</span> </li>").append(b('<span class="dtr-data"/>').append(p(a,h.rowIndex,h.columnIndex))).appendTo(f),g=!0)});return g?f:!1}},listHidden:function(){return function(a,
|
||||||
|
c,d){return(a=b.map(d,function(f){var g=f.className?'class="'+f.className+'"':"";return f.hidden?"<li "+g+' data-dtr-index="'+f.columnIndex+'" data-dt-row="'+f.rowIndex+'" data-dt-column="'+f.columnIndex+'"><span class="dtr-title">'+f.title+'</span> <span class="dtr-data">'+f.data+"</span></li>":""}).join(""))?b('<ul data-dtr-index="'+c+'" class="dtr-details"/>').append(a):!1}},tableAll:function(a){a=b.extend({tableClass:""},a);return function(c,d,f){c=b.map(f,function(g){return"<tr "+(g.className?
|
||||||
|
'class="'+g.className+'"':"")+' data-dt-row="'+g.rowIndex+'" data-dt-column="'+g.columnIndex+'"><td>'+g.title+":</td> <td>"+g.data+"</td></tr>"}).join("");return b('<table class="'+a.tableClass+' dtr-details" width="100%"/>').append(c)}}};u.defaults={breakpoints:u.breakpoints,auto:!0,details:{display:u.display.childRow,renderer:u.renderer.listHidden(),target:0,type:"inline"},orthogonal:"display"};var C=b.fn.dataTable.Api;C.register("responsive()",function(){return this});C.register("responsive.index()",
|
||||||
|
function(a){a=b(a);return{column:a.data("dtr-index"),row:a.parent().data("dtr-index")}});C.register("responsive.rebuild()",function(){return this.iterator("table",function(a){a._responsive&&a._responsive._classLogic()})});C.register("responsive.recalc()",function(){return this.iterator("table",function(a){a._responsive&&(a._responsive._resizeAuto(),a._responsive._resize())})});C.register("responsive.hasHidden()",function(){var a=this.context[0];return a._responsive?-1!==b.inArray(!1,a._responsive._responsiveOnlyHidden()):
|
||||||
|
!1});C.registerPlural("columns().responsiveHidden()","column().responsiveHidden()",function(){return this.iterator("column",function(a,c){return a._responsive?a._responsive._responsiveOnlyHidden()[c]:!1},1)});u.version="2.2.6";b.fn.dataTable.Responsive=u;b.fn.DataTable.Responsive=u;b(m).on("preInit.dt.dtr",function(a,c,d){"dt"===a.namespace&&(b(c.nTable).hasClass("responsive")||b(c.nTable).hasClass("dt-responsive")||c.oInit.responsive||z.defaults.responsive)&&(a=c.oInit.responsive,!1!==a&&new u(c,
|
||||||
|
b.isPlainObject(a)?a:{}))});return u});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Bootstrap 4 integration for DataTables' Responsive
|
||||||
|
©2016 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.findInternal=function(a,b,c){a instanceof String&&(a=String(a));for(var e=a.length,d=0;d<e;d++){var f=a[d];if(b.call(c,f,d,a))return{i:d,v:f}}return{i:-1,v:void 0}};$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;$jscomp.ISOLATE_POLYFILLS=!1;
|
||||||
|
$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(a,b,c){if(a==Array.prototype||a==Object.prototype)return a;a[b]=c.value;return a};$jscomp.getGlobal=function(a){a=["object"==typeof globalThis&&globalThis,a,"object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global];for(var b=0;b<a.length;++b){var c=a[b];if(c&&c.Math==Math)return c}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.IS_SYMBOL_NATIVE="function"===typeof Symbol&&"symbol"===typeof Symbol("x");$jscomp.TRUST_ES6_POLYFILLS=!$jscomp.ISOLATE_POLYFILLS||$jscomp.IS_SYMBOL_NATIVE;$jscomp.polyfills={};$jscomp.propertyToPolyfillSymbol={};$jscomp.POLYFILL_PREFIX="$jscp$";var $jscomp$lookupPolyfilledValue=function(a,b){var c=$jscomp.propertyToPolyfillSymbol[b];if(null==c)return a[b];c=a[c];return void 0!==c?c:a[b]};
|
||||||
|
$jscomp.polyfill=function(a,b,c,e){b&&($jscomp.ISOLATE_POLYFILLS?$jscomp.polyfillIsolated(a,b,c,e):$jscomp.polyfillUnisolated(a,b,c,e))};$jscomp.polyfillUnisolated=function(a,b,c,e){c=$jscomp.global;a=a.split(".");for(e=0;e<a.length-1;e++){var d=a[e];if(!(d in c))return;c=c[d]}a=a[a.length-1];e=c[a];b=b(e);b!=e&&null!=b&&$jscomp.defineProperty(c,a,{configurable:!0,writable:!0,value:b})};
|
||||||
|
$jscomp.polyfillIsolated=function(a,b,c,e){var d=a.split(".");a=1===d.length;e=d[0];e=!a&&e in $jscomp.polyfills?$jscomp.polyfills:$jscomp.global;for(var f=0;f<d.length-1;f++){var g=d[f];if(!(g in e))return;e=e[g]}d=d[d.length-1];c=$jscomp.IS_SYMBOL_NATIVE&&"es6"===c?e[d]:null;b=b(c);null!=b&&(a?$jscomp.defineProperty($jscomp.polyfills,d,{configurable:!0,writable:!0,value:b}):b!==c&&($jscomp.propertyToPolyfillSymbol[d]=$jscomp.IS_SYMBOL_NATIVE?$jscomp.global.Symbol(d):$jscomp.POLYFILL_PREFIX+d,d=
|
||||||
|
$jscomp.propertyToPolyfillSymbol[d],$jscomp.defineProperty(e,d,{configurable:!0,writable:!0,value:b})))};$jscomp.polyfill("Array.prototype.find",function(a){return a?a:function(b,c){return $jscomp.findInternal(this,b,c).v}},"es6","es3");
|
||||||
|
(function(a){"function"===typeof define&&define.amd?define(["jquery","datatables.net-bs4","datatables.net-responsive"],function(b){return a(b,window,document)}):"object"===typeof exports?module.exports=function(b,c){b||(b=window);c&&c.fn.dataTable||(c=require("datatables.net-bs4")(b,c).$);c.fn.dataTable.Responsive||require("datatables.net-responsive")(b,c);return a(c,b,b.document)}:a(jQuery,window,document)})(function(a,b,c,e){b=a.fn.dataTable;c=b.Responsive.display;var d=c.modal,f=a('<div class="modal fade dtr-bs-modal" role="dialog"><div class="modal-dialog" role="document"><div class="modal-content"><div class="modal-header"><button type="button" class="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">×</span></button></div><div class="modal-body"/></div></div></div>');
|
||||||
|
c.modal=function(g){return function(k,h,l){if(!a.fn.modal)d(k,h,l);else if(!h){if(g&&g.header){h=f.find("div.modal-header");var m=h.find("button").detach();h.empty().append('<h4 class="modal-title">'+g.header(k)+"</h4>").append(m)}f.find("div.modal-body").empty().append(l());f.appendTo("body").modal()}}};return b.Responsive});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Copyright 2011-2020 SpryMedia Ltd.
|
||||||
|
|
||||||
|
This source file is free software, available under the following license:
|
||||||
|
MIT license - http://datatables.net/license/mit
|
||||||
|
|
||||||
|
This source file is distributed in the hope that it will be useful, but
|
||||||
|
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
||||||
|
or FITNESS FOR A PARTICULAR PURPOSE. See the license files for details.
|
||||||
|
|
||||||
|
For details please refer to: http://www.datatables.net
|
||||||
|
Scroller 2.0.3
|
||||||
|
©2011-2020 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.findInternal=function(d,f,g){d instanceof String&&(d=String(d));for(var h=d.length,k=0;k<h;k++){var m=d[k];if(f.call(g,m,k,d))return{i:k,v:m}}return{i:-1,v:void 0}};$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;$jscomp.ISOLATE_POLYFILLS=!1;
|
||||||
|
$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(d,f,g){if(d==Array.prototype||d==Object.prototype)return d;d[f]=g.value;return d};$jscomp.getGlobal=function(d){d=["object"==typeof globalThis&&globalThis,d,"object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global];for(var f=0;f<d.length;++f){var g=d[f];if(g&&g.Math==Math)return g}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.IS_SYMBOL_NATIVE="function"===typeof Symbol&&"symbol"===typeof Symbol("x");$jscomp.TRUST_ES6_POLYFILLS=!$jscomp.ISOLATE_POLYFILLS||$jscomp.IS_SYMBOL_NATIVE;$jscomp.polyfills={};$jscomp.propertyToPolyfillSymbol={};$jscomp.POLYFILL_PREFIX="$jscp$";var $jscomp$lookupPolyfilledValue=function(d,f){var g=$jscomp.propertyToPolyfillSymbol[f];if(null==g)return d[f];g=d[g];return void 0!==g?g:d[f]};
|
||||||
|
$jscomp.polyfill=function(d,f,g,h){f&&($jscomp.ISOLATE_POLYFILLS?$jscomp.polyfillIsolated(d,f,g,h):$jscomp.polyfillUnisolated(d,f,g,h))};$jscomp.polyfillUnisolated=function(d,f,g,h){g=$jscomp.global;d=d.split(".");for(h=0;h<d.length-1;h++){var k=d[h];if(!(k in g))return;g=g[k]}d=d[d.length-1];h=g[d];f=f(h);f!=h&&null!=f&&$jscomp.defineProperty(g,d,{configurable:!0,writable:!0,value:f})};
|
||||||
|
$jscomp.polyfillIsolated=function(d,f,g,h){var k=d.split(".");d=1===k.length;h=k[0];h=!d&&h in $jscomp.polyfills?$jscomp.polyfills:$jscomp.global;for(var m=0;m<k.length-1;m++){var q=k[m];if(!(q in h))return;h=h[q]}k=k[k.length-1];g=$jscomp.IS_SYMBOL_NATIVE&&"es6"===g?h[k]:null;f=f(g);null!=f&&(d?$jscomp.defineProperty($jscomp.polyfills,k,{configurable:!0,writable:!0,value:f}):f!==g&&($jscomp.propertyToPolyfillSymbol[k]=$jscomp.IS_SYMBOL_NATIVE?$jscomp.global.Symbol(k):$jscomp.POLYFILL_PREFIX+k,k=
|
||||||
|
$jscomp.propertyToPolyfillSymbol[k],$jscomp.defineProperty(h,k,{configurable:!0,writable:!0,value:f})))};$jscomp.polyfill("Array.prototype.find",function(d){return d?d:function(f,g){return $jscomp.findInternal(this,f,g).v}},"es6","es3");
|
||||||
|
(function(d){"function"===typeof define&&define.amd?define(["jquery","datatables.net"],function(f){return d(f,window,document)}):"object"===typeof exports?module.exports=function(f,g){f||(f=window);g&&g.fn.dataTable||(g=require("datatables.net")(f,g).$);return d(g,f,f.document)}:d(jQuery,window,document)})(function(d,f,g,h){var k=d.fn.dataTable,m=function(a,b){this instanceof m?(b===h&&(b={}),a=d.fn.dataTable.Api(a),this.s={dt:a.settings()[0],dtApi:a,tableTop:0,tableBottom:0,redrawTop:0,redrawBottom:0,
|
||||||
|
autoHeight:!0,viewportRows:0,stateTO:null,stateSaveThrottle:function(){},drawTO:null,heights:{jump:null,page:null,virtual:null,scroll:null,row:null,viewport:null,labelFactor:1},topRowFloat:0,scrollDrawDiff:null,loaderVisible:!1,forceReposition:!1,baseRowTop:0,baseScrollTop:0,mousedown:!1,lastScrollTop:0},this.s=d.extend(this.s,m.oDefaults,b),this.s.heights.row=this.s.rowHeight,this.dom={force:g.createElement("div"),label:d('<div class="dts_label">0</div>'),scroller:null,table:null,loader:null},this.s.dt.oScroller||
|
||||||
|
(this.s.dt.oScroller=this,this.construct())):alert("Scroller warning: Scroller must be initialised with the 'new' keyword.")};d.extend(m.prototype,{measure:function(a){this.s.autoHeight&&this._calcRowHeight();var b=this.s.heights;b.row&&(b.viewport=this._parseHeight(d(this.dom.scroller).css("max-height")),this.s.viewportRows=parseInt(b.viewport/b.row,10)+1,this.s.dt._iDisplayLength=this.s.viewportRows*this.s.displayBuffer);var c=this.dom.label.outerHeight();b.labelFactor=(b.viewport-c)/b.scroll;(a===
|
||||||
|
h||a)&&this.s.dt.oInstance.fnDraw(!1)},pageInfo:function(){var a=this.dom.scroller.scrollTop,b=this.s.dt.fnRecordsDisplay(),c=Math.ceil(this.pixelsToRow(a+this.s.heights.viewport,!1,this.s.ani));return{start:Math.floor(this.pixelsToRow(a,!1,this.s.ani)),end:b<c?b-1:c-1}},pixelsToRow:function(a,b,c){a-=this.s.baseScrollTop;c=c?(this._domain("physicalToVirtual",this.s.baseScrollTop)+a)/this.s.heights.row:a/this.s.heights.row+this.s.baseRowTop;return b||b===h?parseInt(c,10):c},rowToPixels:function(a,
|
||||||
|
b,c){a-=this.s.baseRowTop;c=c?this._domain("virtualToPhysical",this.s.baseScrollTop):this.s.baseScrollTop;c+=a*this.s.heights.row;return b||b===h?parseInt(c,10):c},scrollToRow:function(a,b){var c=this,e=!1,l=this.rowToPixels(a),n=a-(this.s.displayBuffer-1)/2*this.s.viewportRows;0>n&&(n=0);(l>this.s.redrawBottom||l<this.s.redrawTop)&&this.s.dt._iDisplayStart!==n&&(e=!0,l=this._domain("virtualToPhysical",a*this.s.heights.row),this.s.redrawTop<l&&l<this.s.redrawBottom&&(this.s.forceReposition=!0,b=!1));
|
||||||
|
b===h||b?(this.s.ani=e,d(this.dom.scroller).animate({scrollTop:l},function(){setTimeout(function(){c.s.ani=!1},250)})):d(this.dom.scroller).scrollTop(l)},construct:function(){var a=this,b=this.s.dtApi;if(this.s.dt.oFeatures.bPaginate){this.dom.force.style.position="relative";this.dom.force.style.top="0px";this.dom.force.style.left="0px";this.dom.force.style.width="1px";this.dom.scroller=d("div."+this.s.dt.oClasses.sScrollBody,this.s.dt.nTableWrapper)[0];this.dom.scroller.appendChild(this.dom.force);
|
||||||
|
this.dom.scroller.style.position="relative";this.dom.table=d(">table",this.dom.scroller)[0];this.dom.table.style.position="absolute";this.dom.table.style.top="0px";this.dom.table.style.left="0px";d(b.table().container()).addClass("dts DTS");this.s.loadingIndicator&&(this.dom.loader=d('<div class="dataTables_processing dts_loading">'+this.s.dt.oLanguage.sLoadingRecords+"</div>").css("display","none"),d(this.dom.scroller.parentNode).css("position","relative").append(this.dom.loader));this.dom.label.appendTo(this.dom.scroller);
|
||||||
|
this.s.heights.row&&"auto"!=this.s.heights.row&&(this.s.autoHeight=!1);this.s.ingnoreScroll=!0;d(this.dom.scroller).on("scroll.dt-scroller",function(l){a._scroll.call(a)});d(this.dom.scroller).on("touchstart.dt-scroller",function(){a._scroll.call(a)});d(this.dom.scroller).on("mousedown.dt-scroller",function(){a.s.mousedown=!0}).on("mouseup.dt-scroller",function(){a.s.labelVisible=!1;a.s.mousedown=!1;a.dom.label.css("display","none")});d(f).on("resize.dt-scroller",function(){a.measure(!1);a._info()});
|
||||||
|
var c=!0,e=b.state.loaded();b.on("stateSaveParams.scroller",function(l,n,p){c&&e?(p.scroller=e.scroller,c=!1):p.scroller={topRow:a.s.topRowFloat,baseScrollTop:a.s.baseScrollTop,baseRowTop:a.s.baseRowTop,scrollTop:a.s.lastScrollTop}});e&&e.scroller&&(this.s.topRowFloat=e.scroller.topRow,this.s.baseScrollTop=e.scroller.baseScrollTop,this.s.baseRowTop=e.scroller.baseRowTop);this.measure(!1);a.s.stateSaveThrottle=a.s.dt.oApi._fnThrottle(function(){a.s.dtApi.state.save()},500);b.on("init.scroller",function(){a.measure(!1);
|
||||||
|
a.s.scrollType="jump";a._draw();b.on("draw.scroller",function(){a._draw()})});b.on("preDraw.dt.scroller",function(){a._scrollForce()});b.on("destroy.scroller",function(){d(f).off("resize.dt-scroller");d(a.dom.scroller).off(".dt-scroller");d(a.s.dt.nTable).off(".scroller");d(a.s.dt.nTableWrapper).removeClass("DTS");d("div.DTS_Loading",a.dom.scroller.parentNode).remove();a.dom.table.style.position="";a.dom.table.style.top="";a.dom.table.style.left=""})}else this.s.dt.oApi._fnLog(this.s.dt,0,"Pagination must be enabled for Scroller")},
|
||||||
|
_calcRowHeight:function(){var a=this.s.dt,b=a.nTable,c=b.cloneNode(!1),e=d("<tbody/>").appendTo(c),l=d('<div class="'+a.oClasses.sWrapper+' DTS"><div class="'+a.oClasses.sScrollWrapper+'"><div class="'+a.oClasses.sScrollBody+'"></div></div></div>');d("tbody tr:lt(4)",b).clone().appendTo(e);var n=d("tr",e).length;if(1===n)e.prepend("<tr><td> </td></tr>"),e.append("<tr><td> </td></tr>");else for(;3>n;n++)e.append("<tr><td> </td></tr>");d("div."+a.oClasses.sScrollBody,l).append(c);a=this.s.dt.nHolding||
|
||||||
|
b.parentNode;d(a).is(":visible")||(a="body");l.find("input").removeAttr("name");l.appendTo(a);this.s.heights.row=d("tr",e).eq(1).outerHeight();l.remove()},_draw:function(){var a=this,b=this.s.heights,c=this.dom.scroller.scrollTop,e=d(this.s.dt.nTable).height(),l=this.s.dt._iDisplayStart,n=this.s.dt._iDisplayLength,p=this.s.dt.fnRecordsDisplay();this.s.skip=!0;!this.s.dt.bSorted&&!this.s.dt.bFiltered||0!==l||this.s.dt._drawHold||(this.s.topRowFloat=0);c="jump"===this.s.scrollType?this._domain("virtualToPhysical",
|
||||||
|
this.s.topRowFloat*b.row):c;this.s.baseScrollTop=c;this.s.baseRowTop=this.s.topRowFloat;var r=c-(this.s.topRowFloat-l)*b.row;0===l?r=0:l+n>=p&&(r=b.scroll-e);this.dom.table.style.top=r+"px";this.s.tableTop=r;this.s.tableBottom=e+this.s.tableTop;e=(c-this.s.tableTop)*this.s.boundaryScale;this.s.redrawTop=c-e;this.s.redrawBottom=c+e>b.scroll-b.viewport-b.row?b.scroll-b.viewport-b.row:c+e;this.s.skip=!1;this.s.dt.oFeatures.bStateSave&&null!==this.s.dt.oLoadedState&&"undefined"!=typeof this.s.dt.oLoadedState.scroller?
|
||||||
|
((b=!this.s.dt.sAjaxSource&&!a.s.dt.ajax||this.s.dt.oFeatures.bServerSide?!1:!0)&&2==this.s.dt.iDraw||!b&&1==this.s.dt.iDraw)&&setTimeout(function(){d(a.dom.scroller).scrollTop(a.s.dt.oLoadedState.scroller.scrollTop);setTimeout(function(){a.s.ingnoreScroll=!1},0)},0):a.s.ingnoreScroll=!1;this.s.dt.oFeatures.bInfo&&setTimeout(function(){a._info.call(a)},0);this.dom.loader&&this.s.loaderVisible&&(this.dom.loader.css("display","none"),this.s.loaderVisible=!1)},_domain:function(a,b){var c=this.s.heights;
|
||||||
|
if(c.virtual===c.scroll||1E4>b)return b;if("virtualToPhysical"===a&&b>=c.virtual-1E4)return a=c.virtual-b,c.scroll-a;if("physicalToVirtual"===a&&b>=c.scroll-1E4)return a=c.scroll-b,c.virtual-a;c=(c.virtual-1E4-1E4)/(c.scroll-1E4-1E4);var e=1E4-1E4*c;return"virtualToPhysical"===a?(b-e)/c:c*b+e},_info:function(){if(this.s.dt.oFeatures.bInfo){var a=this.s.dt,b=a.oLanguage,c=this.dom.scroller.scrollTop,e=Math.floor(this.pixelsToRow(c,!1,this.s.ani)+1),l=a.fnRecordsTotal(),n=a.fnRecordsDisplay();c=Math.ceil(this.pixelsToRow(c+
|
||||||
|
this.s.heights.viewport,!1,this.s.ani));c=n<c?n:c;var p=a.fnFormatNumber(e),r=a.fnFormatNumber(c),t=a.fnFormatNumber(l),u=a.fnFormatNumber(n);p=0===a.fnRecordsDisplay()&&a.fnRecordsDisplay()==a.fnRecordsTotal()?b.sInfoEmpty+b.sInfoPostFix:0===a.fnRecordsDisplay()?b.sInfoEmpty+" "+b.sInfoFiltered.replace("_MAX_",t)+b.sInfoPostFix:a.fnRecordsDisplay()==a.fnRecordsTotal()?b.sInfo.replace("_START_",p).replace("_END_",r).replace("_MAX_",t).replace("_TOTAL_",u)+b.sInfoPostFix:b.sInfo.replace("_START_",
|
||||||
|
p).replace("_END_",r).replace("_MAX_",t).replace("_TOTAL_",u)+" "+b.sInfoFiltered.replace("_MAX_",a.fnFormatNumber(a.fnRecordsTotal()))+b.sInfoPostFix;(b=b.fnInfoCallback)&&(p=b.call(a.oInstance,a,e,c,l,n,p));e=a.aanFeatures.i;if("undefined"!=typeof e)for(l=0,n=e.length;l<n;l++)d(e[l]).html(p);d(a.nTable).triggerHandler("info.dt")}},_parseHeight:function(a){var b,c=/^([+-]?(?:\d+(?:\.\d+)?|\.\d+))(px|em|rem|vh)$/.exec(a);if(null===c)return 0;a=parseFloat(c[1]);c=c[2];"px"===c?b=a:"vh"===c?b=a/100*
|
||||||
|
d(f).height():"rem"===c?b=a*parseFloat(d(":root").css("font-size")):"em"===c&&(b=a*parseFloat(d("body").css("font-size")));return b?b:0},_scroll:function(){var a=this,b=this.s.heights,c=this.dom.scroller.scrollTop;if(!this.s.skip&&!this.s.ingnoreScroll&&c!==this.s.lastScrollTop)if(this.s.dt.bFiltered||this.s.dt.bSorted)this.s.lastScrollTop=0;else{this._info();clearTimeout(this.s.stateTO);this.s.stateTO=setTimeout(function(){a.s.dtApi.state.save()},250);this.s.scrollType=Math.abs(c-this.s.lastScrollTop)>
|
||||||
|
b.viewport?"jump":"cont";this.s.topRowFloat="cont"===this.s.scrollType?this.pixelsToRow(c,!1,!1):this._domain("physicalToVirtual",c)/b.row;0>this.s.topRowFloat&&(this.s.topRowFloat=0);if(this.s.forceReposition||c<this.s.redrawTop||c>this.s.redrawBottom){var e=Math.ceil((this.s.displayBuffer-1)/2*this.s.viewportRows);e=parseInt(this.s.topRowFloat,10)-e;this.s.forceReposition=!1;0>=e?e=0:e+this.s.dt._iDisplayLength>this.s.dt.fnRecordsDisplay()?(e=this.s.dt.fnRecordsDisplay()-this.s.dt._iDisplayLength,
|
||||||
|
0>e&&(e=0)):0!==e%2&&e++;this.s.targetTop=e;e!=this.s.dt._iDisplayStart&&(this.s.tableTop=d(this.s.dt.nTable).offset().top,this.s.tableBottom=d(this.s.dt.nTable).height()+this.s.tableTop,e=function(){a.s.dt._iDisplayStart=a.s.targetTop;a.s.dt.oApi._fnDraw(a.s.dt)},this.s.dt.oFeatures.bServerSide?(this.s.forceReposition=!0,clearTimeout(this.s.drawTO),this.s.drawTO=setTimeout(e,this.s.serverWait)):e(),this.dom.loader&&!this.s.loaderVisible&&(this.dom.loader.css("display","block"),this.s.loaderVisible=
|
||||||
|
!0))}else this.s.topRowFloat=this.pixelsToRow(c,!1,!0);this.s.lastScrollTop=c;this.s.stateSaveThrottle();"jump"===this.s.scrollType&&this.s.mousedown&&(this.s.labelVisible=!0);this.s.labelVisible&&this.dom.label.html(this.s.dt.fnFormatNumber(parseInt(this.s.topRowFloat,10)+1)).css("top",c+c*b.labelFactor).css("display","block")}},_scrollForce:function(){var a=this.s.heights;a.virtual=a.row*this.s.dt.fnRecordsDisplay();a.scroll=a.virtual;1E6<a.scroll&&(a.scroll=1E6);this.dom.force.style.height=a.scroll>
|
||||||
|
this.s.heights.row?a.scroll+"px":this.s.heights.row+"px"}});m.defaults={boundaryScale:.5,displayBuffer:9,loadingIndicator:!1,rowHeight:"auto",serverWait:200};m.oDefaults=m.defaults;m.version="2.0.3";d(g).on("preInit.dt.dtscroller",function(a,b){if("dt"===a.namespace){a=b.oInit.scroller;var c=k.defaults.scroller;if(a||c)c=d.extend({},a,c),!1!==a&&new m(b,c)}});d.fn.dataTable.Scroller=m;d.fn.DataTable.Scroller=m;var q=d.fn.dataTable.Api;q.register("scroller()",function(){return this});q.register("scroller().rowToPixels()",
|
||||||
|
function(a,b,c){var e=this.context;if(e.length&&e[0].oScroller)return e[0].oScroller.rowToPixels(a,b,c)});q.register("scroller().pixelsToRow()",function(a,b,c){var e=this.context;if(e.length&&e[0].oScroller)return e[0].oScroller.pixelsToRow(a,b,c)});q.register(["scroller().scrollToRow()","scroller.toPosition()"],function(a,b){this.iterator("table",function(c){c.oScroller&&c.oScroller.scrollToRow(a,b)});return this});q.register("row().scrollTo()",function(a){var b=this;this.iterator("row",function(c,
|
||||||
|
e){c.oScroller&&(e=b.rows({order:"applied",search:"applied"}).indexes().indexOf(e),c.oScroller.scrollToRow(e,a))});return this});q.register("scroller.measure()",function(a){this.iterator("table",function(b){b.oScroller&&b.oScroller.measure(a)});return this});q.register("scroller.page()",function(){var a=this.context;if(a.length&&a[0].oScroller)return a[0].oScroller.pageInfo()});return m});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
Bootstrap 4 styling wrapper for Scroller
|
||||||
|
©2018 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
(function(c){"function"===typeof define&&define.amd?define(["jquery","datatables.net-bs4","datatables.net-scroller"],function(a){return c(a,window,document)}):"object"===typeof exports?module.exports=function(a,b){a||(a=window);b&&b.fn.dataTable||(b=require("datatables.net-bs4")(a,b).$);b.fn.dataTable.Scroller||require("datatables.net-scroller")(a,b);return c(b,a,a.document)}:c(jQuery,window,document)})(function(c,a,b,d){return c.fn.dataTable});
|
||||||
|
|
||||||
|
|
||||||
|
/*!
|
||||||
|
SearchPanes 1.2.2
|
||||||
|
2019-2020 SpryMedia Ltd - datatables.net/license
|
||||||
|
*/
|
||||||
|
var $jscomp=$jscomp||{};$jscomp.scope={};$jscomp.getGlobal=function(m){m=["object"==typeof globalThis&&globalThis,m,"object"==typeof window&&window,"object"==typeof self&&self,"object"==typeof global&&global];for(var t=0;t<m.length;++t){var h=m[t];if(h&&h.Math==Math)return h}throw Error("Cannot find global object");};$jscomp.global=$jscomp.getGlobal(this);
|
||||||
|
$jscomp.checkEs6ConformanceViaProxy=function(){try{var m={},t=Object.create(new $jscomp.global.Proxy(m,{get:function(h,r,v){return h==m&&"q"==r&&v==t}}));return!0===t.q}catch(h){return!1}};$jscomp.USE_PROXY_FOR_ES6_CONFORMANCE_CHECKS=!1;$jscomp.ES6_CONFORMANCE=$jscomp.USE_PROXY_FOR_ES6_CONFORMANCE_CHECKS&&$jscomp.checkEs6ConformanceViaProxy();$jscomp.arrayIteratorImpl=function(m){var t=0;return function(){return t<m.length?{done:!1,value:m[t++]}:{done:!0}}};$jscomp.arrayIterator=function(m){return{next:$jscomp.arrayIteratorImpl(m)}};
|
||||||
|
$jscomp.ASSUME_ES5=!1;$jscomp.ASSUME_NO_NATIVE_MAP=!1;$jscomp.ASSUME_NO_NATIVE_SET=!1;$jscomp.SIMPLE_FROUND_POLYFILL=!1;$jscomp.ISOLATE_POLYFILLS=!1;$jscomp.defineProperty=$jscomp.ASSUME_ES5||"function"==typeof Object.defineProperties?Object.defineProperty:function(m,t,h){if(m==Array.prototype||m==Object.prototype)return m;m[t]=h.value;return m};$jscomp.IS_SYMBOL_NATIVE="function"===typeof Symbol&&"symbol"===typeof Symbol("x");$jscomp.TRUST_ES6_POLYFILLS=!$jscomp.ISOLATE_POLYFILLS||$jscomp.IS_SYMBOL_NATIVE;
|
||||||
|
$jscomp.polyfills={};$jscomp.propertyToPolyfillSymbol={};$jscomp.POLYFILL_PREFIX="$jscp$";var $jscomp$lookupPolyfilledValue=function(m,t){var h=$jscomp.propertyToPolyfillSymbol[t];if(null==h)return m[t];h=m[h];return void 0!==h?h:m[t]};$jscomp.polyfill=function(m,t,h,r){t&&($jscomp.ISOLATE_POLYFILLS?$jscomp.polyfillIsolated(m,t,h,r):$jscomp.polyfillUnisolated(m,t,h,r))};
|
||||||
|
$jscomp.polyfillUnisolated=function(m,t,h,r){h=$jscomp.global;m=m.split(".");for(r=0;r<m.length-1;r++){var v=m[r];if(!(v in h))return;h=h[v]}m=m[m.length-1];r=h[m];t=t(r);t!=r&&null!=t&&$jscomp.defineProperty(h,m,{configurable:!0,writable:!0,value:t})};
|
||||||
|
$jscomp.polyfillIsolated=function(m,t,h,r){var v=m.split(".");m=1===v.length;r=v[0];r=!m&&r in $jscomp.polyfills?$jscomp.polyfills:$jscomp.global;for(var q=0;q<v.length-1;q++){var A=v[q];if(!(A in r))return;r=r[A]}v=v[v.length-1];h=$jscomp.IS_SYMBOL_NATIVE&&"es6"===h?r[v]:null;t=t(h);null!=t&&(m?$jscomp.defineProperty($jscomp.polyfills,v,{configurable:!0,writable:!0,value:t}):t!==h&&($jscomp.propertyToPolyfillSymbol[v]=$jscomp.IS_SYMBOL_NATIVE?$jscomp.global.Symbol(v):$jscomp.POLYFILL_PREFIX+v,v=
|
||||||
|
$jscomp.propertyToPolyfillSymbol[v],$jscomp.defineProperty(r,v,{configurable:!0,writable:!0,value:t})))};$jscomp.initSymbol=function(){};
|
||||||
|
$jscomp.polyfill("Symbol",function(m){if(m)return m;var t=function(v,q){this.$jscomp$symbol$id_=v;$jscomp.defineProperty(this,"description",{configurable:!0,writable:!0,value:q})};t.prototype.toString=function(){return this.$jscomp$symbol$id_};var h=0,r=function(v){if(this instanceof r)throw new TypeError("Symbol is not a constructor");return new t("jscomp_symbol_"+(v||"")+"_"+h++,v)};return r},"es6","es3");$jscomp.initSymbolIterator=function(){};
|
||||||
|
$jscomp.polyfill("Symbol.iterator",function(m){if(m)return m;m=Symbol("Symbol.iterator");for(var t="Array Int8Array Uint8Array Uint8ClampedArray Int16Array Uint16Array Int32Array Uint32Array Float32Array Float64Array".split(" "),h=0;h<t.length;h++){var r=$jscomp.global[t[h]];"function"===typeof r&&"function"!=typeof r.prototype[m]&&$jscomp.defineProperty(r.prototype,m,{configurable:!0,writable:!0,value:function(){return $jscomp.iteratorPrototype($jscomp.arrayIteratorImpl(this))}})}return m},"es6",
|
||||||
|
"es3");$jscomp.initSymbolAsyncIterator=function(){};$jscomp.iteratorPrototype=function(m){m={next:m};m[Symbol.iterator]=function(){return this};return m};$jscomp.makeIterator=function(m){var t="undefined"!=typeof Symbol&&Symbol.iterator&&m[Symbol.iterator];return t?t.call(m):$jscomp.arrayIterator(m)};$jscomp.owns=function(m,t){return Object.prototype.hasOwnProperty.call(m,t)};
|
||||||
|
$jscomp.polyfill("WeakMap",function(m){function t(){if(!m||!Object.seal)return!1;try{var a=Object.seal({}),b=Object.seal({}),c=new m([[a,2],[b,3]]);if(2!=c.get(a)||3!=c.get(b))return!1;c.delete(a);c.set(b,4);return!c.has(a)&&4==c.get(b)}catch(d){return!1}}function h(){}function r(a){var b=typeof a;return"object"===b&&null!==a||"function"===b}function v(a){if(!$jscomp.owns(a,A)){var b=new h;$jscomp.defineProperty(a,A,{value:b})}}function q(a){if(!$jscomp.ISOLATE_POLYFILLS){var b=Object[a];b&&(Object[a]=
|
||||||
|
function(c){if(c instanceof h)return c;Object.isExtensible(c)&&v(c);return b(c)})}}if($jscomp.USE_PROXY_FOR_ES6_CONFORMANCE_CHECKS){if(m&&$jscomp.ES6_CONFORMANCE)return m}else if(t())return m;var A="$jscomp_hidden_"+Math.random();q("freeze");q("preventExtensions");q("seal");var G=0,k=function(a){this.id_=(G+=Math.random()+1).toString();if(a){a=$jscomp.makeIterator(a);for(var b;!(b=a.next()).done;)b=b.value,this.set(b[0],b[1])}};k.prototype.set=function(a,b){if(!r(a))throw Error("Invalid WeakMap key");
|
||||||
|
v(a);if(!$jscomp.owns(a,A))throw Error("WeakMap key fail: "+a);a[A][this.id_]=b;return this};k.prototype.get=function(a){return r(a)&&$jscomp.owns(a,A)?a[A][this.id_]:void 0};k.prototype.has=function(a){return r(a)&&$jscomp.owns(a,A)&&$jscomp.owns(a[A],this.id_)};k.prototype.delete=function(a){return r(a)&&$jscomp.owns(a,A)&&$jscomp.owns(a[A],this.id_)?delete a[A][this.id_]:!1};return k},"es6","es3");$jscomp.MapEntry=function(){};
|
||||||
|
$jscomp.polyfill("Map",function(m){function t(){if($jscomp.ASSUME_NO_NATIVE_MAP||!m||"function"!=typeof m||!m.prototype.entries||"function"!=typeof Object.seal)return!1;try{var k=Object.seal({x:4}),a=new m($jscomp.makeIterator([[k,"s"]]));if("s"!=a.get(k)||1!=a.size||a.get({x:4})||a.set({x:4},"t")!=a||2!=a.size)return!1;var b=a.entries(),c=b.next();if(c.done||c.value[0]!=k||"s"!=c.value[1])return!1;c=b.next();return c.done||4!=c.value[0].x||"t"!=c.value[1]||!b.next().done?!1:!0}catch(d){return!1}}
|
||||||
|
if($jscomp.USE_PROXY_FOR_ES6_CONFORMANCE_CHECKS){if(m&&$jscomp.ES6_CONFORMANCE)return m}else if(t())return m;var h=new WeakMap,r=function(k){this.data_={};this.head_=A();this.size=0;if(k){k=$jscomp.makeIterator(k);for(var a;!(a=k.next()).done;)a=a.value,this.set(a[0],a[1])}};r.prototype.set=function(k,a){k=0===k?0:k;var b=v(this,k);b.list||(b.list=this.data_[b.id]=[]);b.entry?b.entry.value=a:(b.entry={next:this.head_,previous:this.head_.previous,head:this.head_,key:k,value:a},b.list.push(b.entry),
|
||||||
|
this.head_.previous.next=b.entry,this.head_.previous=b.entry,this.size++);return this};r.prototype.delete=function(k){k=v(this,k);return k.entry&&k.list?(k.list.splice(k.index,1),k.list.length||delete this.data_[k.id],k.entry.previous.next=k.entry.next,k.entry.next.previous=k.entry.previous,k.entry.head=null,this.size--,!0):!1};r.prototype.clear=function(){this.data_={};this.head_=this.head_.previous=A();this.size=0};r.prototype.has=function(k){return!!v(this,k).entry};r.prototype.get=function(k){return(k=
|
||||||
|
v(this,k).entry)&&k.value};r.prototype.entries=function(){return q(this,function(k){return[k.key,k.value]})};r.prototype.keys=function(){return q(this,function(k){return k.key})};r.prototype.values=function(){return q(this,function(k){return k.value})};r.prototype.forEach=function(k,a){for(var b=this.entries(),c;!(c=b.next()).done;)c=c.value,k.call(a,c[1],c[0],this)};r.prototype[Symbol.iterator]=r.prototype.entries;var v=function(k,a){var b=a&&typeof a;"object"==b||"function"==b?h.has(a)?b=h.get(a):
|
||||||
|
(b=""+ ++G,h.set(a,b)):b="p_"+a;var c=k.data_[b];if(c&&$jscomp.owns(k.data_,b))for(k=0;k<c.length;k++){var d=c[k];if(a!==a&&d.key!==d.key||a===d.key)return{id:b,list:c,index:k,entry:d}}return{id:b,list:c,index:-1,entry:void 0}},q=function(k,a){var b=k.head_;return $jscomp.iteratorPrototype(function(){if(b){for(;b.head!=k.head_;)b=b.previous;for(;b.next!=b.head;)return b=b.next,{done:!1,value:a(b)};b=null}return{done:!0,value:void 0}})},A=function(){var k={};return k.previous=k.next=k.head=k},G=0;
|
||||||
|
return r},"es6","es3");$jscomp.findInternal=function(m,t,h){m instanceof String&&(m=String(m));for(var r=m.length,v=0;v<r;v++){var q=m[v];if(t.call(h,q,v,m))return{i:v,v:q}}return{i:-1,v:void 0}};$jscomp.polyfill("Array.prototype.find",function(m){return m?m:function(t,h){return $jscomp.findInternal(this,t,h).v}},"es6","es3");
|
||||||
|
$jscomp.iteratorFromArray=function(m,t){m instanceof String&&(m+="");var h=0,r={next:function(){if(h<m.length){var v=h++;return{value:t(v,m[v]),done:!1}}r.next=function(){return{done:!0,value:void 0}};return r.next()}};r[Symbol.iterator]=function(){return r};return r};$jscomp.polyfill("Array.prototype.keys",function(m){return m?m:function(){return $jscomp.iteratorFromArray(this,function(t){return t})}},"es6","es3");
|
||||||
|
$jscomp.polyfill("Array.prototype.findIndex",function(m){return m?m:function(t,h){return $jscomp.findInternal(this,t,h).i}},"es6","es3");
|
||||||
|
(function(){function m(k){h=k;r=k.fn.dataTable}function t(k){q=k;A=k.fn.dataTable}var h,r,v=function(){function k(a,b,c,d,e,g){var f=this;void 0===g&&(g=null);if(!r||!r.versionCheck||!r.versionCheck("1.10.0"))throw Error("SearchPane requires DataTables 1.10 or newer");if(!r.select)throw Error("SearchPane requires Select");a=new r.Api(a);this.classes=h.extend(!0,{},k.classes);this.c=h.extend(!0,{},k.defaults,b);this.customPaneSettings=g;this.s={cascadeRegen:!1,clearing:!1,colOpts:[],deselect:!1,displayed:!1,
|
||||||
|
dt:a,dtPane:void 0,filteringActive:!1,index:c,indexes:[],lastCascade:!1,lastSelect:!1,listSet:!1,name:void 0,redraw:!1,rowData:{arrayFilter:[],arrayOriginal:[],arrayTotals:[],bins:{},binsOriginal:{},binsTotal:{},filterMap:new Map,totalOptions:0},scrollTop:0,searchFunction:void 0,selectPresent:!1,serverSelect:[],serverSelecting:!1,showFiltered:!1,tableLength:null,updating:!1};b=a.columns().eq(0).toArray().length;this.colExists=this.s.index<b;this.c.layout=d;b=parseInt(d.split("-")[1],10);this.dom=
|
||||||
|
{buttonGroup:h("<div/>").addClass(this.classes.buttonGroup),clear:h('<button type="button">×</button>').addClass(this.classes.dull).addClass(this.classes.paneButton).addClass(this.classes.clearButton),container:h("<div/>").addClass(this.classes.container).addClass(this.classes.layout+(10>b?d:d.split("-")[0]+"-9")),countButton:h('<button type="button"></button>').addClass(this.classes.paneButton).addClass(this.classes.countButton),dtP:h("<table><thead><tr><th>"+(this.colExists?h(a.column(this.colExists?
|
||||||
|
this.s.index:0).header()).text():this.customPaneSettings.header||"Custom Pane")+"</th><th/></tr></thead></table>"),lower:h("<div/>").addClass(this.classes.subRow2).addClass(this.classes.narrowButton),nameButton:h('<button type="button"></button>').addClass(this.classes.paneButton).addClass(this.classes.nameButton),panesContainer:e,searchBox:h("<input/>").addClass(this.classes.paneInputButton).addClass(this.classes.search),searchButton:h('<button type = "button" class="'+this.classes.searchIcon+'"></button>').addClass(this.classes.paneButton),
|
||||||
|
searchCont:h("<div/>").addClass(this.classes.searchCont),searchLabelCont:h("<div/>").addClass(this.classes.searchLabelCont),topRow:h("<div/>").addClass(this.classes.topRow),upper:h("<div/>").addClass(this.classes.subRow1).addClass(this.classes.narrowSearch)};this.s.displayed=!1;a=this.s.dt;this.selections=[];this.s.colOpts=this.colExists?this._getOptions():this._getBonusOptions();var l=this.s.colOpts;d=h('<button type="button">X</button>').addClass(this.classes.paneButton);h(d).text(a.i18n("searchPanes.clearPane",
|
||||||
|
"X"));this.dom.container.addClass(l.className);this.dom.container.addClass(null!==this.customPaneSettings&&void 0!==this.customPaneSettings.className?this.customPaneSettings.className:"");this.s.name=void 0!==this.s.colOpts.name?this.s.colOpts.name:null!==this.customPaneSettings&&void 0!==this.customPaneSettings.name?this.customPaneSettings.name:this.colExists?h(a.column(this.s.index).header()).text():this.customPaneSettings.header||"Custom Pane";h(e).append(this.dom.container);var p=a.table(0).node();
|
||||||
|
this.s.searchFunction=function(n,x,z,y){if(0===f.selections.length||n.nTable!==p)return!0;n=null;f.colExists&&(n=x[f.s.index],"filter"!==l.orthogonal.filter&&(n=f.s.rowData.filterMap.get(z),n instanceof h.fn.dataTable.Api&&(n=n.toArray())));return f._search(n,z)};h.fn.dataTable.ext.search.push(this.s.searchFunction);if(this.c.clear)h(d).on("click",function(){f.dom.container.find(f.classes.search).each(function(){h(this).val("");h(this).trigger("input")});f.clearPane()});a.on("draw.dtsp",function(){f._adjustTopRow()});
|
||||||
|
a.on("buttons-action",function(){f._adjustTopRow()});h(window).on("resize.dtsp",r.util.throttle(function(){f._adjustTopRow()}));a.on("column-reorder.dtsp",function(n,x,z){f.s.index=z.mapping[f.s.index]});return this}k.prototype.clearData=function(){this.s.rowData={arrayFilter:[],arrayOriginal:[],arrayTotals:[],bins:{},binsOriginal:{},binsTotal:{},filterMap:new Map,totalOptions:0}};k.prototype.clearPane=function(){this.s.dtPane.rows({selected:!0}).deselect();this.updateTable();return this};k.prototype.destroy=
|
||||||
|
function(){h(this.s.dtPane).off(".dtsp");h(this.s.dt).off(".dtsp");h(this.dom.nameButton).off(".dtsp");h(this.dom.countButton).off(".dtsp");h(this.dom.clear).off(".dtsp");h(this.dom.searchButton).off(".dtsp");h(this.dom.container).remove();for(var a=h.fn.dataTable.ext.search.indexOf(this.s.searchFunction);-1!==a;)h.fn.dataTable.ext.search.splice(a,1),a=h.fn.dataTable.ext.search.indexOf(this.s.searchFunction);void 0!==this.s.dtPane&&this.s.dtPane.destroy();this.s.listSet=!1};k.prototype.getPaneCount=
|
||||||
|
function(){return void 0!==this.s.dtPane?this.s.dtPane.rows({selected:!0}).data().toArray().length:0};k.prototype.rebuildPane=function(a,b,c,d){void 0===a&&(a=!1);void 0===b&&(b=null);void 0===c&&(c=null);void 0===d&&(d=!1);this.clearData();var e=[];this.s.serverSelect=[];var g=null;void 0!==this.s.dtPane&&(d&&(this.s.dt.page.info().serverSide?this.s.serverSelect=this.s.dtPane.rows({selected:!0}).data().toArray():e=this.s.dtPane.rows({selected:!0}).data().toArray()),this.s.dtPane.clear().destroy(),
|
||||||
|
g=h(this.dom.container).prev(),this.destroy(),this.s.dtPane=void 0,h.fn.dataTable.ext.search.push(this.s.searchFunction));this.dom.container.removeClass(this.classes.hidden);this.s.displayed=!1;this._buildPane(this.s.dt.page.info().serverSide?this.s.serverSelect:e,a,b,c,g);return this};k.prototype.removePane=function(){this.s.displayed=!1;h(this.dom.container).hide()};k.prototype.setCascadeRegen=function(a){this.s.cascadeRegen=a};k.prototype.setClear=function(a){this.s.clearing=a};k.prototype.updatePane=
|
||||||
|
function(a){void 0===a&&(a=!1);this.s.updating=!0;this._updateCommon(a);this.s.updating=!1};k.prototype.updateTable=function(){this.selections=this.s.dtPane.rows({selected:!0}).data().toArray();this._searchExtras();(this.c.cascadePanes||this.c.viewTotal)&&this.updatePane()};k.prototype._setListeners=function(){var a=this,b=this.s.rowData,c;this.s.dtPane.on("select.dtsp",function(){clearTimeout(c);a.s.dt.page.info().serverSide&&!a.s.updating?a.s.serverSelecting||(a.s.serverSelect=a.s.dtPane.rows({selected:!0}).data().toArray(),
|
||||||
|
a.s.scrollTop=h(a.s.dtPane.table().node()).parent()[0].scrollTop,a.s.selectPresent=!0,a.s.dt.draw(!1)):(h(a.dom.clear).removeClass(a.classes.dull),a.s.selectPresent=!0,a.s.updating||a._makeSelection(),a.s.selectPresent=!1)});this.s.dtPane.on("deselect.dtsp",function(){c=setTimeout(function(){a.s.dt.page.info().serverSide&&!a.s.updating?a.s.serverSelecting||(a.s.serverSelect=a.s.dtPane.rows({selected:!0}).data().toArray(),a.s.deselect=!0,a.s.dt.draw(!1)):(a.s.deselect=!0,0===a.s.dtPane.rows({selected:!0}).data().toArray().length&&
|
||||||
|
h(a.dom.clear).addClass(a.classes.dull),a._makeSelection(),a.s.deselect=!1,a.s.dt.state.save())},50)});this.s.dt.on("stateSaveParams.dtsp",function(d,e,g){if(h.isEmptyObject(g))a.s.dtPane.state.clear();else{d=[];if(void 0!==a.s.dtPane){d=a.s.dtPane.rows({selected:!0}).data().map(function(x){return x.filter.toString()}).toArray();var f=h(a.dom.searchBox).val();var l=a.s.dtPane.order();var p=b.binsOriginal;var n=b.arrayOriginal}void 0===g.searchPanes&&(g.searchPanes={});void 0===g.searchPanes.panes&&
|
||||||
|
(g.searchPanes.panes=[]);for(e=0;e<g.searchPanes.panes.length;e++)g.searchPanes.panes[e].id===a.s.index&&(g.searchPanes.panes.splice(e,1),e--);g.searchPanes.panes.push({arrayFilter:n,bins:p,id:a.s.index,order:l,searchTerm:f,selected:d})}});this.s.dtPane.on("user-select.dtsp",function(d,e,g,f,l){l.stopPropagation()});this.s.dtPane.on("draw.dtsp",function(){a._adjustTopRow()});h(this.dom.nameButton).on("click.dtsp",function(){var d=a.s.dtPane.order()[0][1];a.s.dtPane.order([0,"asc"===d?"desc":"asc"]).draw();
|
||||||
|
a.s.dt.state.save()});h(this.dom.countButton).on("click.dtsp",function(){var d=a.s.dtPane.order()[0][1];a.s.dtPane.order([1,"asc"===d?"desc":"asc"]).draw();a.s.dt.state.save()});h(this.dom.clear).on("click.dtsp",function(){a.dom.container.find("."+a.classes.search).each(function(){h(this).val("");h(this).trigger("input")});a.clearPane()});h(this.dom.searchButton).on("click.dtsp",function(){h(a.dom.searchBox).focus()});h(this.dom.searchBox).on("input.dtsp",function(){a.s.dtPane.search(h(a.dom.searchBox).val()).draw();
|
||||||
|
a.s.dt.state.save()});this.s.dt.state.save();return!0};k.prototype._addOption=function(a,b,c,d,e,g){if(Array.isArray(a)||a instanceof r.Api)if(a instanceof r.Api&&(a=a.toArray(),b=b.toArray()),a.length===b.length)for(var f=0;f<a.length;f++)g[a[f]]?g[a[f]]++:(g[a[f]]=1,e.push({display:b[f],filter:a[f],sort:c[f],type:d[f]})),this.s.rowData.totalOptions++;else throw Error("display and filter not the same length");else"string"===typeof this.s.colOpts.orthogonal?(g[a]?g[a]++:(g[a]=1,e.push({display:b,
|
||||||
|
filter:a,sort:c,type:d})),this.s.rowData.totalOptions++):e.push({display:b,filter:a,sort:c,type:d})};k.prototype._addRow=function(a,b,c,d,e,g,f){for(var l,p=0,n=this.s.indexes;p<n.length;p++){var x=n[p];x.filter===b&&(l=x.index)}void 0===l&&(l=this.s.indexes.length,this.s.indexes.push({filter:b,index:l}));return this.s.dtPane.row.add({className:f,display:""!==a?a:!1!==this.s.colOpts.emptyMessage?this.s.colOpts.emptyMessage:this.c.emptyMessage,filter:b,index:l,shown:c,sort:""!==e?e:!1!==this.s.colOpts.emptyMessage?
|
||||||
|
this.s.colOpts.emptyMessage:this.c.emptyMessage,total:d,type:g})};k.prototype._adjustTopRow=function(){var a=this.dom.container.find("."+this.classes.subRowsContainer),b=this.dom.container.find(".dtsp-subRow1"),c=this.dom.container.find(".dtsp-subRow2"),d=this.dom.container.find("."+this.classes.topRow);(252>h(a[0]).width()||252>h(d[0]).width())&&0!==h(a[0]).width()?(h(a[0]).addClass(this.classes.narrow),h(b[0]).addClass(this.classes.narrowSub).removeClass(this.classes.narrowSearch),h(c[0]).addClass(this.classes.narrowSub).removeClass(this.classes.narrowButton)):
|
||||||
|
(h(a[0]).removeClass(this.classes.narrow),h(b[0]).removeClass(this.classes.narrowSub).addClass(this.classes.narrowSearch),h(c[0]).removeClass(this.classes.narrowSub).addClass(this.classes.narrowButton))};k.prototype._buildPane=function(a,b,c,d,e){var g=this;void 0===a&&(a=[]);void 0===b&&(b=!1);void 0===c&&(c=null);void 0===d&&(d=null);void 0===e&&(e=null);this.selections=[];var f=this.s.dt,l=f.column(this.colExists?this.s.index:0),p=this.s.colOpts,n=this.s.rowData,x=f.i18n("searchPanes.count","{total}"),
|
||||||
|
z=f.i18n("searchPanes.countFiltered","{shown} ({total})"),y=f.state.loaded();this.s.listSet&&(y=f.state());if(this.colExists){var w=-1;if(y&&y.searchPanes&&y.searchPanes.panes)for(var u=0;u<y.searchPanes.panes.length;u++)if(y.searchPanes.panes[u].id===this.s.index){w=u;break}if((!1===p.show||void 0!==p.show&&!0!==p.show)&&-1===w)return this.dom.container.addClass(this.classes.hidden),this.s.displayed=!1;if(!0===p.show||-1!==w)this.s.displayed=!0;if(!this.s.dt.page.info().serverSide&&(null===c||null===
|
||||||
|
c.searchPanes||null===c.searchPanes.options)){if(0===n.arrayFilter.length){this._populatePane(b);this.s.rowData.totalOptions=0;this._detailsPane();if(y&&y.searchPanes&&y.searchPanes.panes&&-1===w){this.dom.container.addClass(this.classes.hidden);this.s.displayed=!1;return}n.arrayOriginal=n.arrayTotals;n.binsOriginal=n.binsTotal}u=Object.keys(n.binsOriginal).length;b=this._uniqueRatio(u,f.rows()[0].length);if(!1===this.s.displayed&&((void 0===p.show&&null===p.threshold?b>this.c.threshold:b>p.threshold)||
|
||||||
|
!0!==p.show&&1>=u)){this.dom.container.addClass(this.classes.hidden);this.s.displayed=!1;return}this.c.viewTotal&&0===n.arrayTotals.length?(this.s.rowData.totalOptions=0,this._detailsPane()):n.binsTotal=n.bins;this.dom.container.addClass(this.classes.show);this.s.displayed=!0}else if(null!==c&&null!==c.searchPanes&&null!==c.searchPanes.options){if(void 0!==c.tableLength)this.s.tableLength=c.tableLength,this.s.rowData.totalOptions=this.s.tableLength;else if(null===this.s.tableLength||f.rows()[0].length>
|
||||||
|
this.s.tableLength)this.s.tableLength=f.rows()[0].length,this.s.rowData.totalOptions=this.s.tableLength;b=f.column(this.s.index).dataSrc();if(void 0!==c.searchPanes.options[b])for(u=0,b=c.searchPanes.options[b];u<b.length;u++)w=b[u],this.s.rowData.arrayFilter.push({display:w.label,filter:w.value,sort:w.label,type:w.label}),this.s.rowData.bins[w.value]=this.c.viewTotal||this.c.cascadePanes?w.count:w.total,this.s.rowData.binsTotal[w.value]=w.total;u=Object.keys(n.binsTotal).length;b=this._uniqueRatio(u,
|
||||||
|
this.s.tableLength);if(!1===this.s.displayed&&((void 0===p.show&&null===p.threshold?b>this.c.threshold:b>p.threshold)||!0!==p.show&&1>=u)){this.dom.container.addClass(this.classes.hidden);this.s.displayed=!1;return}this.s.rowData.arrayOriginal=this.s.rowData.arrayFilter;this.s.rowData.binsOriginal=this.s.rowData.bins;this.s.displayed=!0}}else this.s.displayed=!0;this._displayPane();if(!this.s.listSet)this.dom.dtP.on("stateLoadParams.dt",function(E,F,D){h.isEmptyObject(f.state.loaded())&&h.each(D,
|
||||||
|
function(C,I){delete D[C]})});null!==e&&0<h(this.dom.panesContainer).has(e).length?h(this.dom.container).insertAfter(e):h(this.dom.panesContainer).prepend(this.dom.container);u=h.fn.dataTable.ext.errMode;h.fn.dataTable.ext.errMode="none";e=r.Scroller;this.s.dtPane=h(this.dom.dtP).DataTable(h.extend(!0,{columnDefs:[{className:"dtsp-nameColumn",data:"display",render:function(E,F,D){if("sort"===F)return D.sort;if("type"===F)return D.type;var C;(g.s.filteringActive||g.s.showFiltered)&&g.c.viewTotal?C=
|
||||||
|
z.replace(/{total}/,D.total):C=x.replace(/{total}/,D.total);for(C=C.replace(/{shown}/,D.shown);-1!==C.indexOf("{total}");)C=C.replace(/{total}/,D.total);for(;-1!==C.indexOf("{shown}");)C=C.replace(/{shown}/,D.shown);F='<span class="'+g.classes.pill+'">'+C+"</span>";if(g.c.hideCount||p.hideCount)F="";return'<div class="'+g.classes.nameCont+'"><span title="'+("string"===typeof E&&null!==E.match(/<[^>]*>/)?E.replace(/<[^>]*>/g,""):E)+'" class="'+g.classes.name+'">'+E+"</span>"+F+"</div>"},targets:0,
|
||||||
|
type:void 0!==f.settings()[0].aoColumns[this.s.index]?f.settings()[0].aoColumns[this.s.index]._sManualType:null},{className:"dtsp-countColumn "+this.classes.badgePill,data:"shown",orderData:[1,2],targets:1,visible:!1},{data:"total",targets:2,visible:!1}],deferRender:!0,dom:"t",info:!1,language:this.s.dt.settings()[0].oLanguage,paging:e?!0:!1,scrollX:!1,scrollY:"200px",scroller:e?!0:!1,select:!0,stateSave:f.settings()[0].oFeatures.bStateSave?!0:!1},this.c.dtOpts,void 0!==p?p.dtOpts:{},void 0===this.s.colOpts.options&&
|
||||||
|
this.colExists?void 0:{createdRow:function(E,F,D){h(E).addClass(F.className)}},null!==this.customPaneSettings&&void 0!==this.customPaneSettings.dtOpts?this.customPaneSettings.dtOpts:{}));h(this.dom.dtP).addClass(this.classes.table);h(this.dom.searchBox).attr("placeholder",void 0!==p.header?p.header:this.colExists?f.settings()[0].aoColumns[this.s.index].sTitle:this.customPaneSettings.header||"Custom Pane");h.fn.dataTable.select.init(this.s.dtPane);h.fn.dataTable.ext.errMode=u;if(this.colExists){l=
|
||||||
|
(l=l.search())?l.substr(1,l.length-2).split("|"):[];var B=0;n.arrayFilter.forEach(function(E){""===E.filter&&B++});u=0;for(e=n.arrayFilter.length;u<e;u++){l=!1;w=0;for(var H=this.s.serverSelect;w<H.length;w++)b=H[w],b.filter===n.arrayFilter[u].filter&&(l=!0);if(this.s.dt.page.info().serverSide&&(!this.c.cascadePanes||this.c.cascadePanes&&0!==n.bins[n.arrayFilter[u].filter]||this.c.cascadePanes&&null!==d||l))for(l=this._addRow(n.arrayFilter[u].display,n.arrayFilter[u].filter,d?n.binsTotal[n.arrayFilter[u].filter]:
|
||||||
|
n.bins[n.arrayFilter[u].filter],this.c.viewTotal||d?String(n.binsTotal[n.arrayFilter[u].filter]):n.bins[n.arrayFilter[u].filter],n.arrayFilter[u].sort,n.arrayFilter[u].type),w=0,H=this.s.serverSelect;w<H.length;w++)b=H[w],b.filter===n.arrayFilter[u].filter&&(this.s.serverSelecting=!0,l.select(),this.s.serverSelecting=!1);else this.s.dt.page.info().serverSide||!n.arrayFilter[u]||void 0===n.bins[n.arrayFilter[u].filter]&&this.c.cascadePanes?this.s.dt.page.info().serverSide||this._addRow("",B,B,"","",
|
||||||
|
""):this._addRow(n.arrayFilter[u].display,n.arrayFilter[u].filter,n.bins[n.arrayFilter[u].filter],n.binsTotal[n.arrayFilter[u].filter],n.arrayFilter[u].sort,n.arrayFilter[u].type)}}r.select.init(this.s.dtPane);(void 0!==p.options||null!==this.customPaneSettings&&void 0!==this.customPaneSettings.options)&&this._getComparisonRows();this.s.dtPane.draw();this._adjustTopRow();this.s.listSet||(this._setListeners(),this.s.listSet=!0);for(d=0;d<a.length;d++)if(n=a[d],void 0!==n)for(u=0,e=this.s.dtPane.rows().indexes().toArray();u<
|
||||||
|
e.length;u++)l=e[u],void 0!==this.s.dtPane.row(l).data()&&n.filter===this.s.dtPane.row(l).data().filter&&(this.s.dt.page.info().serverSide?(this.s.serverSelecting=!0,this.s.dtPane.row(l).select(),this.s.serverSelecting=!1):this.s.dtPane.row(l).select());this.s.dt.page.info().serverSide&&this.s.dtPane.search(h(this.dom.searchBox).val()).draw();if(y&&y.searchPanes&&y.searchPanes.panes&&(null===c||1===c.draw))for(this.c.cascadePanes||this._reloadSelect(y),c=0,y=y.searchPanes.panes;c<y.length;c++)a=y[c],
|
||||||
|
a.id===this.s.index&&(h(this.dom.searchBox).val(a.searchTerm),h(this.dom.searchBox).trigger("input"),this.s.dtPane.order(a.order).draw());this.s.dt.state.save();return!0};k.prototype._detailsPane=function(){var a=this.s.dt;this.s.rowData.arrayTotals=[];this.s.rowData.binsTotal={};var b=this.s.dt.settings()[0];a=a.rows().indexes();if(!this.s.dt.page.info().serverSide)for(var c=0;c<a.length;c++)this._populatePaneArray(a[c],this.s.rowData.arrayTotals,b,this.s.rowData.binsTotal)};k.prototype._displayPane=
|
||||||
|
function(){var a=this.dom.container,b=this.s.colOpts,c=parseInt(this.c.layout.split("-")[1],10);h(this.dom.topRow).empty();h(this.dom.dtP).empty();h(this.dom.topRow).addClass(this.classes.topRow);3<c&&h(this.dom.container).addClass(this.classes.smallGap);h(this.dom.topRow).addClass(this.classes.subRowsContainer);h(this.dom.upper).appendTo(this.dom.topRow);h(this.dom.lower).appendTo(this.dom.topRow);h(this.dom.searchCont).appendTo(this.dom.upper);h(this.dom.buttonGroup).appendTo(this.dom.lower);(!1===
|
||||||
|
this.c.dtOpts.searching||void 0!==b.dtOpts&&!1===b.dtOpts.searching||!this.c.controls||!b.controls||null!==this.customPaneSettings&&void 0!==this.customPaneSettings.dtOpts&&void 0!==this.customPaneSettings.dtOpts.searching&&!this.customPaneSettings.dtOpts.searching)&&h(this.dom.searchBox).attr("disabled","disabled").removeClass(this.classes.paneInputButton).addClass(this.classes.disabledButton);h(this.dom.searchBox).appendTo(this.dom.searchCont);this._searchContSetup();this.c.clear&&this.c.controls&&
|
||||||
|
b.controls&&h(this.dom.clear).appendTo(this.dom.buttonGroup);this.c.orderable&&b.orderable&&this.c.controls&&b.controls&&h(this.dom.nameButton).appendTo(this.dom.buttonGroup);!this.c.hideCount&&!b.hideCount&&this.c.orderable&&b.orderable&&this.c.controls&&b.controls&&h(this.dom.countButton).appendTo(this.dom.buttonGroup);h(this.dom.topRow).prependTo(this.dom.container);h(a).append(this.dom.dtP);h(a).show()};k.prototype._getBonusOptions=function(){return h.extend(!0,{},k.defaults,{orthogonal:{threshold:null},
|
||||||
|
threshold:null},void 0!==this.c?this.c:{})};k.prototype._getComparisonRows=function(){var a=this.s.colOpts;a=void 0!==a.options?a.options:null!==this.customPaneSettings&&void 0!==this.customPaneSettings.options?this.customPaneSettings.options:void 0;if(void 0!==a){var b=this.s.dt.rows({search:"applied"}).data().toArray(),c=this.s.dt.rows({search:"applied"}),d=this.s.dt.rows().data().toArray(),e=this.s.dt.rows(),g=[];this.s.dtPane.clear();for(var f=0;f<a.length;f++){var l=a[f],p=""!==l.label?l.label:
|
||||||
|
this.c.emptyMessage,n=l.className,x=p,z="function"===typeof l.value?l.value:[],y=0,w=p,u=0;if("function"===typeof l.value){for(var B=0;B<b.length;B++)l.value.call(this.s.dt,b[B],c[0][B])&&y++;for(B=0;B<d.length;B++)l.value.call(this.s.dt,d[B],e[0][B])&&u++;"function"!==typeof z&&z.push(l.filter)}(!this.c.cascadePanes||this.c.cascadePanes&&0!==y)&&g.push(this._addRow(x,z,y,u,w,p,n))}return g}};k.prototype._getOptions=function(){return h.extend(!0,{},k.defaults,{emptyMessage:!1,orthogonal:{threshold:null},
|
||||||
|
threshold:null},this.s.dt.settings()[0].aoColumns[this.s.index].searchPanes)};k.prototype._makeSelection=function(){this.updateTable();this.s.updating=!0;this.s.dt.draw();this.s.updating=!1};k.prototype._populatePane=function(a){void 0===a&&(a=!1);var b=this.s.dt;this.s.rowData.arrayFilter=[];this.s.rowData.bins={};var c=this.s.dt.settings()[0];if(!this.s.dt.page.info().serverSide){var d=0;for(a=(!this.c.cascadePanes&&!this.c.viewTotal||this.s.clearing||a?b.rows().indexes():b.rows({search:"applied"}).indexes()).toArray();d<
|
||||||
|
a.length;d++)this._populatePaneArray(a[d],this.s.rowData.arrayFilter,c)}};k.prototype._populatePaneArray=function(a,b,c,d){void 0===d&&(d=this.s.rowData.bins);var e=this.s.colOpts;if("string"===typeof e.orthogonal)c=c.oApi._fnGetCellData(c,a,this.s.index,e.orthogonal),this.s.rowData.filterMap.set(a,c),this._addOption(c,c,c,c,b,d);else{var g=c.oApi._fnGetCellData(c,a,this.s.index,e.orthogonal.search);null===g&&(g="");"string"===typeof g&&(g=g.replace(/<[^>]*>/g,""));this.s.rowData.filterMap.set(a,
|
||||||
|
g);d[g]?d[g]++:(d[g]=1,this._addOption(g,c.oApi._fnGetCellData(c,a,this.s.index,e.orthogonal.display),c.oApi._fnGetCellData(c,a,this.s.index,e.orthogonal.sort),c.oApi._fnGetCellData(c,a,this.s.index,e.orthogonal.type),b,d));this.s.rowData.totalOptions++}};k.prototype._reloadSelect=function(a){if(void 0!==a){for(var b,c=0;c<a.searchPanes.panes.length;c++)if(a.searchPanes.panes[c].id===this.s.index){b=c;break}if(void 0!==b){c=this.s.dtPane;var d=c.rows({order:"index"}).data().map(function(f){return null!==
|
||||||
|
f.filter?f.filter.toString():null}).toArray(),e=0;for(a=a.searchPanes.panes[b].selected;e<a.length;e++){b=a[e];var g=-1;null!==b&&(g=d.indexOf(b.toString()));-1<g&&(this.s.serverSelecting=!0,c.row(g).select(),this.s.serverSelecting=!1)}}}};k.prototype._search=function(a,b){for(var c=this.s.colOpts,d=this.s.dt,e=0,g=this.selections;e<g.length;e++){var f=g[e];"string"===typeof f.filter&&(f.filter=f.filter.replaceAll("&","&"));if(Array.isArray(a)){if(-1!==a.indexOf(f.filter))return!0}else if("function"===
|
||||||
|
typeof f.filter)if(f.filter.call(d,d.row(b).data(),b)){if("or"===c.combiner)return!0}else{if("and"===c.combiner)return!1}else if(a===f.filter||("string"!==typeof a||0!==a.length)&&a==f.filter||null===f.filter&&"string"===typeof a&&""===a)return!0}return"and"===c.combiner?!0:!1};k.prototype._searchContSetup=function(){this.c.controls&&this.s.colOpts.controls&&h(this.dom.searchButton).appendTo(this.dom.searchLabelCont);!1===this.c.dtOpts.searching||!1===this.s.colOpts.dtOpts.searching||null!==this.customPaneSettings&&
|
||||||
|
void 0!==this.customPaneSettings.dtOpts&&void 0!==this.customPaneSettings.dtOpts.searching&&!this.customPaneSettings.dtOpts.searching||h(this.dom.searchLabelCont).appendTo(this.dom.searchCont)};k.prototype._searchExtras=function(){var a=this.s.updating;this.s.updating=!0;var b=this.s.dtPane.rows({selected:!0}).data().pluck("filter").toArray(),c=b.indexOf(!1!==this.s.colOpts.emptyMessage?this.s.colOpts.emptyMessage:this.c.emptyMessage),d=h(this.s.dtPane.table().container());-1<c&&(b[c]="");0<b.length?
|
||||||
|
d.addClass(this.classes.selected):0===b.length&&d.removeClass(this.classes.selected);this.s.updating=a};k.prototype._uniqueRatio=function(a,b){return 0<b&&(0<this.s.rowData.totalOptions&&!this.s.dt.page.info().serverSide||this.s.dt.page.info().serverSide&&0<this.s.tableLength)?a/this.s.rowData.totalOptions:1};k.prototype._updateCommon=function(a){void 0===a&&(a=!1);if(!(this.s.dt.page.info().serverSide||void 0===this.s.dtPane||this.s.filteringActive&&!this.c.cascadePanes&&!0!==a||!0===this.c.cascadePanes&&
|
||||||
|
!0===this.s.selectPresent||this.s.lastSelect&&this.s.lastCascade)){var b=this.s.colOpts,c=this.s.dtPane.rows({selected:!0}).data().toArray();a=h(this.s.dtPane.table().node()).parent()[0].scrollTop;var d=this.s.rowData;this.s.dtPane.clear();if(this.colExists){0===d.arrayFilter.length?this._populatePane():this.c.cascadePanes&&this.s.dt.rows().data().toArray().length===this.s.dt.rows({search:"applied"}).data().toArray().length?(d.arrayFilter=d.arrayOriginal,d.bins=d.binsOriginal):(this.c.viewTotal||
|
||||||
|
this.c.cascadePanes)&&this._populatePane();this.c.viewTotal?this._detailsPane():d.binsTotal=d.bins;this.c.viewTotal&&!this.c.cascadePanes&&(d.arrayFilter=d.arrayTotals);for(var e=function(p){if(p&&(void 0!==d.bins[p.filter]&&0!==d.bins[p.filter]&&g.c.cascadePanes||!g.c.cascadePanes||g.s.clearing)){var n=g._addRow(p.display,p.filter,g.c.viewTotal?void 0!==d.bins[p.filter]?d.bins[p.filter]:0:d.bins[p.filter],g.c.viewTotal?String(d.binsTotal[p.filter]):d.bins[p.filter],p.sort,p.type),x=c.findIndex(function(z){return z.filter===
|
||||||
|
p.filter});-1!==x&&(n.select(),c.splice(x,1))}},g=this,f=0,l=d.arrayFilter;f<l.length;f++)e(l[f])}if(void 0!==b.searchPanes&&void 0!==b.searchPanes.options||void 0!==b.options||null!==this.customPaneSettings&&void 0!==this.customPaneSettings.options)for(e=function(p){var n=c.findIndex(function(x){if(x.display===p.data().display)return!0});-1!==n&&(p.select(),c.splice(n,1))},f=0,l=this._getComparisonRows();f<l.length;f++)b=l[f],e(b);for(e=0;e<c.length;e++)b=c[e],b=this._addRow(b.display,b.filter,0,
|
||||||
|
this.c.viewTotal?b.total:0,b.display,b.display),this.s.updating=!0,b.select(),this.s.updating=!1;this.s.dtPane.draw();this.s.dtPane.table().node().parentNode.scrollTop=a}};k.version="1.1.0";k.classes={buttonGroup:"dtsp-buttonGroup",buttonSub:"dtsp-buttonSub",clear:"dtsp-clear",clearAll:"dtsp-clearAll",clearButton:"clearButton",container:"dtsp-searchPane",countButton:"dtsp-countButton",disabledButton:"dtsp-disabledButton",dull:"dtsp-dull",hidden:"dtsp-hidden",hide:"dtsp-hide",layout:"dtsp-",name:"dtsp-name",
|
||||||
|
nameButton:"dtsp-nameButton",nameCont:"dtsp-nameCont",narrow:"dtsp-narrow",paneButton:"dtsp-paneButton",paneInputButton:"dtsp-paneInputButton",pill:"dtsp-pill",search:"dtsp-search",searchCont:"dtsp-searchCont",searchIcon:"dtsp-searchIcon",searchLabelCont:"dtsp-searchButtonCont",selected:"dtsp-selected",smallGap:"dtsp-smallGap",subRow1:"dtsp-subRow1",subRow2:"dtsp-subRow2",subRowsContainer:"dtsp-subRowsContainer",title:"dtsp-title",topRow:"dtsp-topRow"};k.defaults={cascadePanes:!1,clear:!0,combiner:"or",
|
||||||
|
controls:!0,container:function(a){return a.table().container()},dtOpts:{},emptyMessage:"<i>No Data</i>",hideCount:!1,layout:"columns-3",name:void 0,orderable:!0,orthogonal:{display:"display",filter:"filter",hideCount:!1,search:"filter",show:void 0,sort:"sort",threshold:.6,type:"type"},preSelect:[],threshold:.6,viewTotal:!1};return k}(),q,A,G=function(){function k(a,b,c){var d=this;void 0===c&&(c=!1);this.regenerating=!1;if(!A||!A.versionCheck||!A.versionCheck("1.10.0"))throw Error("SearchPane requires DataTables 1.10 or newer");
|
||||||
|
if(!A.select)throw Error("SearchPane requires Select");var e=new A.Api(a);this.classes=q.extend(!0,{},k.classes);this.c=q.extend(!0,{},k.defaults,b);this.dom={clearAll:q('<button type="button">Clear All</button>').addClass(this.classes.clearAll),container:q("<div/>").addClass(this.classes.panes).text(e.i18n("searchPanes.loadMessage","Loading Search Panes...")),emptyMessage:q("<div/>").addClass(this.classes.emptyMessage),options:q("<div/>").addClass(this.classes.container),panes:q("<div/>").addClass(this.classes.container),
|
||||||
|
title:q("<div/>").addClass(this.classes.title),titleRow:q("<div/>").addClass(this.classes.titleRow),wrapper:q("<div/>")};this.s={colOpts:[],dt:e,filterCount:0,filterPane:-1,page:0,panes:[],selectionList:[],serverData:{},stateRead:!1,updating:!1};if(void 0===e.settings()[0]._searchPanes){this._getState();if(this.s.dt.page.info().serverSide)e.on("preXhr.dt",function(g,f,l){void 0===l.searchPanes&&(l.searchPanes={});g=0;for(f=d.s.selectionList;g<f.length;g++){var p=f[g],n=d.s.dt.column(p.index).dataSrc();
|
||||||
|
void 0===l.searchPanes[n]&&(l.searchPanes[n]={});for(var x=0;x<p.rows.length;x++)l.searchPanes[n][x]=p.rows[x].filter}});e.on("xhr",function(g,f,l,p){l&&l.searchPanes&&l.searchPanes.options&&(d.s.serverData=l,d.s.serverData.tableLength=l.recordsTotal,d._serverTotals())});e.settings()[0]._searchPanes=this;this.dom.clearAll.text(e.i18n("searchPanes.clearMessage","Clear All"));if(this.s.dt.settings()[0]._bInitComplete||c)this._paneDeclare(e,a,b);else e.one("preInit.dt",function(g){d._paneDeclare(e,a,
|
||||||
|
b)});return this}}k.prototype.clearSelections=function(){this.dom.container.find(this.classes.search).each(function(){q(this).val("");q(this).trigger("input")});for(var a=[],b=0,c=this.s.panes;b<c.length;b++){var d=c[b];void 0!==d.s.dtPane&&a.push(d.clearPane())}this.s.dt.draw();return a};k.prototype.getNode=function(){return this.dom.container};k.prototype.rebuild=function(a,b){void 0===a&&(a=!1);void 0===b&&(b=!1);q(this.dom.emptyMessage).remove();var c=[];!1===a&&q(this.dom.panes).empty();for(var d=
|
||||||
|
0,e=this.s.panes;d<e.length;d++){var g=e[d];if(!1===a||g.s.index===a)g.clearData(),c.push(g.rebuildPane(void 0!==this.s.selectionList[this.s.selectionList.length-1]?g.s.index===this.s.selectionList[this.s.selectionList.length-1].index:!1,this.s.dt.page.info().serverSide?this.s.serverData:void 0,null,b)),q(this.dom.panes).append(g.dom.container)}this.s.dt.page.info().serverSide||this.s.dt.draw();this.c.cascadePanes||this.c.viewTotal?this.redrawPanes(!0):this._updateSelection();this._updateFilterCount();
|
||||||
|
this._attachPaneContainer();this.s.dt.draw();return 1===c.length?c[0]:c};k.prototype.redrawPanes=function(a){void 0===a&&(a=!1);var b=this.s.dt;if(!this.s.updating&&!this.s.dt.page.info().serverSide){var c=!0,d=this.s.filterPane;if(b.rows({search:"applied"}).data().toArray().length===b.rows().data().toArray().length)c=!1;else if(this.c.viewTotal)for(var e=0,g=this.s.panes;e<g.length;e++){var f=g[e];if(void 0!==f.s.dtPane){var l=f.s.dtPane.rows({selected:!0}).data().toArray().length;if(0===l)for(var p=
|
||||||
|
0,n=this.s.selectionList;p<n.length;p++){var x=n[p];x.index===f.s.index&&0!==x.rows.length&&(l=x.rows.length)}0<l&&-1===d?d=f.s.index:0<l&&(d=null)}}g=void 0;e=[];if(this.regenerating){g=-1;1===e.length&&(g=e[0].index);a=0;for(e=this.s.panes;a<e.length;a++)if(f=e[a],void 0!==f.s.dtPane){b=!0;f.s.filteringActive=!0;if(-1!==d&&null!==d&&d===f.s.index||!1===c||f.s.index===g)b=!1,f.s.filteringActive=!1;f.updatePane(b?c:b)}this._updateFilterCount()}else{l=0;for(p=this.s.panes;l<p.length;l++)if(f=p[l],
|
||||||
|
f.s.selectPresent){this.s.selectionList.push({index:f.s.index,rows:f.s.dtPane.rows({selected:!0}).data().toArray(),protect:!1});b.state.save();break}else f.s.deselect&&(g=f.s.index,n=f.s.dtPane.rows({selected:!0}).data().toArray(),0<n.length&&this.s.selectionList.push({index:f.s.index,rows:n,protect:!0}));if(0<this.s.selectionList.length)for(b=this.s.selectionList[this.s.selectionList.length-1].index,l=0,p=this.s.panes;l<p.length;l++)f=p[l],f.s.lastSelect=f.s.index===b;for(f=0;f<this.s.selectionList.length;f++)if(this.s.selectionList[f].index!==
|
||||||
|
g||!0===this.s.selectionList[f].protect){b=!1;for(l=f+1;l<this.s.selectionList.length;l++)this.s.selectionList[l].index===this.s.selectionList[f].index&&(b=!0);b||(e.push(this.s.selectionList[f]),this.s.selectionList[f].protect=!1)}g=-1;1===e.length&&(g=e[0].index);l=0;for(p=this.s.panes;l<p.length;l++)if(f=p[l],void 0!==f.s.dtPane){b=!0;f.s.filteringActive=!0;if(-1!==d&&null!==d&&d===f.s.index||!1===c||f.s.index===g)b=!1,f.s.filteringActive=!1;f.updatePane(b?c:!1)}this._updateFilterCount();if(0<
|
||||||
|
e.length&&(e.length<this.s.selectionList.length||a))for(this._cascadeRegen(e),b=e[e.length-1].index,d=0,a=this.s.panes;d<a.length;d++)f=a[d],f.s.lastSelect=f.s.index===b;else if(0<e.length)for(f=0,a=this.s.panes;f<a.length;f++)if(e=a[f],void 0!==e.s.dtPane){b=!0;e.s.filteringActive=!0;if(-1!==d&&null!==d&&d===e.s.index||!1===c)b=!1,e.s.filteringActive=!1;e.updatePane(b?c:b)}}c||(this.s.selectionList=[])}};k.prototype._attach=function(){var a=this;q(this.dom.container).removeClass(this.classes.hide);
|
||||||
|
q(this.dom.titleRow).removeClass(this.classes.hide);q(this.dom.titleRow).remove();q(this.dom.title).appendTo(this.dom.titleRow);this.c.clear&&(q(this.dom.clearAll).appendTo(this.dom.titleRow),q(this.dom.clearAll).on("click.dtsps",function(){a.clearSelections()}));q(this.dom.titleRow).appendTo(this.dom.container);for(var b=0,c=this.s.panes;b<c.length;b++)q(c[b].dom.container).appendTo(this.dom.panes);q(this.dom.panes).appendTo(this.dom.container);0===q("div."+this.classes.container).length&&q(this.dom.container).prependTo(this.s.dt);
|
||||||
|
return this.dom.container};k.prototype._attachExtras=function(){q(this.dom.container).removeClass(this.classes.hide);q(this.dom.titleRow).removeClass(this.classes.hide);q(this.dom.titleRow).remove();q(this.dom.title).appendTo(this.dom.titleRow);this.c.clear&&q(this.dom.clearAll).appendTo(this.dom.titleRow);q(this.dom.titleRow).appendTo(this.dom.container);return this.dom.container};k.prototype._attachMessage=function(){try{var a=this.s.dt.i18n("searchPanes.emptyPanes","No SearchPanes")}catch(b){a=
|
||||||
|
null}if(null===a)q(this.dom.container).addClass(this.classes.hide),q(this.dom.titleRow).removeClass(this.classes.hide);else return q(this.dom.container).removeClass(this.classes.hide),q(this.dom.titleRow).addClass(this.classes.hide),q(this.dom.emptyMessage).text(a),this.dom.emptyMessage.appendTo(this.dom.container),this.dom.container};k.prototype._attachPaneContainer=function(){for(var a=0,b=this.s.panes;a<b.length;a++)if(!0===b[a].s.displayed)return this._attach();return this._attachMessage()};k.prototype._cascadeRegen=
|
||||||
|
function(a){this.regenerating=!0;var b=-1;1===a.length&&(b=a[0].index);for(var c=0,d=this.s.panes;c<d.length;c++){var e=d[c];e.setCascadeRegen(!0);e.setClear(!0);(void 0!==e.s.dtPane&&e.s.index===b||void 0!==e.s.dtPane)&&e.clearPane();e.setClear(!1)}this._makeCascadeSelections(a);this.s.selectionList=a;a=0;for(b=this.s.panes;a<b.length;a++)e=b[a],e.setCascadeRegen(!1);this.regenerating=!1};k.prototype._checkMessage=function(){for(var a=0,b=this.s.panes;a<b.length;a++)if(!0===b[a].s.displayed)return;
|
||||||
|
return this._attachMessage()};k.prototype._getState=function(){var a=this.s.dt.state.loaded();a&&a.searchPanes&&void 0!==a.searchPanes.selectionList&&(this.s.selectionList=a.searchPanes.selectionList)};k.prototype._makeCascadeSelections=function(a){for(var b=0;b<a.length;b++)for(var c=function(f){if(f.s.index===a[b].index&&void 0!==f.s.dtPane){b===a.length-1&&(f.s.lastCascade=!0);0<f.s.dtPane.rows({selected:!0}).data().toArray().length&&void 0!==f.s.dtPane&&(f.setClear(!0),f.clearPane(),f.setClear(!1));
|
||||||
|
for(var l=function(x){f.s.dtPane.rows().every(function(z){void 0!==f.s.dtPane.row(z).data()&&void 0!==x&&f.s.dtPane.row(z).data().filter===x.filter&&f.s.dtPane.row(z).select()})},p=0,n=a[b].rows;p<n.length;p++)l(n[p]);d._updateFilterCount();f.s.lastCascade=!1}},d=this,e=0,g=this.s.panes;e<g.length;e++)c(g[e]);this.s.dt.state.save()};k.prototype._paneDeclare=function(a,b,c){var d=this;a.columns(0<this.c.columns.length?this.c.columns:void 0).eq(0).each(function(l){d.s.panes.push(new v(b,c,l,d.c.layout,
|
||||||
|
d.dom.panes))});for(var e=a.columns().eq(0).toArray().length,g=this.c.panes.length,f=0;f<g;f++)this.s.panes.push(new v(b,c,e+f,this.c.layout,this.dom.panes,this.c.panes[f]));if(0<this.c.order.length)for(e=this.c.order.map(function(l,p,n){return d._findPane(l)}),this.dom.panes.empty(),this.s.panes=e,e=0,g=this.s.panes;e<g.length;e++)this.dom.panes.append(g[e].dom.container);this.s.dt.settings()[0]._bInitComplete?this._startup(a):this.s.dt.settings()[0].aoInitComplete.push({fn:function(){d._startup(a)}})};
|
||||||
|
k.prototype._findPane=function(a){for(var b=0,c=this.s.panes;b<c.length;b++){var d=c[b];if(a===d.s.name)return d}};k.prototype._serverTotals=function(){for(var a=!1,b=!1,c=this.s.dt,d=0,e=this.s.panes;d<e.length;d++){var g=e[d];if(g.s.selectPresent){this.s.selectionList.push({index:g.s.index,rows:g.s.dtPane.rows({selected:!0}).data().toArray(),protect:!1});c.state.save();g.s.selectPresent=!1;a=!0;break}else g.s.deselect&&(b=g.s.dtPane.rows({selected:!0}).data().toArray(),0<b.length&&this.s.selectionList.push({index:g.s.index,
|
||||||
|
rows:b,protect:!0}),b=a=!0)}if(a){c=[];for(d=0;d<this.s.selectionList.length;d++){g=!1;for(e=d+1;e<this.s.selectionList.length;e++)this.s.selectionList[e].index===this.s.selectionList[d].index&&(g=!0);if(!g){e=!1;a=0;for(var f=this.s.panes;a<f.length;a++)g=f[a],g.s.index===this.s.selectionList[d].index&&0<g.s.dtPane.rows({selected:!0}).data().toArray().length&&(e=!0);e&&c.push(this.s.selectionList[d])}}this.s.selectionList=c}else this.s.selectionList=[];c=-1;if(b&&1===this.s.selectionList.length)for(b=
|
||||||
|
0,d=this.s.panes;b<d.length;b++)g=d[b],g.s.lastSelect=!1,g.s.deselect=!1,void 0!==g.s.dtPane&&0<g.s.dtPane.rows({selected:!0}).data().toArray().length&&(c=g.s.index);else if(0<this.s.selectionList.length)for(b=this.s.selectionList[this.s.selectionList.length-1].index,d=0,e=this.s.panes;d<e.length;d++)g=e[d],g.s.lastSelect=g.s.index===b,g.s.deselect=!1;else if(0===this.s.selectionList.length)for(b=0,d=this.s.panes;b<d.length;b++)g=d[b],g.s.lastSelect=!1,g.s.deselect=!1;q(this.dom.panes).empty();b=
|
||||||
|
0;for(d=this.s.panes;b<d.length;b++)g=d[b],g.s.lastSelect?g._setListeners():g.rebuildPane(void 0,this.s.dt.page.info().serverSide?this.s.serverData:void 0,g.s.index===c?!0:null,!0),q(this.dom.panes).append(g.dom.container),void 0!==g.s.dtPane&&(q(g.s.dtPane.table().node()).parent()[0].scrollTop=g.s.scrollTop,q.fn.dataTable.select.init(g.s.dtPane));this.s.dt.page.info().serverSide||this.s.dt.draw()};k.prototype._startup=function(a){var b=this;q(this.dom.container).text("");this._attachExtras();q(this.dom.container).append(this.dom.panes);
|
||||||
|
q(this.dom.panes).empty();var c=this.s.dt.state.loaded();if(this.c.viewTotal&&!this.c.cascadePanes&&null!==c&&void 0!==c&&void 0!==c.searchPanes&&void 0!==c.searchPanes.panes){for(var d=!1,e=0,g=c.searchPanes.panes;e<g.length;e++){var f=g[e];if(0<f.selected.length){d=!0;break}}if(d)for(d=0,e=this.s.panes;d<e.length;d++)f=e[d],f.s.showFiltered=!0}d=0;for(e=this.s.panes;d<e.length;d++)f=e[d],f.rebuildPane(void 0,0<Object.keys(this.s.serverData).length?this.s.serverData:void 0),q(this.dom.panes).append(f.dom.container);
|
||||||
|
this.s.dt.page.info().serverSide||this.s.dt.draw();this.s.stateRead||null===c||void 0===c||(this.s.dt.page(c.start/this.s.dt.page.len()),this.s.dt.draw("page"));this.s.stateRead=!0;if(this.c.viewTotal&&!this.c.cascadePanes)for(c=0,d=this.s.panes;c<d.length;c++)f=d[c],f.updatePane();this._updateFilterCount();this._checkMessage();a.on("preDraw.dtsps",function(){b._updateFilterCount();!b.c.cascadePanes&&!b.c.viewTotal||b.s.dt.page.info().serverSide?b._updateSelection():b.redrawPanes();b.s.filterPane=
|
||||||
|
-1});this.s.dt.on("stateSaveParams.dtsp",function(l,p,n){void 0===n.searchPanes&&(n.searchPanes={});n.searchPanes.selectionList=b.s.selectionList});if(this.s.dt.page.info().serverSide)a.off("page"),a.on("page",function(){b.s.page=b.s.dt.page()}),a.off("preXhr.dt"),a.on("preXhr.dt",function(l,p,n){void 0===n.searchPanes&&(n.searchPanes={});p=l=0;for(var x=b.s.panes;p<x.length;p++){var z=x[p],y=b.s.dt.column(z.s.index).dataSrc();void 0===n.searchPanes[y]&&(n.searchPanes[y]={});if(void 0!==z.s.dtPane){z=
|
||||||
|
z.s.dtPane.rows({selected:!0}).data().toArray();for(var w=0;w<z.length;w++)n.searchPanes[y][w]=z[w].filter,l++}}b.c.viewTotal&&b._prepViewTotal();0<l&&(l!==b.s.filterCount?(n.start=0,b.s.page=0):n.start=b.s.page*b.s.dt.page.len(),b.s.dt.page(b.s.page),b.s.filterCount=l)});else a.on("preXhr.dt",function(l,p,n){l=0;for(p=b.s.panes;l<p.length;l++)p[l].clearData()});this.s.dt.on("xhr",function(l,p,n,x){var z=!1;if(!b.s.dt.page.info().serverSide)b.s.dt.one("preDraw",function(){if(!z){var y=b.s.dt.page();
|
||||||
|
z=!0;q(b.dom.panes).empty();for(var w=0,u=b.s.panes;w<u.length;w++){var B=u[w];B.clearData();B.rebuildPane(void 0!==b.s.selectionList[b.s.selectionList.length-1]?B.s.index===b.s.selectionList[b.s.selectionList.length-1].index:!1,void 0,void 0,!0);q(b.dom.panes).append(B.dom.container)}b.s.dt.page.info().serverSide||b.s.dt.draw();b.c.cascadePanes||b.c.viewTotal?b.redrawPanes(b.c.cascadePanes):b._updateSelection();b._checkMessage();b.s.dt.one("draw",function(){b.s.dt.page(y).draw(!1)})}})});c=0;for(d=
|
||||||
|
this.s.panes;c<d.length;c++)if(f=d[c],void 0!==f&&void 0!==f.s.dtPane&&(void 0!==f.s.colOpts.preSelect&&0<f.s.colOpts.preSelect.length||null!==f.customPaneSettings&&void 0!==f.customPaneSettings.preSelect&&0<f.customPaneSettings.preSelect.length)){e=f.s.dtPane.rows().data().toArray().length;for(g=0;g<e;g++)(-1!==f.s.colOpts.preSelect.indexOf(f.s.dtPane.cell(g,0).data())||null!==f.customPaneSettings&&void 0!==f.customPaneSettings.preSelect&&-1!==f.customPaneSettings.preSelect.indexOf(f.s.dtPane.cell(g,
|
||||||
|
0).data()))&&f.s.dtPane.row(g).select();f.updateTable()}if(void 0!==this.s.selectionList&&0<this.s.selectionList.length)for(c=this.s.selectionList[this.s.selectionList.length-1].index,d=0,e=this.s.panes;d<e.length;d++)f=e[d],f.s.lastSelect=f.s.index===c;0<this.s.selectionList.length&&this.c.cascadePanes&&this._cascadeRegen(this.s.selectionList);this._updateFilterCount();a.on("destroy.dtsps",function(){for(var l=0,p=b.s.panes;l<p.length;l++)p[l].destroy();a.off(".dtsps");q(b.dom.clearAll).off(".dtsps");
|
||||||
|
q(b.dom.container).remove();b.clearSelections()});if(this.c.clear)q(this.dom.clearAll).on("click.dtsps",function(){b.clearSelections()});a.settings()[0]._searchPanes=this};k.prototype._prepViewTotal=function(){for(var a=this.s.filterPane,b=!1,c=0,d=this.s.panes;c<d.length;c++){var e=d[c];if(void 0!==e.s.dtPane){var g=e.s.dtPane.rows({selected:!0}).data().toArray().length;0<g&&-1===a?(a=e.s.index,b=!0):0<g&&(a=null)}}c=0;for(d=this.s.panes;c<d.length;c++)if(e=d[c],void 0!==e.s.dtPane&&(e.s.filteringActive=
|
||||||
|
!0,-1!==a&&null!==a&&a===e.s.index||!1===b))e.s.filteringActive=!1};k.prototype._updateFilterCount=function(){for(var a=0,b=0,c=this.s.panes;b<c.length;b++){var d=c[b];void 0!==d.s.dtPane&&(a+=d.getPaneCount())}b=this.s.dt.i18n("searchPanes.title","Filters Active - %d",a);q(this.dom.title).text(b);void 0!==this.c.filterChanged&&"function"===typeof this.c.filterChanged&&this.c.filterChanged.call(this.s.dt,a)};k.prototype._updateSelection=function(){this.s.selectionList=[];for(var a=0,b=this.s.panes;a<
|
||||||
|
b.length;a++){var c=b[a];void 0!==c.s.dtPane&&this.s.selectionList.push({index:c.s.index,rows:c.s.dtPane.rows({selected:!0}).data().toArray(),protect:!1})}this.s.dt.state.save()};k.version="1.2.2";k.classes={clear:"dtsp-clear",clearAll:"dtsp-clearAll",container:"dtsp-searchPanes",emptyMessage:"dtsp-emptyMessage",hide:"dtsp-hidden",panes:"dtsp-panesContainer",search:"dtsp-search",title:"dtsp-title",titleRow:"dtsp-titleRow"};k.defaults={cascadePanes:!1,clear:!0,container:function(a){return a.table().container()},
|
||||||
|
columns:[],filterChanged:void 0,layout:"columns-3",order:[],panes:[],viewTotal:!1};return k}();(function(k){"function"===typeof define&&define.amd?define(["jquery","datatables.net"],function(a){return k(a,window,document)}):"object"===typeof exports?module.exports=function(a,b){a||(a=window);b&&b.fn.dataTable||(b=require("datatables.net")(a,b).$);return k(b,a,a.document)}:k(window.jQuery,window,document)})(function(k,a,b){function c(e,g){void 0===g&&(g=!1);e=new d.Api(e);var f=e.init().searchPanes||
|
||||||
|
d.defaults.searchPanes;return(new G(e,f,g)).getNode()}m(k);t(k);var d=k.fn.dataTable;k.fn.dataTable.SearchPanes=G;k.fn.DataTable.SearchPanes=G;k.fn.dataTable.SearchPane=v;k.fn.DataTable.SearchPane=v;a=k.fn.dataTable.Api.register;a("searchPanes()",function(){return this});a("searchPanes.clearSelections()",function(){return this.iterator("table",function(e){e._searchPanes&&e._searchPanes.clearSelections()})});a("searchPanes.rebuildPane()",function(e,g){return this.iterator("table",function(f){f._searchPanes&&
|
||||||
|
f._searchPanes.rebuild(e,g)})});a("searchPanes.container()",function(){var e=this.context[0];return e._searchPanes?e._searchPanes.getNode():null});k.fn.dataTable.ext.buttons.searchPanesClear={text:"Clear Panes",action:function(e,g,f,l){g.searchPanes.clearSelections()}};k.fn.dataTable.ext.buttons.searchPanes={action:function(e,g,f,l){e.stopPropagation();this.popover(l._panes.getNode(),{align:"dt-container"});l._panes.rebuild(void 0,!0)},config:{},init:function(e,g,f){var l=new k.fn.dataTable.SearchPanes(e,
|
||||||
|
k.extend({filterChanged:function(n){e.button(g).text(e.i18n("searchPanes.collapse",{0:"SearchPanes",_:"SearchPanes (%d)"},n))}},f.config)),p=e.i18n("searchPanes.collapse","SearchPanes",0);e.button(g).text(p);f._panes=l},text:"Search Panes"};k(b).on("preInit.dt.dtsp",function(e,g,f){"dt"===e.namespace&&(g.oInit.searchPanes||d.defaults.searchPanes)&&(g._searchPanes||c(g,!0))});d.ext.feature.push({cFeature:"P",fnInit:c});d.ext.features&&d.ext.features.register("searchPanes",c)})})();
|
||||||
|
|
||||||
|
|
||||||
|
(function(c){"function"===typeof define&&define.amd?define(["jquery","datatables.net-bs4","datatables.net-searchpanes"],function(a){return c(a,window,document)}):"object"===typeof exports?module.exports=function(a,b){a||(a=window);b&&b.fn.dataTable||(b=require("datatables.net-bs4")(a,b).$);console.log(b.fn.dataTable);b.fn.dataTable.SearchPanes||(console.log("not present"),require("datatables.net-searchpanes")(a,b));return c(b,a,a.document)}:c(jQuery,window,document)})(function(c,a,b){a=c.fn.dataTable;
|
||||||
|
c.extend(!0,a.SearchPane.classes,{buttonGroup:"btn-group col justify-content-end",disabledButton:"disabled",dull:"",narrow:"col",pane:{container:"table"},paneButton:"btn btn-light",pill:"pill badge badge-pill badge-secondary",search:"col-sm form-control search",searchCont:"input-group col-sm",searchLabelCont:"input-group-append",subRow1:"dtsp-subRow1",subRow2:"dtsp-subRow2",table:"table table-sm table-borderless",topRow:"dtsp-topRow row"});c.extend(!0,a.SearchPanes.classes,{clearAll:"dtsp-clearAll col-auto btn btn-light",
|
||||||
|
container:"dtsp-searchPanes",panes:"dtsp-panes dtsp-container",title:"dtsp-title col",titleRow:"dtsp-titleRow row"});return a.searchPanes});
|
||||||
|
|
||||||
|
|
7
app/frontend/static/assets/vendors/js/hammer.min.js
vendored
Normal file
7
app/frontend/static/assets/vendors/js/hammer.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
5
app/frontend/static/assets/vendors/js/jquery.min.js
vendored
Normal file
5
app/frontend/static/assets/vendors/js/jquery.min.js
vendored
Normal file
File diff suppressed because one or more lines are too long
@ -1,14 +0,0 @@
|
|||||||
{% for item in data['notify_data'] %}
|
|
||||||
<!-- <div class="hidden">{{ item['id'] }}</div>-->
|
|
||||||
<div class="event">
|
|
||||||
<p class="font-weight-medium">{{ item['title'] }}</p>
|
|
||||||
<a class="d-flex align-items-center">
|
|
||||||
<div class="badge badge-primary">{{ item['date'] }}</div>
|
|
||||||
<span class="text-muted ml-2">{{ item['desc'] }}</span>
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
{% end %}
|
|
||||||
|
|
@ -14,11 +14,10 @@
|
|||||||
<link rel="stylesheet" href="/static/assets/vendors/ti-icons/css/themify-icons.css">
|
<link rel="stylesheet" href="/static/assets/vendors/ti-icons/css/themify-icons.css">
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/typicons/typicons.css">
|
<link rel="stylesheet" href="/static/assets/vendors/typicons/typicons.css">
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/fontawesome6/css/all.css">
|
<link rel="stylesheet" href="/static/assets/vendors/fontawesome6/css/all.css">
|
||||||
<link rel="stylesheet" type="text/css"
|
<link rel="stylesheet" type="text/css" href="https://cdn.datatables.net/v/bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2/datatables.min.css" />
|
||||||
href="https://cdn.datatables.net/v/bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2/datatables.min.css" />
|
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/css/vendor.bundle.base.css">
|
<link rel="stylesheet" href="/static/assets/vendors/css/vendor.bundle.base.css">
|
||||||
<link rel="stylesheet" href="/static/assets/css/crafty.css">
|
<link rel="stylesheet" href="/static/assets/css/crafty.css">
|
||||||
|
|
||||||
<link rel="manifest" href="/static/assets/crafty.webmanifest">
|
<link rel="manifest" href="/static/assets/crafty.webmanifest">
|
||||||
<meta name="mobile-web-app-capable" content="yes">
|
<meta name="mobile-web-app-capable" content="yes">
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<meta name="apple-mobile-web-app-capable" content="yes">
|
||||||
@ -32,7 +31,7 @@
|
|||||||
<!-- endinject -->
|
<!-- endinject -->
|
||||||
|
|
||||||
<!-- Plugin css for this page-->
|
<!-- Plugin css for this page-->
|
||||||
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.12.4/jquery.min.js"></script>
|
<script src="../static/assets/vendors/js/jquery.min.js"></script>
|
||||||
<!-- End Plugin css for this page-->
|
<!-- End Plugin css for this page-->
|
||||||
|
|
||||||
<!-- Layout styles -->
|
<!-- Layout styles -->
|
||||||
@ -44,21 +43,15 @@
|
|||||||
<link rel="stylesheet" href="/static/assets/css/crafty.css">
|
<link rel="stylesheet" href="/static/assets/css/crafty.css">
|
||||||
|
|
||||||
<!-- Alpine.js - The modern jQuery alternative -->
|
<!-- Alpine.js - The modern jQuery alternative -->
|
||||||
<script defer src="https://unpkg.com/alpinejs@3.x.x/dist/cdn.min.js"></script>
|
<script defer src="../static/assets/vendors/js/cdn.min.js"></script>
|
||||||
<!-- End Alpine.js -->
|
<!-- End Alpine.js -->
|
||||||
|
|
||||||
<!-- Bootstrap Toggle -->
|
<!-- Bootstrap Toggle -->
|
||||||
<link href="https://gitcdn.github.io/bootstrap-toggle/2.2.2/css/bootstrap-toggle.min.css" rel="stylesheet">
|
<link href="../static/assets/vendors/css/bootstrap-toggle.min.css" rel="stylesheet">
|
||||||
<script defer src="https://gitcdn.github.io/bootstrap-toggle/2.2.2/js/bootstrap-toggle.min.js"></script>
|
<script defer src="../static/assets/vendors/js/bootstrap-toggle.min.js"></script>
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/3.9.1/chart.min.js"
|
<script src="../static/assets/vendors/js/chart.min.js"></script>
|
||||||
integrity="sha512-ElRFoEQdI5Ht6kZvyzXhYG9NqjtkmlkfYk0wr6wHxU9JEHakS7UJZNeml5ALk+8IKlU6jDgMabC3vkumRokgJA=="
|
<script src="../static/assets/vendors/js/hammer.min.js"></script>
|
||||||
crossorigin="anonymous" referrerpolicy="no-referrer"></script>
|
<script src="../static/assets/vendors/js/chartjs-plugin-zoom.min.js"></script>
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/hammer.js/2.0.8/hammer.min.js"
|
|
||||||
integrity="sha512-UXumZrZNiOwnTcZSHLOfcTs0aos2MzBWHXOHOuB0J/R44QB0dwY5JgfbvljXcklVf65Gc4El6RjZ+lnwd2az2g=="
|
|
||||||
crossorigin="anonymous" referrerpolicy="no-referrer"></script>
|
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/chartjs-plugin-zoom/1.2.1/chartjs-plugin-zoom.min.js"
|
|
||||||
integrity="sha512-klQv6lz2YR+MecyFYMFRuU2eAl8IPRo6zHnsc9n142TJuJHS8CG0ix4Oq9na9ceeg1u5EkBfZsFcV3U7J51iew=="
|
|
||||||
crossorigin="anonymous" referrerpolicy="no-referrer"></script>
|
|
||||||
|
|
||||||
<!-- End Bootstrap Toggle -->
|
<!-- End Bootstrap Toggle -->
|
||||||
|
|
||||||
@ -91,8 +84,7 @@
|
|||||||
|
|
||||||
{% include notify.html %}
|
{% include notify.html %}
|
||||||
|
|
||||||
<button class="navbar-toggler navbar-toggler-right d-lg-none align-self-center" type="button"
|
<button class="navbar-toggler navbar-toggler-right d-lg-none align-self-center" type="button" data-toggle="offcanvas">
|
||||||
data-toggle="offcanvas">
|
|
||||||
<span class="mdi mdi-menu"></span>
|
<span class="mdi mdi-menu"></span>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@ -132,19 +124,32 @@
|
|||||||
|
|
||||||
.notification {
|
.notification {
|
||||||
position: relative;
|
position: relative;
|
||||||
box-sizing: border-box;
|
margin-right: 1rem;
|
||||||
padding: 0.5rem;
|
|
||||||
padding-left: 0.7rem;
|
|
||||||
width: 180px;
|
|
||||||
margin-left: 10px;
|
|
||||||
margin-right: 10px;
|
|
||||||
background: var(--card-banner-bg);
|
background: var(--card-banner-bg);
|
||||||
|
-webkit-transition: right 0.75s, opacity 0.75s, top 0.75s;
|
||||||
|
-moz-transition: right 0.75s, opacity 0.75s, top 0.75s;
|
||||||
|
-o-transition: right 0.75s, opacity 0.75s, top 0.75s;
|
||||||
transition: right 0.75s, opacity 0.75s, top 0.75s;
|
transition: right 0.75s, opacity 0.75s, top 0.75s;
|
||||||
right: -6rem;
|
right: -20rem;
|
||||||
opacity: 0.1;
|
opacity: 0.1;
|
||||||
margin-bottom: 1rem;
|
}
|
||||||
z-index: 999;
|
|
||||||
top: 0px;
|
.toast-header {
|
||||||
|
background-color: var(--card-banner-bg);
|
||||||
|
color: var(--base-text);
|
||||||
|
}
|
||||||
|
|
||||||
|
.toast-body {
|
||||||
|
background-color: var(--dropdown-bg);
|
||||||
|
color: var(--base-text);
|
||||||
|
}
|
||||||
|
|
||||||
|
.notification img {
|
||||||
|
max-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.notification strong {
|
||||||
|
line-height: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notification.active {
|
.notification.active {
|
||||||
@ -158,34 +163,23 @@
|
|||||||
top: -2rem;
|
top: -2rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.notification p {
|
|
||||||
margin: 0px;
|
|
||||||
width: calc(160.8px - 16px);
|
|
||||||
z-index: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
.notification span {
|
.notification span {
|
||||||
position: absolute;
|
|
||||||
right: 0.5rem;
|
|
||||||
top: 0.46rem;
|
|
||||||
cursor: pointer;
|
|
||||||
font-weight: bold;
|
|
||||||
line-height: 20px;
|
line-height: 20px;
|
||||||
font-size: 22px;
|
font-size: 15px;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
z-index: inherit;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
<div class="notifications"></div>
|
|
||||||
|
|
||||||
|
|
||||||
|
<div class="notifications"></div>
|
||||||
|
|
||||||
<script src="/static/assets/vendors/js/vendor.bundle.base.js"></script>
|
<script src="/static/assets/vendors/js/vendor.bundle.base.js"></script>
|
||||||
<script src="/static/assets/js/shared/off-canvas.js"></script>
|
<script src="/static/assets/js/shared/off-canvas.js"></script>
|
||||||
<script src="/static/assets/js/shared/hoverable-collapse.js"></script>
|
<script src="/static/assets/js/shared/hoverable-collapse.js"></script>
|
||||||
<script src="/static/assets/js/shared/misc.js"></script>
|
<script src="/static/assets/js/shared/misc.js"></script>
|
||||||
<script type="text/javascript"
|
<script type="text/javascript" src="../static/assets/vendors/js/datatables.min.js"></script>
|
||||||
src="https://cdn.datatables.net/v/bs4/dt-1.10.22/fh-3.1.7/r-2.2.6/sc-2.0.3/sp-1.2.2/datatables.min.js"></script>
|
<script src="../static/assets/vendors/js/bootbox.min.js"></script>
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootbox.js/5.4.0/bootbox.min.js"></script>
|
|
||||||
<script type="text/javascript" src="/static/assets/js/motd.js"></script>
|
<script type="text/javascript" src="/static/assets/js/motd.js"></script>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
@ -241,6 +235,7 @@
|
|||||||
|
|
||||||
let usingWebSockets = false;
|
let usingWebSockets = false;
|
||||||
let webSocket = null;
|
let webSocket = null;
|
||||||
|
let websocketTimeoutId = null;
|
||||||
// {% if request.protocol == 'https' %}
|
// {% if request.protocol == 'https' %}
|
||||||
usingWebSockets = true;
|
usingWebSockets = true;
|
||||||
|
|
||||||
@ -291,18 +286,19 @@
|
|||||||
wsOpen = false;
|
wsOpen = false;
|
||||||
console.log('Closed WebSocket', closeEvent);
|
console.log('Closed WebSocket', closeEvent);
|
||||||
|
|
||||||
if (typeof reconnectorId !== 'number') {
|
if (!document.hidden) {
|
||||||
setTimeout(sendWssError, 7000);
|
if (typeof reconnectorId !== 'number') {
|
||||||
|
setTimeout(sendWssError, 7000);
|
||||||
|
}
|
||||||
|
console.info("Reconnecting with a timeout of", (getRandomArbitrary(0, 2 ** failedConnectionCounter - 1) + 5) * 1000, "milliseconds");
|
||||||
|
// Discard old websocket and create a new one in 5 seconds
|
||||||
|
wsInternal = null
|
||||||
|
reconnectorId = setTimeout(startWebSocket, (getRandomArbitrary(0, 2 ** failedConnectionCounter - 1) + 5) * 1000)
|
||||||
|
|
||||||
|
failedConnectionCounter++;
|
||||||
}
|
}
|
||||||
console.info("Reconnecting with a timeout of", (getRandomArbitrary(0, 2 ** failedConnectionCounter - 1) + 5) * 1000, "milliseconds");
|
|
||||||
// Discard old websocket and create a new one in 5 seconds
|
|
||||||
wsInternal = null
|
|
||||||
reconnectorId = setTimeout(startWebSocket, (getRandomArbitrary(0, 2 ** failedConnectionCounter - 1) + 5) * 1000)
|
|
||||||
|
|
||||||
failedConnectionCounter++;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
webSocket = {
|
webSocket = {
|
||||||
on: function (event, callback) {
|
on: function (event, callback) {
|
||||||
console.log('registered ' + event + ' event');
|
console.log('registered ' + event + ' event');
|
||||||
@ -315,6 +311,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
wsInternal.send(JSON.stringify(message));
|
wsInternal.send(JSON.stringify(message));
|
||||||
|
},
|
||||||
|
close: function (code, reason) {
|
||||||
|
wsInternal.close(code, reason);
|
||||||
|
},
|
||||||
|
getStatus: function () {
|
||||||
|
return wsInternal.readyState;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -328,6 +330,21 @@
|
|||||||
warn('WebSockets are not supported in Crafty if not using the https protocol')
|
warn('WebSockets are not supported in Crafty if not using the https protocol')
|
||||||
// {% end%}
|
// {% end%}
|
||||||
|
|
||||||
|
// Managing Connexions for Multi Tab opened to reduce bandwith usage
|
||||||
|
document.addEventListener("visibilitychange", () => {
|
||||||
|
if (document.visibilityState == "hidden") {
|
||||||
|
websocketTimeoutId = setTimeout(() => {
|
||||||
|
webSocket.close(1000, "Closed due to Inactivity");
|
||||||
|
console.log('%c[Crafty Controller] %cClose Websocket due to Tab not active after 5s', 'font-weight: 900; color: #800080;', 'font-weight: 900; color: #eee;');
|
||||||
|
}, 10000);
|
||||||
|
} else {
|
||||||
|
clearTimeout(websocketTimeoutId)
|
||||||
|
if (webSocket.getStatus() == WebSocket.CLOSED) {
|
||||||
|
startWebSocket();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
if (webSocket) {
|
if (webSocket) {
|
||||||
webSocket.on('send_start_error', function (start_error) {
|
webSocket.on('send_start_error', function (start_error) {
|
||||||
var x = document.querySelector('.bootbox');
|
var x = document.querySelector('.bootbox');
|
||||||
@ -412,20 +429,26 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function eulaAgree(server_id, command) {
|
async function eulaAgree(server_id, command) {
|
||||||
//< !--this getCookie function is in base.html-- >
|
//< !--this getCookie function is in base.html-- >
|
||||||
var token = getCookie("_xsrf");
|
const token = getCookie("_xsrf");
|
||||||
|
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/servers/${server_id}/action/eula/`, {
|
||||||
type: "POST",
|
method: 'POST',
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: {
|
||||||
url: '/ajax/eula?id=' + server_id,
|
'X-XSRFToken': token
|
||||||
success: function (data) {
|
},
|
||||||
console.log("got response:");
|
|
||||||
console.log(data);
|
|
||||||
location.reload();
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.reload();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -482,30 +505,55 @@
|
|||||||
|
|
||||||
function notify(message) {
|
function notify(message) {
|
||||||
console.log(`notify(${message})`);
|
console.log(`notify(${message})`);
|
||||||
var paragraphEl = document.createElement('p');
|
var intId = getRandomInt(0, 100);
|
||||||
var closeEl = document.createElement('span');
|
var toastDiv = document.createElement('div');
|
||||||
|
toastDiv.setAttribute("id", "toast_" + intId);
|
||||||
|
toastDiv.setAttribute("class", "notification toast");
|
||||||
|
toastDiv.setAttribute("role", "alert");
|
||||||
|
toastDiv.setAttribute("aria-lived", "assertive");
|
||||||
|
toastDiv.setAttribute("aria-atomic", "true");
|
||||||
|
toastDiv.setAttribute("data-delay", "3000");
|
||||||
|
toastDiv.setAttribute("data-animation", "true");
|
||||||
|
toastDiv.setAttribute("data-autohide", "false");
|
||||||
|
|
||||||
paragraphEl.textContent = message;
|
var toastHeaderDiv = document.createElement('div');
|
||||||
|
toastHeaderDiv.setAttribute("class", "toast-header");
|
||||||
|
|
||||||
closeEl.innerHTML = '×';
|
var toastHeaderImg = document.createElement('img');
|
||||||
closeEl.addEventListener('click', function () { closeNotification(this) });
|
toastHeaderImg.setAttribute("src", "/static/assets/images/logo_small.svg");
|
||||||
|
toastHeaderImg.setAttribute("class", "mr-auto");
|
||||||
|
toastHeaderImg.setAttribute("alt", "logo");
|
||||||
|
toastHeaderDiv.appendChild(toastHeaderImg);
|
||||||
|
|
||||||
var parentEl = document.createElement('div');
|
var toastHeaderTitle = document.createElement('strong');
|
||||||
parentEl.appendChild(paragraphEl);
|
toastHeaderTitle.setAttribute("class", "mr-auto");
|
||||||
parentEl.appendChild(closeEl);
|
toastHeaderTitle.innerHTML = " Crafty Controller ";
|
||||||
|
toastHeaderDiv.appendChild(toastHeaderTitle);
|
||||||
|
|
||||||
parentEl.classList.add('notification');
|
var toastHeaderCloseSpan = document.createElement('span');
|
||||||
|
toastHeaderCloseSpan.setAttribute("class", "fa-solid fa-xmark");
|
||||||
|
toastHeaderCloseSpan.setAttribute("aria-hidden", "true");
|
||||||
|
toastHeaderCloseSpan.addEventListener('click', function () { closeNotification(this.parentElement) });
|
||||||
|
toastHeaderDiv.appendChild(toastHeaderCloseSpan);
|
||||||
|
|
||||||
document.querySelector('.notifications').appendChild(parentEl);
|
var toastBodyDiv = document.createElement('div');
|
||||||
|
toastBodyDiv.setAttribute("class", "toast-body");
|
||||||
|
toastBodyDiv.textContent = message;
|
||||||
|
|
||||||
|
toastDiv.appendChild(toastHeaderDiv);
|
||||||
|
toastDiv.appendChild(toastBodyDiv);
|
||||||
|
|
||||||
|
document.querySelector('.notifications').appendChild(toastDiv);
|
||||||
|
|
||||||
|
$('#toast_' + intId).toast('show');
|
||||||
|
|
||||||
setTimeout(function () {
|
setTimeout(function () {
|
||||||
parentEl.classList.add('active');
|
toastDiv.classList.add('active');
|
||||||
}, 200);
|
}, 200);
|
||||||
|
|
||||||
setTimeout(function (element) {
|
setTimeout(function (element) {
|
||||||
closeNotification(element);
|
closeNotification(element.parentElement);
|
||||||
}, 7500, closeEl);
|
}, 7500, toastHeaderCloseSpan);
|
||||||
|
|
||||||
`
|
`
|
||||||
<div class="notification">
|
<div class="notification">
|
||||||
@ -518,10 +566,10 @@
|
|||||||
|
|
||||||
document.addEventListener('alpine:init', () => {
|
document.addEventListener('alpine:init', () => {
|
||||||
console.log('%c[Crafty Controller] %cAlpine.js pre-initialization!', 'font-weight: 900; color: #800080;', 'color: #eee;');
|
console.log('%c[Crafty Controller] %cAlpine.js pre-initialization!', 'font-weight: 900; color: #800080;', 'color: #eee;');
|
||||||
})
|
});
|
||||||
document.addEventListener('alpine:initialized', () => {
|
document.addEventListener('alpine:initialized', () => {
|
||||||
console.log('%c[Crafty Controller] %cAlpine.js initialized!', 'font-weight: 900; color: #800080;', 'color: #eee;');
|
console.log('%c[Crafty Controller] %cAlpine.js initialized!', 'font-weight: 900; color: #800080;', 'color: #eee;');
|
||||||
})
|
});
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
console.log('%c[Crafty Controller] %cReady for JS!', 'font-weight: 900; color: #800080;', 'font-weight: 900; color: #eee;');
|
console.log('%c[Crafty Controller] %cReady for JS!', 'font-weight: 900; color: #800080;', 'font-weight: 900; color: #eee;');
|
||||||
@ -538,7 +586,7 @@
|
|||||||
});
|
});
|
||||||
$(document).ready(() => {
|
$(document).ready(() => {
|
||||||
if ('serviceWorker' in navigator) {
|
if ('serviceWorker' in navigator) {
|
||||||
navigator.serviceWorker.register('/static/assets/js/shared/service-worker.js', {scope: '/'})
|
navigator.serviceWorker.register('/static/assets/js/shared/service-worker.js', { scope: '/' })
|
||||||
.then(function (registration) {
|
.then(function (registration) {
|
||||||
console.log('Service Worker Registered');
|
console.log('Service Worker Registered');
|
||||||
});
|
});
|
||||||
|
@ -1,17 +1,19 @@
|
|||||||
<ul class="navbar-nav ml-auto">
|
<ul class="navbar-nav ml-auto">
|
||||||
<li class="nav-item dropdown">
|
<li class="nav-item dropdown notif-li">
|
||||||
<a class="nav-link count-indicator">
|
<a class="nav-link count-indicator dropdown-toggle" id="notifDropdown" href="#" aria-expanded="false">
|
||||||
<i class="fas fa-broadcast-tower
|
<i class="fas fa-broadcast-tower
|
||||||
{% if data.get('update_available') %}
|
{% if data.get('update_available') %}
|
||||||
text-danger
|
text-danger
|
||||||
{% end %}
|
{% end %}
|
||||||
"></i>
|
"></i><span id="notif-count" class="badge badge-notify"></span> </a>
|
||||||
<!-- <span class="count bg-success">3</span>-->
|
<div class="dropdown-menu dropdown-menu-right navbar-dropdown notif-div" style="width: 40vw; max-height: 80vh;" aria-labelledby="notifDropdown">
|
||||||
</a>
|
<ul style="padding-top: 10px;" id="announcements">
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
</li>
|
</li>
|
||||||
|
|
||||||
<li class="nav-item dropdown">
|
<li class="nav-item dropdown">
|
||||||
<a class="nav-link count-indicator" href="/panel/panel_config">
|
<a class="nav-link" href="/panel/panel_config">
|
||||||
<i class="fas fa-cogs"></i>
|
<i class="fas fa-cogs"></i>
|
||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
@ -33,27 +35,155 @@
|
|||||||
<p class="font-weight-light text-muted mb-0">Email: {{ data['user_data']['email'] }}</p>
|
<p class="font-weight-light text-muted mb-0">Email: {{ data['user_data']['email'] }}</p>
|
||||||
</div>
|
</div>
|
||||||
{% if data['user_data']['preparing'] %}
|
{% if data['user_data']['preparing'] %}
|
||||||
<span class="dropdown-item" id="support_progress"><i class="dropdown-item-icon mdi mdi-download-outline text-primary"></i>{{ translate('notify', 'supportLogs', data['lang']) }}<br><br></span>
|
<span class="dropdown-item" id="support_progress"><i class="dropdown-item-icon mdi mdi-download-outline text-primary"></i>{{ translate('notify', 'supportLogs',
|
||||||
|
data['lang']) }}<br><br></span>
|
||||||
<span class="dropdown-item" id="support_progress">
|
<span class="dropdown-item" id="support_progress">
|
||||||
<div class="support_progress" style="height: 15px; width: 100%;">
|
<div class="support_progress" style="height: 15px; width: 100%;">
|
||||||
<div class="progress-bar progress-bar-striped progress-bar-animated" id="logs_progress_bar" role="progressbar" style="width:0%;" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100">0%</div>
|
<div class="progress-bar progress-bar-striped progress-bar-animated" id="logs_progress_bar" role="progressbar" style="width:0%;" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100">0%</div>
|
||||||
</div>
|
</div>
|
||||||
</span>
|
</span>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a class="dropdown-item" id="support_logs"><i class="dropdown-item-icon mdi mdi-download-outline text-primary"></i>{{ translate('notify', 'supportLogs', data['lang']) }}</i></a>
|
<a class="dropdown-item" id="support_logs"><i class="dropdown-item-icon mdi mdi-download-outline text-primary"></i>{{ translate('notify', 'supportLogs',
|
||||||
|
data['lang']) }}</i></a>
|
||||||
{% end %}
|
{% end %}
|
||||||
{% if data['superuser'] %}
|
{% if data['superuser'] %}
|
||||||
<a class="dropdown-item" href="/panel/activity_logs"><i class="dropdown-item-icon mdi mdi-calendar-check-outline text-primary"></i>{{ translate('notify', 'activityLog', data['lang']) }}</a>
|
<a class="dropdown-item" href="/panel/activity_logs"><i class="dropdown-item-icon mdi mdi-calendar-check-outline text-primary"></i>{{ translate('notify',
|
||||||
|
'activityLog', data['lang']) }}</a>
|
||||||
{% end %}
|
{% end %}
|
||||||
<a class="dropdown-item" href="/logout"><i class="dropdown-item-icon mdi mdi-power text-primary"></i>{{ translate('notify', 'logout', data['lang']) }}</a>
|
<a class="dropdown-item" href="/logout"><i class="dropdown-item-icon mdi mdi-power text-primary"></i>{{
|
||||||
|
translate('notify', 'logout', data['lang']) }}</a>
|
||||||
</div>
|
</div>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
<style>
|
||||||
|
.badge-notify {
|
||||||
|
background: var(--purple);
|
||||||
|
color: var(--dark);
|
||||||
|
position: absolute;
|
||||||
|
-moz-transform: translate(-70%, -70%);
|
||||||
|
/* For Firefox */
|
||||||
|
-ms-transform: translate(-70%, -70%);
|
||||||
|
/* for IE */
|
||||||
|
-webkit-transform: translate(-70%, -70%);
|
||||||
|
/* For Safari, Chrome, iOS */
|
||||||
|
-o-transform: translate(-70%, -70%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.clear-button:hover {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide scrollbar for Chrome, Safari and Opera */
|
||||||
|
.notif-div::-webkit-scrollbar {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Hide scrollbar for IE, Edge and Firefox */
|
||||||
|
.notif-div {
|
||||||
|
-ms-overflow-style: none;
|
||||||
|
/* IE and Edge */
|
||||||
|
scrollbar-width: none;
|
||||||
|
/* Firefox */
|
||||||
|
}
|
||||||
|
</style>
|
||||||
<script>
|
<script>
|
||||||
function pfpError(image) {
|
function pfpError(image) {
|
||||||
image.onerror = "";
|
image.onerror = "";
|
||||||
image.src = "/static/assets/images/faces-clipart/pic-3.png";
|
image.src = "/static/assets/images/faces-clipart/pic-3.png";
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
function updateAnnouncements(data) {
|
||||||
|
console.log(data)
|
||||||
|
let text = "";
|
||||||
|
for (let value of data) {
|
||||||
|
text += `<li class="card-header header-sm justify-content-between align-items-center" id="${value.id}"><p style="float: right;"><i data-id="${value.id}"class="clear-button fa-regular fa-x"></i></p><a style="color: var(--purple);" href=${value.link} target="_blank"><h6>${value.title}</h6><small><p>${value.date}</p></small><p>${value.desc}</p></li></a>`
|
||||||
|
}
|
||||||
|
if (data.length > 0) {
|
||||||
|
localStorage.setItem("notif-count", data.length);
|
||||||
|
$("#notif-count").show()
|
||||||
|
$("#notif-count").html(data.length);
|
||||||
|
$("#announcements").html(text);
|
||||||
|
} else {
|
||||||
|
$("#announcements").html(`<p style='margin-top: 15px;' class='text-center'><i class="fa fa-bell-slash" aria-hidden="true"></i>
|
||||||
|
</p>`);
|
||||||
|
$("#notif-count").hide()
|
||||||
|
}
|
||||||
|
$(".clear-button").on("click", function (event) {
|
||||||
|
console.log("CLEAR BUTTON")
|
||||||
|
event.stopPropagation();
|
||||||
|
let uuid = event.target.getAttribute("data-id");
|
||||||
|
$(`#${uuid}`).remove();
|
||||||
|
send_clear(uuid);
|
||||||
|
let notif_count = localStorage.getItem("notif-count") - 1;
|
||||||
|
if (notif_count > 0) {
|
||||||
|
localStorage.setItem("notif-count", notif_count);
|
||||||
|
$("#notif-count").html(notif_count);
|
||||||
|
} else {
|
||||||
|
$("#announcements").html(`<p style='margin-top: 15px;' class='text-center'><i class="fa fa-bell-slash" aria-hidden="true"></i>
|
||||||
|
</p>`)
|
||||||
|
$("#notif-count").html("");
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
}
|
||||||
|
async function getAnnouncements() {
|
||||||
|
var token = getCookie("_xsrf");
|
||||||
|
let res = await fetch(`/api/v2/crafty/announcements/`, {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
console.log(responseData);
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
updateAnnouncements(responseData.data)
|
||||||
|
} else {
|
||||||
|
updateAnnouncements("<li><p>Trouble Getting Annoucements</p></li>")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
async function send_clear(uuid) {
|
||||||
|
var token = getCookie("_xsrf");
|
||||||
|
let body = JSON.stringify({ "id": uuid });
|
||||||
|
console.log(body)
|
||||||
|
let res = await fetch(`/api/v2/crafty/announcements/`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token,
|
||||||
|
},
|
||||||
|
body: body,
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
console.log(responseData);
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
bootbox.alert(responseData.error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Open / Close with Button */
|
||||||
|
$('li.dropdown.notif-li a').on('click', function (event) {
|
||||||
|
$(this).parent().toggleClass("show");
|
||||||
|
$('div.notif-div').toggleClass("show");
|
||||||
|
if ($('li.dropdown.notif-li a').attr('aria-expanded') === 'false') {
|
||||||
|
$('li.dropdown.notif-li a').attr('aria-expanded', "true");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
$('li.dropdown.notif-li a').attr('aria-expanded', "false");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/* Close when clicking ouside */
|
||||||
|
$('body').on('click', function (e) {
|
||||||
|
if (!$('li.dropdown.notif-li').is(e.target) && $('li.dropdown.notif-li').has(e.target).length === 0 && $('show').has(e.target).length === 0) {
|
||||||
|
$('li.notif-li').removeClass("show");
|
||||||
|
$('li.dropdown.notif-li a').attr('aria-expanded', "false");
|
||||||
|
$('div.notif-div').removeClass("show");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$(document).ready(function () {
|
||||||
|
getAnnouncements();
|
||||||
|
})
|
||||||
</script>
|
</script>
|
@ -6,7 +6,8 @@
|
|||||||
{% block title %}Crafty Controller - {{ translate('panelConfig', 'pageTitle', data['lang']) }}{% end %}
|
{% block title %}Crafty Controller - {{ translate('panelConfig', 'pageTitle', data['lang']) }}{% end %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap-select/1.13.10/css/bootstrap-select.min.css">
|
<link rel="stylesheet"
|
||||||
|
href="https://cdnjs.cloudflare.com/ajax/libs/bootstrap-select/1.13.10/css/bootstrap-select.min.css">
|
||||||
|
|
||||||
|
|
||||||
<div class="content-wrapper">
|
<div class="content-wrapper">
|
||||||
@ -50,7 +51,6 @@
|
|||||||
<!-- Page Title Header Ends-->
|
<!-- Page Title Header Ends-->
|
||||||
|
|
||||||
<form id="config-form" class="forms-sample" method="post" action="/panel/config_json">
|
<form id="config-form" class="forms-sample" method="post" action="/panel/config_json">
|
||||||
{% raw xsrf_form_html() %}
|
|
||||||
|
|
||||||
{% for item in data['config-json'].items() %}
|
{% for item in data['config-json'].items() %}
|
||||||
{% if item[0] == "reset_secrets_on_next_boot" %}
|
{% if item[0] == "reset_secrets_on_next_boot" %}
|
||||||
@ -73,8 +73,11 @@
|
|||||||
</select>
|
</select>
|
||||||
{% elif item[0] == 'disabled_language_files' %}
|
{% elif item[0] == 'disabled_language_files' %}
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<button type="button" class="btn btn-outline-default custom-picker" onclick="$('option', $('#lang_select')).each(function(element) {$(this).removeAttr('selected').prop('selected', false); $('.selectpicker').selectpicker('refresh')});">{{ translate('panelConfig', 'enableLang', data['lang']) }}</button>
|
<button type="button" class="btn btn-outline-default custom-picker"
|
||||||
<select id="lang_select" class="form-control selectpicker show-tick custom-picker" data-icon-base="fas" data-tick-icon="fa-check" multiple data-style="custom-picker">
|
onclick="$('option', $('#lang_select')).each(function(element) {$(this).removeAttr('selected').prop('selected', false); $('.selectpicker').selectpicker('refresh')});">{{
|
||||||
|
translate('panelConfig', 'enableLang', data['lang']) }}</button>
|
||||||
|
<select id="lang_select" class="form-control selectpicker show-tick custom-picker"
|
||||||
|
data-icon-base="fas" data-tick-icon="fa-check" multiple data-style="custom-picker">
|
||||||
{% for lang in data['all_languages'] %}
|
{% for lang in data['all_languages'] %}
|
||||||
{% if lang in item[1] %}
|
{% if lang in item[1] %}
|
||||||
<option selected>{{lang}}</option>
|
<option selected>{{lang}}</option>
|
||||||
@ -83,12 +86,17 @@
|
|||||||
{% end %}
|
{% end %}
|
||||||
{% end %}
|
{% end %}
|
||||||
</select>
|
</select>
|
||||||
<textarea id="disabled_lang" name="{{item[0]}}" class="form-control list hidden" rows="{{ len(data['all_languages']) }}" value="{{','.join(item[1])}}" hidden>{{','.join(item[1])}}</textarea>
|
<textarea id="disabled_lang" name="{{item[0]}}" class="form-control list hidden"
|
||||||
|
rows="{{ len(data['all_languages']) }}" value="{{','.join(item[1])}}"
|
||||||
|
hidden>{{','.join(item[1])}}</textarea>
|
||||||
</div>
|
</div>
|
||||||
{% elif item[0] == 'monitored_mounts'%}
|
{% elif item[0] == 'monitored_mounts'%}
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<button type="button" class="btn btn-outline-default custom-picker" onclick="$('option', $('#mount_select')).each(function(element) {$(this).removeAttr('selected').prop('selected', false); $('.selectpicker').selectpicker('refresh')});">{{ translate('panelConfig', 'noMounts', data['lang']) }}</button>
|
<button type="button" class="btn btn-outline-default custom-picker"
|
||||||
<select id="mount_select" class="form-control selectpicker show-tick" data-icon-base="fas" data-tick-icon="fa-check" multiple data-style="custom-picker">
|
onclick="$('option', $('#mount_select')).each(function(element) {$(this).removeAttr('selected').prop('selected', false); $('.selectpicker').selectpicker('refresh')});">{{
|
||||||
|
translate('panelConfig', 'noMounts', data['lang']) }}</button>
|
||||||
|
<select id="mount_select" class="form-control selectpicker show-tick" data-icon-base="fas"
|
||||||
|
data-tick-icon="fa-check" multiple data-style="custom-picker">
|
||||||
{% for mount in data['all_partitions'] %}
|
{% for mount in data['all_partitions'] %}
|
||||||
{% if mount in item[1] %}
|
{% if mount in item[1] %}
|
||||||
<option selected>{{mount}}</option>
|
<option selected>{{mount}}</option>
|
||||||
@ -97,10 +105,13 @@
|
|||||||
{% end %}
|
{% end %}
|
||||||
{% end %}
|
{% end %}
|
||||||
</select>
|
</select>
|
||||||
<textarea id="monitored_mounts" name="{{item[0]}}" class="form-control list hidden" rows="{{ len(data['all_partitions']) }}" value="{{','.join(item[1])}}" hidden>{{','.join(item[1])}}</textarea>
|
<textarea id="monitored_mounts" name="{{item[0]}}" class="form-control list hidden"
|
||||||
|
rows="{{ len(data['all_partitions']) }}" value="{{','.join(item[1])}}"
|
||||||
|
hidden>{{','.join(item[1])}}</textarea>
|
||||||
</div>
|
</div>
|
||||||
{% elif isinstance(item[1], list) %}
|
{% elif isinstance(item[1], list) %}
|
||||||
<textarea value="{{','.join(item[1])}}" type="text" name="{{item[0]}}" class="form-control list">{{','.join(item[1])}}</textarea>
|
<textarea id="{{item[0]}}" value="{{','.join(item[1])}}" type="text" name="{{item[0]}}"
|
||||||
|
class="form-control list">{{','.join(item[1])}}</textarea>
|
||||||
{% elif isinstance(item[1], bool) %}
|
{% elif isinstance(item[1], bool) %}
|
||||||
<div style="margin-left: 30px;">
|
<div style="margin-left: 30px;">
|
||||||
{% if item[1] == True %}
|
{% if item[1] == True %}
|
||||||
@ -116,9 +127,11 @@
|
|||||||
{% end %}
|
{% end %}
|
||||||
</div>
|
</div>
|
||||||
{% elif isinstance(item[1], int) %}
|
{% elif isinstance(item[1], int) %}
|
||||||
<input type="number" class="form-control" name="{{item[0]}}" id="{{item[0]}}" value="{{ item[1] }}" step="1" min="0" required>
|
<input type="number" class="form-control" name="{{item[0]}}" id="{{item[0]}}" value="{{ item[1] }}"
|
||||||
|
step="1" min="0" required>
|
||||||
{% else %}
|
{% else %}
|
||||||
<input type="text" class="form-control" name="{{item[0]}}" id="{{item[0]}}" value="{{ item[1] }}" step="2" min="0" required>
|
<input type="text" class="form-control" name="{{item[0]}}" id="{{item[0]}}" value="{{ item[1] }}"
|
||||||
|
step="2" min="0" required>
|
||||||
{% end %}
|
{% end %}
|
||||||
</div>
|
</div>
|
||||||
{% end %}
|
{% end %}
|
||||||
@ -156,36 +169,66 @@
|
|||||||
|
|
||||||
{% block js %}
|
{% block js %}
|
||||||
<script>
|
<script>
|
||||||
$("#config-form").submit(function (e) {
|
function replacer(key, value) {
|
||||||
let uuid = uuidv4();
|
if (key == "disabled_language_files") {
|
||||||
var token = getCookie("_xsrf")
|
if (value == 0) {
|
||||||
|
return []
|
||||||
|
} else {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (typeof value == "boolean") {
|
||||||
|
return value
|
||||||
|
} else {
|
||||||
|
return (isNaN(value) ? value : +value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$("#config-form").on("submit", async function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
$("#submit-status").html('<i class="fa fa-spinner fa-spin"></i>');
|
$("#submit-status").html('<i class="fa fa-spinner fa-spin"></i>');
|
||||||
/* Convert multiple select to text list */
|
const token = getCookie("_xsrf")
|
||||||
let selected_Lang = $('#lang_select').val();
|
let configForm = document.getElementById("config-form");
|
||||||
$('#disabled_lang').val(selected_Lang);
|
|
||||||
|
|
||||||
let mounts = $('#mount_select').val();
|
let formData = new FormData(configForm);
|
||||||
$('#monitored_mounts').val(mounts);
|
formData.delete("disabled_lang");
|
||||||
|
formData.delete("lang_select");
|
||||||
|
|
||||||
let class_list = document.getElementsByClassName("list");
|
//Create an object from the form data entries
|
||||||
let form_json = convertFormToJSON($("#config-form"));
|
let formDataObject = Object.fromEntries(formData.entries());
|
||||||
for (let i = 0; i < class_list.length; i++) {
|
//We need to make sure these are sent regardless of whether or not they're checked
|
||||||
let str = String($(class_list.item(i)).val())
|
formDataObject.disabled_language_files = $('#lang_select').val();
|
||||||
form_json[$(class_list.item(i)).attr("name")] = uuid + "," + str.replace(/\s/g, '');
|
formDataObject.monitored_mounts = $('#mount_select').val();
|
||||||
};
|
formDataObject.keywords = $('#keywords').val().split(",");
|
||||||
form_json['uuid'] = uuid;
|
$('#config-form input[type="radio"]:checked').each(function () {
|
||||||
$.ajax({
|
if ($(this).val() == 'True') {
|
||||||
type: "POST",
|
formDataObject[this.name] = true;
|
||||||
headers: { 'X-XSRFToken': token },
|
} else {
|
||||||
dataType: "text",
|
formDataObject[this.name] = false;
|
||||||
url: '/panel/config_json',
|
}
|
||||||
data: form_json,
|
|
||||||
success: function (data) {
|
|
||||||
$("#submit-status").html('<i class="fa fa-check"></i>');
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
console.log(formDataObject);
|
||||||
|
// Format the plain form data as JSON
|
||||||
|
let formDataJsonString = JSON.stringify(formDataObject, replacer);
|
||||||
|
|
||||||
|
console.log(formDataJsonString);
|
||||||
|
|
||||||
|
let res = await fetch(`/api/v2/crafty/config/`, {
|
||||||
|
method: 'PATCH',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: formDataJsonString,
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
$("#submit-status").html('<i class="fa fa-check"></i>');
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function uuidv4() {
|
function uuidv4() {
|
||||||
@ -257,7 +300,7 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
$('.clear-comm').click(function () {
|
$('.clear-comm').click(function () {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "POST",
|
type: "POST",
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: { 'X-XSRFToken': token },
|
||||||
@ -268,7 +311,7 @@
|
|||||||
})
|
})
|
||||||
|
|
||||||
$('.delete-photo').click(function () {
|
$('.delete-photo').click(function () {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
let photo = $('#photo').find(":selected").val();
|
let photo = $('#photo').find(":selected").val();
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "POST",
|
type: "POST",
|
||||||
@ -281,7 +324,7 @@
|
|||||||
})
|
})
|
||||||
|
|
||||||
$('.select-photo').click(function () {
|
$('.select-photo').click(function () {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
let photo = $('#photo').find(":selected").val();
|
let photo = $('#photo').find(":selected").val();
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "POST",
|
type: "POST",
|
||||||
@ -293,6 +336,6 @@
|
|||||||
});
|
});
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
<script src="https://cdnjs.cloudflare.com/ajax/libs/bootstrap-select/1.13.10/js/bootstrap-select.min.js">
|
<script src="../../static/assets/vendors/js/bootstrap-select.min.js">
|
||||||
</script>
|
</script>
|
||||||
{% end %}
|
{% end %}
|
@ -62,11 +62,14 @@
|
|||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<div id="upload_input" class="input-group">
|
<div id="upload_input" class="input-group">
|
||||||
<div class="custom-file">
|
<div class="custom-file">
|
||||||
<input type="file" class="custom-file-input" id="file" name="file" multiple="false" required>
|
<input type="file" class="custom-file-input" id="file" name="file" multiple="false"
|
||||||
<label id="fileLabel" class="custom-file-label" for="file">{{ translate('customLogin', 'labelLoginImage', data['lang']) }}</label>
|
required>
|
||||||
|
<label id="fileLabel" class="custom-file-label" for="file">{{ translate('customLogin',
|
||||||
|
'labelLoginImage', data['lang']) }}</label>
|
||||||
</div>
|
</div>
|
||||||
<div class="input-group-append">
|
<div class="input-group-append">
|
||||||
<button type="button" class="btn btn-info upload-button" id="upload-button" onclick="sendFile()" disabled>UPLOAD</button>
|
<button type="button" class="btn btn-info upload-button" id="upload-button"
|
||||||
|
onclick="sendFile()" disabled>UPLOAD</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -81,7 +84,8 @@
|
|||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
<label for="photo" class="col-sm-6 col-form-label">Selected Background Image</label>
|
<label for="photo" class="col-sm-6 col-form-label">Selected Background Image</label>
|
||||||
<div class="col-sm-6">
|
<div class="col-sm-6">
|
||||||
<select class="form-select form-control form-control-lg select-css form-control-plaintext" id="photo" name="photo" form="photo_form" onchange="updateBackgroundPreview()">
|
<select class="form-select form-control form-control-lg select-css form-control-plaintext"
|
||||||
|
id="photo" name="photo" form="photo_form" onchange="updateBackgroundPreview()">
|
||||||
{% for image in data["backgrounds"] %}
|
{% for image in data["backgrounds"] %}
|
||||||
<option value="{{image}}">{{image}}</option>
|
<option value="{{image}}">{{image}}</option>
|
||||||
{% end %}
|
{% end %}
|
||||||
@ -90,7 +94,9 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="photo_loading" class="form-group" hidden>
|
<div id="photo_loading" class="form-group" hidden>
|
||||||
<div class="progress">
|
<div class="progress">
|
||||||
<div class="progress-bar progress-bar-striped progress-bar-animated" role="progressbar" aria-valuenow="100" aria-valuemin="0" aria-valuemax="100" style="width: 100%"> <i class="fa-solid fa-spinner"></i></div>
|
<div class="progress-bar progress-bar-striped progress-bar-animated" role="progressbar"
|
||||||
|
aria-valuenow="100" aria-valuemin="0" aria-valuemax="100" style="width: 100%"> <i
|
||||||
|
class="fa-solid fa-spinner"></i></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group row">
|
<div class="form-group row">
|
||||||
@ -98,11 +104,13 @@
|
|||||||
data['lang']) }}</label>
|
data['lang']) }}</label>
|
||||||
<label class="col-sm-1" id="opacityValue">{{ data['login_opacity'] }}%</label>
|
<label class="col-sm-1" id="opacityValue">{{ data['login_opacity'] }}%</label>
|
||||||
<div class="range col-sm-8">
|
<div class="range col-sm-8">
|
||||||
<input type="range" class="form-control-range" id="modal_opacity" name="modal_opacity" onchange="previewOpacity()" min="0" max="100" value="{{ data['login_opacity'] }}">
|
<input type="range" class="form-control-range" id="modal_opacity" name="modal_opacity"
|
||||||
|
onchange="previewOpacity()" min="0" max="100" value="{{ data['login_opacity'] }}">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="login_preview" style="position: relative;">
|
<div id="login_preview" style="position: relative;">
|
||||||
<img id="bg-preview" src="../../static/assets/images/auth/{{ data['background'] }}" class="img-fluid" alt="Responsive image">
|
<img id="bg-preview" src="../../static/assets/images/auth/{{ data['background'] }}"
|
||||||
|
class="img-fluid" alt="Responsive image">
|
||||||
<div id="login-form-preview">
|
<div id="login-form-preview">
|
||||||
<div id="login-form-background" class="auto-form-wrapper login-modal">
|
<div id="login-form-background" class="auto-form-wrapper login-modal">
|
||||||
<div class="text-center auto-form-logo">
|
<div class="text-center auto-form-logo">
|
||||||
@ -166,17 +174,20 @@
|
|||||||
</style>
|
</style>
|
||||||
|
|
||||||
<div id="login_form_data">
|
<div id="login_form_data">
|
||||||
<input type="hidden" name="_xsrf" value="2|1d603267|809fb6bd82f677d440e484dde7c3a310|1671726040" disabled>
|
<input type="hidden" name="_xsrf"
|
||||||
|
value="2|1d603267|809fb6bd82f677d440e484dde7c3a310|1671726040" disabled>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="label">Username</label>
|
<label class="label">Username</label>
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<input type="text" class="form-control login-text-input login-input" placeholder="Username" name="username" id="username" required="true" disabled>
|
<input type="text" class="form-control login-text-input login-input"
|
||||||
|
placeholder="Username" name="username" id="username" required="true" disabled>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="label">Password</label>
|
<label class="label">Password</label>
|
||||||
<div class="input-group">
|
<div class="input-group">
|
||||||
<input type="password" class="form-control login-text-input login-input" placeholder="Password" name="password" id="password" required="true" disabled>
|
<input type="password" class="form-control login-text-input login-input"
|
||||||
|
placeholder="Password" name="password" id="password" required="true" disabled>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
@ -195,7 +206,8 @@
|
|||||||
<a href="#" class="text-small forgot-password" disabled>Forgot Password</a>
|
<a href="#" class="text-small forgot-password" disabled>Forgot Password</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="text-block text-center my-3">
|
<div class="text-block text-center my-3">
|
||||||
<span class="text-small font-weight-semibold"><a href="https://craftycontrol.com/">Crafty Control
|
<span class="text-small font-weight-semibold"><a
|
||||||
|
href="https://craftycontrol.com/">Crafty Control
|
||||||
4.0.20</a> </span>
|
4.0.20</a> </span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -297,33 +309,50 @@
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
$('.delete-photo').click(function () {
|
$('.delete-photo').click(async function () {
|
||||||
var token = getCookie("_xsrf")
|
|
||||||
let photo = $('#photo').find(":selected").val();
|
let photo = $('#photo').find(":selected").val();
|
||||||
$.ajax({
|
const token = getCookie("_xsrf")
|
||||||
type: "POST",
|
let res = await fetch(`/api/v2/crafty/config/customize`, {
|
||||||
headers: { 'X-XSRFToken': token },
|
method: 'DELETE',
|
||||||
url: '/ajax/delete_photo?photo=' + encodeURIComponent(photo),
|
headers: {
|
||||||
success: function (data) {
|
'X-XSRFToken': token
|
||||||
location.reload();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "photo": photo }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.reload();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
$('.select-photo').click(function () {
|
$('.select-photo').click(async function () {
|
||||||
var token = getCookie("_xsrf")
|
|
||||||
let photo = $('#photo').find(":selected").val();
|
let photo = $('#photo').find(":selected").val();
|
||||||
let opacity = $('#modal_opacity').val();
|
let opacity = $('#modal_opacity').val();
|
||||||
let enc_photo = encodeURIComponent(photo);
|
console.log(JSON.stringify({ "photo": photo, "opacity": opacity }))
|
||||||
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/crafty/config/customize`, {
|
||||||
type: "POST",
|
method: 'PATCH',
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: {
|
||||||
url: '/ajax/select_photo?photo=' + enc_photo + '&opacity=' + opacity,
|
'X-XSRFToken': token
|
||||||
success: function (data) {
|
|
||||||
window.location.reload();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "photo": photo, "opacity": opacity }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.reload();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
|
@ -58,13 +58,11 @@
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="wrapper my-auto ml-auto ml-lg-4">
|
<div class="wrapper my-auto ml-auto ml-lg-4">
|
||||||
<p id="cpu_data" class="mb-0 text-success" data-toggle="tooltip" data-placement="top" data-html="true"
|
<p id="cpu_data" class="mb-0 text-success" data-toggle="tooltip" data-placement="top" data-html="true" title="{% raw translate('dashboard', 'cpuCores', data['lang']) %}: {{ data.get('hosts_data').get('cpu_cores') }} <br /> {% raw translate('dashboard', 'cpuCurFreq', data['lang']) %}: {{ data.get('hosts_data').get('cpu_cur_freq') }} <br /> {% raw translate('dashboard', 'cpuMaxFreq', data['lang']) %}: {{ data.get('hosts_data').get('cpu_max_freq') }}">
|
||||||
title="{% raw translate('dashboard', 'cpuCores', data['lang']) %}: {{ data.get('hosts_data').get('cpu_cores') }} <br /> {% raw translate('dashboard', 'cpuCurFreq', data['lang']) %}: {{ data.get('hosts_data').get('cpu_cur_freq') }} <br /> {% raw translate('dashboard', 'cpuMaxFreq', data['lang']) %}: {{ data.get('hosts_data').get('cpu_max_freq') }}">
|
|
||||||
{{ translate('dashboard', 'cpuUsage', data['lang']) }}: <span id="cpu_usage">{{
|
{{ translate('dashboard', 'cpuUsage', data['lang']) }}: <span id="cpu_usage">{{
|
||||||
data.get('hosts_data').get('cpu_usage') }}</span>
|
data.get('hosts_data').get('cpu_usage') }}</span>
|
||||||
</p>
|
</p>
|
||||||
<p id="mem_usage" class="mb-0 text-danger" data-toggle="tooltip" data-placement="top"
|
<p id="mem_usage" class="mb-0 text-danger" data-toggle="tooltip" data-placement="top" title="{{ translate('dashboard', 'memUsage', data['lang']) }}: {{ data.get('hosts_data').get('mem_usage') }}">
|
||||||
title="{{ translate('dashboard', 'memUsage', data['lang']) }}: {{ data.get('hosts_data').get('mem_usage') }}">
|
|
||||||
{{ translate('dashboard', 'memUsage', data['lang']) }}: <span id="mem_percent">{{
|
{{ translate('dashboard', 'memUsage', data['lang']) }}: <span id="mem_percent">{{
|
||||||
data.get('hosts_data').get('mem_percent') }}%</span>
|
data.get('hosts_data').get('mem_percent') }}%</span>
|
||||||
</p>
|
</p>
|
||||||
@ -111,12 +109,9 @@
|
|||||||
{% for item in data['hosts_data']['disk_json'] %}
|
{% for item in data['hosts_data']['disk_json'] %}
|
||||||
{% if item["mount"] in data["monitored"] %}
|
{% if item["mount"] in data["monitored"] %}
|
||||||
<div id="{{item['device']}}" class="col-xl-3 col-lg-3 col-md-4 col-12">
|
<div id="{{item['device']}}" class="col-xl-3 col-lg-3 col-md-4 col-12">
|
||||||
<h4 class="mb-0 font-weight-semibold d-inline-block text-truncate storage-heading"
|
<h4 class="mb-0 font-weight-semibold d-inline-block text-truncate storage-heading" id="title_{{item['device']}}" data-toggle="tooltip" data-placement="bottom" title="{{item['mount']}}" style="max-width: 100%;"><i class="fas fa-hdd"></i>
|
||||||
id="title_{{item['device']}}" data-toggle="tooltip" data-placement="bottom"
|
|
||||||
title="{{item['mount']}}" style="max-width: 100%;"><i class="fas fa-hdd"></i>
|
|
||||||
{{item["mount"]}}</h4>
|
{{item["mount"]}}</h4>
|
||||||
<div class="progress d-inline-block"
|
<div class="progress d-inline-block" style="height: 20px; width: 100%; background-color: rgb(139, 139, 139) !important;">
|
||||||
style="height: 20px; width: 100%; background-color: rgb(139, 139, 139) !important;">
|
|
||||||
<div class="progress-bar
|
<div class="progress-bar
|
||||||
{% if item['percent_used'] <= 58 %}
|
{% if item['percent_used'] <= 58 %}
|
||||||
bg-success
|
bg-success
|
||||||
@ -125,8 +120,7 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
bg-danger
|
bg-danger
|
||||||
{% end %}
|
{% end %}
|
||||||
" role="progressbar" style="color: black; height: 100%; width: {{item['percent_used']}}%;"
|
" role="progressbar" style="color: black; height: 100%; width: {{item['percent_used']}}%;" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100">{{item["used"]}} /
|
||||||
aria-valuenow="0" aria-valuemin="0" aria-valuemax="100">{{item["used"]}} /
|
|
||||||
{{item["total"]}}
|
{{item["total"]}}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -153,9 +147,7 @@
|
|||||||
data['lang']) }}</h4>
|
data['lang']) }}</h4>
|
||||||
{% if len(data['servers']) > 0 %}
|
{% if len(data['servers']) > 0 %}
|
||||||
{% if data['user_data']['hints'] %}
|
{% if data['user_data']['hints'] %}
|
||||||
<span class="too_small" title="{{ translate('dashboard', 'cannotSeeOnMobile', data['lang']) }}" ,
|
<span class="too_small" title="{{ translate('dashboard', 'cannotSeeOnMobile', data['lang']) }}" , data-content="{{ translate('dashboard', 'cannotSeeOnMobile2', data['lang']) }}" , data-placement="top"></span>
|
||||||
data-content="{{ translate('dashboard', 'cannotSeeOnMobile2', data['lang']) }}" ,
|
|
||||||
data-placement="top"></span>
|
|
||||||
{% end %}
|
{% end %}
|
||||||
{% end %}
|
{% end %}
|
||||||
<div><a class="nav-link" href="/server/step1"><i class="fas fa-plus-circle"></i> {{
|
<div><a class="nav-link" href="/server/step1"><i class="fas fa-plus-circle"></i> {{
|
||||||
@ -193,8 +185,7 @@
|
|||||||
<td draggable="false">
|
<td draggable="false">
|
||||||
<i class="fas fa-server"></i>
|
<i class="fas fa-server"></i>
|
||||||
{% if server['alert'] %}
|
{% if server['alert'] %}
|
||||||
<a style="color: red !important;" draggable="false"
|
<a style="color: red !important;" draggable="false" href="/panel/server_detail?id={{server['server_data']['server_id']}}">
|
||||||
href="/panel/server_detail?id={{server['server_data']['server_id']}}">
|
|
||||||
{{ server['server_data']['server_name'] }} <i class="fas fa-exclamation-triangle"></i>
|
{{ server['server_data']['server_name'] }} <i class="fas fa-exclamation-triangle"></i>
|
||||||
</a>
|
</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
@ -208,13 +199,11 @@
|
|||||||
{% if server['user_command_permission'] %}
|
{% if server['user_command_permission'] %}
|
||||||
{% if server['stats']['importing'] and server['stats']['running'] %}
|
{% if server['stats']['importing'] and server['stats']['running'] %}
|
||||||
<!-- WHAT HAPPENED HERE -->
|
<!-- WHAT HAPPENED HERE -->
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class=""><i
|
<a data-id="{{server['server_data']['server_id']}}" class=""><i class="fa fa-spinner fa-spin"></i> {{ translate('serverTerm', 'installing',
|
||||||
class="fa fa-spinner fa-spin"></i> {{ translate('serverTerm', 'installing',
|
|
||||||
data['lang']) }}</i></a>
|
data['lang']) }}</i></a>
|
||||||
{% elif server['stats']['updating']%}
|
{% elif server['stats']['updating']%}
|
||||||
<!-- WHAT HAPPENED HERE -->
|
<!-- WHAT HAPPENED HERE -->
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class=""><i
|
<a data-id="{{server['server_data']['server_id']}}" class=""><i class="fa fa-spinner fa-spin"></i> {{ translate('serverTerm', 'updating',
|
||||||
class="fa fa-spinner fa-spin"></i> {{ translate('serverTerm', 'updating',
|
|
||||||
data['lang']) }}</i></a>
|
data['lang']) }}</i></a>
|
||||||
{% elif server['stats']['waiting_start']%}
|
{% elif server['stats']['waiting_start']%}
|
||||||
<!-- WHAT HAPPENED HERE -->
|
<!-- WHAT HAPPENED HERE -->
|
||||||
@ -226,31 +215,25 @@
|
|||||||
{{ translate('serverTerm', 'importing',
|
{{ translate('serverTerm', 'importing',
|
||||||
data['lang']) }}</a>
|
data['lang']) }}</a>
|
||||||
{% elif server['stats']['running'] %}
|
{% elif server['stats']['running'] %}
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="stop_button" data-toggle="tooltip"
|
<a data-id="{{server['server_data']['server_id']}}" class="stop_button" data-toggle="tooltip" title="{{ translate('dashboard', 'stop' , data['lang']) }}">
|
||||||
title="{{ translate('dashboard', 'stop' , data['lang']) }}">
|
|
||||||
<i class="fas fa-stop"></i>
|
<i class="fas fa-stop"></i>
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="restart_button" data-toggle="tooltip"
|
<a data-id="{{server['server_data']['server_id']}}" class="restart_button" data-toggle="tooltip" title="{{ translate('dashboard', 'restart' , data['lang']) }}">
|
||||||
title="{{ translate('dashboard', 'restart' , data['lang']) }}">
|
|
||||||
<i class="fas fa-sync"></i>
|
<i class="fas fa-sync"></i>
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="kill_button" data-toggle="tooltip"
|
<a data-id="{{server['server_data']['server_id']}}" class="kill_button" data-toggle="tooltip" title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
||||||
title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
|
||||||
<i class="fas fa-skull"></i>
|
<i class="fas fa-skull"></i>
|
||||||
</a>
|
</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="play_button" data-toggle="tooltip"
|
<a data-id="{{server['server_data']['server_id']}}" class="play_button" data-toggle="tooltip" title="{{ translate('dashboard', 'start' , data['lang']) }}">
|
||||||
title="{{ translate('dashboard', 'start' , data['lang']) }}">
|
|
||||||
<i class="fas fa-play"></i>
|
<i class="fas fa-play"></i>
|
||||||
</a>
|
</a>
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="clone_button" data-toggle="tooltip"
|
<a data-id="{{server['server_data']['server_id']}}" class="clone_button" data-toggle="tooltip" title="{{ translate('dashboard', 'clone' , data['lang']) }}">
|
||||||
title="{{ translate('dashboard', 'clone' , data['lang']) }}">
|
|
||||||
<i class="fas fa-clone"></i>
|
<i class="fas fa-clone"></i>
|
||||||
</a>
|
</a>
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="kill_button" data-toggle="tooltip"
|
<a data-id="{{server['server_data']['server_id']}}" class="kill_button" data-toggle="tooltip" title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
||||||
title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
|
||||||
<i class="fas fa-skull"></i>
|
<i class="fas fa-skull"></i>
|
||||||
</a>
|
</a>
|
||||||
{% end %}
|
{% end %}
|
||||||
@ -258,8 +241,7 @@
|
|||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td draggable="false" id="server_cpu_{{server['server_data']['server_id']}}">
|
<td draggable="false" id="server_cpu_{{server['server_data']['server_id']}}">
|
||||||
<div class="progress mb-1" data-toggle="tooltip" data-placement="top"
|
<div class="progress mb-1" data-toggle="tooltip" data-placement="top" title="{{server['stats']['cpu']}}">
|
||||||
title="{{server['stats']['cpu']}}">
|
|
||||||
<div class="progress-bar
|
<div class="progress-bar
|
||||||
{% if server['stats']['cpu'] <= 33 %}
|
{% if server['stats']['cpu'] <= 33 %}
|
||||||
bg-success
|
bg-success
|
||||||
@ -268,15 +250,13 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
bg-danger
|
bg-danger
|
||||||
{% end %}
|
{% end %}
|
||||||
" role="progressbar" style="width: {{server['stats']['cpu']}}%" aria-valuenow="0"
|
" role="progressbar" style="width: {{server['stats']['cpu']}}%" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100"></div>
|
||||||
aria-valuemin="0" aria-valuemax="100"></div>
|
|
||||||
</div>
|
</div>
|
||||||
{{server['stats']['cpu']}}%
|
{{server['stats']['cpu']}}%
|
||||||
</td>
|
</td>
|
||||||
|
|
||||||
<td draggable="false" id="server_mem_{{server['server_data']['server_id']}}">
|
<td draggable="false" id="server_mem_{{server['server_data']['server_id']}}">
|
||||||
<div class="progress mb-1" data-toggle="tooltip" data-placement="top"
|
<div class="progress mb-1" data-toggle="tooltip" data-placement="top" title="{{server['stats']['mem']}}">
|
||||||
title="{{server['stats']['mem']}}">
|
|
||||||
<div class="progress-bar
|
<div class="progress-bar
|
||||||
{% if server['stats']['mem_percent'] <= 33 %}
|
{% if server['stats']['mem_percent'] <= 33 %}
|
||||||
bg-success
|
bg-success
|
||||||
@ -285,8 +265,7 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
bg-danger
|
bg-danger
|
||||||
{% end %}
|
{% end %}
|
||||||
" role="progressbar" style="width: {{server['stats']['mem_percent']}}%" aria-valuenow="0"
|
" role="progressbar" style="width: {{server['stats']['mem_percent']}}%" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100"></div>
|
||||||
aria-valuemin="0" aria-valuemax="100"></div>
|
|
||||||
</div>
|
</div>
|
||||||
{{server['stats']['mem_percent']}}% -
|
{{server['stats']['mem_percent']}}% -
|
||||||
|
|
||||||
@ -305,8 +284,7 @@
|
|||||||
data['lang']) }} <br />
|
data['lang']) }} <br />
|
||||||
|
|
||||||
{% if server['stats']['desc'] != 'False' %}
|
{% if server['stats']['desc'] != 'False' %}
|
||||||
<div id="desc_id"
|
<div id="desc_id" style="overflow-wrap: break-word !important; max-width: 85px !important; overflow: scroll;">{{
|
||||||
style="overflow-wrap: break-word !important; max-width: 85px !important; overflow: scroll;">{{
|
|
||||||
server['stats']['desc'] }}</div> <br />
|
server['stats']['desc'] }}</div> <br />
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
||||||
@ -334,16 +312,14 @@
|
|||||||
<br />
|
<br />
|
||||||
<br />
|
<br />
|
||||||
</td>
|
</td>
|
||||||
<span class="server-player-totals" id="server_players_{{server['server_data']['server_id']}}"
|
<span class="server-player-totals" id="server_players_{{server['server_data']['server_id']}}" data-players="{{ server['stats']['online']}}" data-max="{{ server['stats']['max'] }}"></span>
|
||||||
data-players="{{ server['stats']['online']}}" data-max="{{ server['stats']['max'] }}"></span>
|
|
||||||
</tr>
|
</tr>
|
||||||
{% end %}
|
{% end %}
|
||||||
</div>
|
</div>
|
||||||
</span>
|
</span>
|
||||||
{% for server in data['failed_servers'] %}
|
{% for server in data['failed_servers'] %}
|
||||||
<tr id="{{server['server_id']}}" draggable="false">
|
<tr id="{{server['server_id']}}" draggable="false">
|
||||||
<td class="text-warning"><i class="fas fa-server"></i> <a class="text-warning"
|
<td class="text-warning"><i class="fas fa-server"></i> <a class="text-warning" href="/panel/server_detail?id={{server['server_id']}}&subpage=config">{{server['server_name']}}</a>
|
||||||
href="/panel/server_detail?id={{server['server_id']}}&subpage=config">{{server['server_name']}}</a>
|
|
||||||
</td>
|
</td>
|
||||||
<td></td>
|
<td></td>
|
||||||
<td></td>
|
<td></td>
|
||||||
@ -368,28 +344,22 @@
|
|||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-10 col-lg-3 mx-0 px-0">
|
<div class="col-10 col-lg-3 mx-0 px-0">
|
||||||
{% if server['alert'] %}
|
{% if server['alert'] %}
|
||||||
<a style="color: red !important" class="btn btn-link d-flex justify-content-start" type="button"
|
<a style="color: red !important" class="btn btn-link d-flex justify-content-start" type="button" href="/panel/server_detail?id={{server['server_data']['server_id']}}">
|
||||||
href="/panel/server_detail?id={{server['server_data']['server_id']}}">
|
<i class="fas fa-server"></i> {{ server['server_data']['server_name'] }} <i class="fas fa-exclamation-triangle"></i>
|
||||||
<i class="fas fa-server"></i> {{ server['server_data']['server_name'] }} <i
|
|
||||||
class="fas fa-exclamation-triangle"></i>
|
|
||||||
</a>
|
</a>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a class="btn btn-link d-flex justify-content-start" type="button"
|
<a class="btn btn-link d-flex justify-content-start" type="button" href="/panel/server_detail?id={{server['server_data']['server_id']}}">
|
||||||
href="/panel/server_detail?id={{server['server_data']['server_id']}}">
|
|
||||||
<i class="fas fa-server"></i> {{ server['server_data']['server_name'] }}
|
<i class="fas fa-server"></i> {{ server['server_data']['server_name'] }}
|
||||||
</a>
|
</a>
|
||||||
{% end %}
|
{% end %}
|
||||||
</div>
|
</div>
|
||||||
<div class="col-2 col-lg-3 mx-0 px-0">
|
<div class="col-2 col-lg-3 mx-0 px-0">
|
||||||
<a class="btn btn-link d-flex justify-content-center" type="button" data-toggle="collapse"
|
<a class="btn btn-link d-flex justify-content-center" type="button" data-toggle="collapse" data-target="#collapse-{{server['server_data']['server_id']}}" aria-expanded="false" aria-controls="collapse-{{server['server_data']['server_id']}}">
|
||||||
data-target="#collapse-{{server['server_data']['server_id']}}" aria-expanded="false"
|
|
||||||
aria-controls="collapse-{{server['server_data']['server_id']}}">
|
|
||||||
<i class="fas fa-chart-bar"></i>
|
<i class="fas fa-chart-bar"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-4 col-lg-3 mx-0 px-0">
|
<div class="col-4 col-lg-3 mx-0 px-0">
|
||||||
<a id="m_server_running_status_{{server['server_data']['server_id']}}"
|
<a id="m_server_running_status_{{server['server_data']['server_id']}}" class="btn btn-link d-flex justify-content-start" type="button">
|
||||||
class="btn btn-link d-flex justify-content-start" type="button">
|
|
||||||
{% if server['stats']['running'] %}
|
{% if server['stats']['running'] %}
|
||||||
<span class="text-success"><i class="fas fa-signal"></i> {{ translate('dashboard', 'online',
|
<span class="text-success"><i class="fas fa-signal"></i> {{ translate('dashboard', 'online',
|
||||||
data['lang']) }}</span>
|
data['lang']) }}</span>
|
||||||
@ -410,23 +380,17 @@
|
|||||||
{% if server['stats']['running'] %}
|
{% if server['stats']['running'] %}
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-4 px-0">
|
<div class="col-4 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}"
|
<a data-id="{{server['server_data']['server_id']}}" class="btn btn-link stop_button actions_serveritem" data-toggle="tooltip" title="{{ translate('dashboard', 'stop' , data['lang']) }}">
|
||||||
class="btn btn-link stop_button actions_serveritem" data-toggle="tooltip"
|
|
||||||
title="{{ translate('dashboard', 'stop' , data['lang']) }}">
|
|
||||||
<i class="fas fa-stop"></i>
|
<i class="fas fa-stop"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-4 px-0">
|
<div class="col-4 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}"
|
<a data-id="{{server['server_data']['server_id']}}" class="btn btn-link restart_button actions_serveritem" data-toggle="tooltip" title="{{ translate('dashboard', 'restart' , data['lang']) }}">
|
||||||
class="btn btn-link restart_button actions_serveritem" data-toggle="tooltip"
|
|
||||||
title="{{ translate('dashboard', 'restart' , data['lang']) }}">
|
|
||||||
<i class="fas fa-sync"></i>
|
<i class="fas fa-sync"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-4 px-0">
|
<div class="col-4 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}"
|
<a data-id="{{server['server_data']['server_id']}}" class="btn btn-link kill_button actions_serveritem" data-toggle="tooltip" title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
||||||
class="btn btn-link kill_button actions_serveritem" data-toggle="tooltip"
|
|
||||||
title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
|
||||||
<i class="fas fa-skull"></i>
|
<i class="fas fa-skull"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
@ -452,31 +416,24 @@
|
|||||||
{% elif server['stats']['importing']%}
|
{% elif server['stats']['importing']%}
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-12 px-0">
|
<div class="col-12 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}" class="btn btn-link"><i
|
<a data-id="{{server['server_data']['server_id']}}" class="btn btn-link"><i class="fa fa-spinner fa-spin"></i>
|
||||||
class="fa fa-spinner fa-spin"></i>
|
|
||||||
{{ translate('serverTerm', 'importing', data['lang']) }}</a>
|
{{ translate('serverTerm', 'importing', data['lang']) }}</a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% else %}
|
{% else %}
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-4 px-0">
|
<div class="col-4 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}"
|
<a data-id="{{server['server_data']['server_id']}}" class="btn play_button actions_serveritem" data-toggle="tooltip" title="{{ translate('dashboard', 'start' , data['lang']) }}">
|
||||||
class="btn play_button actions_serveritem" data-toggle="tooltip"
|
|
||||||
title="{{ translate('dashboard', 'start' , data['lang']) }}">
|
|
||||||
<i class="fas fa-play"></i>
|
<i class="fas fa-play"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-4 px-0">
|
<div class="col-4 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}"
|
<a data-id="{{server['server_data']['server_id']}}" class="btn clone_button actions_serveritem" data-toggle="tooltip" title="{{ translate('dashboard', 'clone' , data['lang']) }}">
|
||||||
class="btn clone_button actions_serveritem" data-toggle="tooltip"
|
|
||||||
title="{{ translate('dashboard', 'clone' , data['lang']) }}">
|
|
||||||
<i class="fas fa-clone"></i>
|
<i class="fas fa-clone"></i>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-4 px-0">
|
<div class="col-4 px-0">
|
||||||
<a data-id="{{server['server_data']['server_id']}}"
|
<a data-id="{{server['server_data']['server_id']}}" class="btn kill_button actions_serveritem" data-toggle="tooltip" title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
||||||
class="btn kill_button actions_serveritem" data-toggle="tooltip"
|
|
||||||
title="{{ translate('dashboard', 'kill' , data['lang']) }}">
|
|
||||||
<i class="fas fa-skull"></i></a>
|
<i class="fas fa-skull"></i></a>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -488,15 +445,13 @@
|
|||||||
</h2>
|
</h2>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="collapse-{{server['server_data']['server_id']}}" class="collapse"
|
<div id="collapse-{{server['server_data']['server_id']}}" class="collapse" aria-labelledby="heading-{{server['server_data']['server_id']}}" data-parent="#accordionServers">
|
||||||
aria-labelledby="heading-{{server['server_data']['server_id']}}" data-parent="#accordionServers">
|
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-6">
|
<div class="col-6">
|
||||||
<h6>{{ translate('dashboard', 'cpuUsage', data['lang']) }}</h6>
|
<h6>{{ translate('dashboard', 'cpuUsage', data['lang']) }}</h6>
|
||||||
<div id="m_server_cpu_{{server['server_data']['server_id']}}">
|
<div id="m_server_cpu_{{server['server_data']['server_id']}}">
|
||||||
<div class="progress mb-1" data-toggle="tooltip" data-placement="top"
|
<div class="progress mb-1" data-toggle="tooltip" data-placement="top" title="{{server['stats']['cpu']}}">
|
||||||
title="{{server['stats']['cpu']}}">
|
|
||||||
<div class="progress-bar
|
<div class="progress-bar
|
||||||
{% if server['stats']['cpu'] <= 33 %}
|
{% if server['stats']['cpu'] <= 33 %}
|
||||||
bg-success
|
bg-success
|
||||||
@ -505,8 +460,7 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
bg-danger
|
bg-danger
|
||||||
{% end %}
|
{% end %}
|
||||||
" role="progressbar" style="width: {{server['stats']['cpu']}}%" aria-valuenow="0" aria-valuemin="0"
|
" role="progressbar" style="width: {{server['stats']['cpu']}}%" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100"></div>
|
||||||
aria-valuemax="100"></div>
|
|
||||||
</div>
|
</div>
|
||||||
{{server['stats']['cpu']}}%
|
{{server['stats']['cpu']}}%
|
||||||
</div>
|
</div>
|
||||||
@ -514,8 +468,7 @@
|
|||||||
<div class="col-6">
|
<div class="col-6">
|
||||||
<h6>{{ translate('dashboard', 'memUsage', data['lang']) }}</h6>
|
<h6>{{ translate('dashboard', 'memUsage', data['lang']) }}</h6>
|
||||||
<div draggable="false" id="m_server_mem_{{server['server_data']['server_id']}}">
|
<div draggable="false" id="m_server_mem_{{server['server_data']['server_id']}}">
|
||||||
<div class="progress mb-1" data-toggle="tooltip" data-placement="top"
|
<div class="progress mb-1" data-toggle="tooltip" data-placement="top" title="{{server['stats']['mem']}}">
|
||||||
title="{{server['stats']['mem']}}">
|
|
||||||
<div class="progress-bar
|
<div class="progress-bar
|
||||||
{% if server['stats']['mem_percent'] <= 33 %}
|
{% if server['stats']['mem_percent'] <= 33 %}
|
||||||
bg-success
|
bg-success
|
||||||
@ -524,8 +477,7 @@
|
|||||||
{% else %}
|
{% else %}
|
||||||
bg-danger
|
bg-danger
|
||||||
{% end %}
|
{% end %}
|
||||||
" role="progressbar" style="width: {{server['stats']['mem_percent']}}%" aria-valuenow="0"
|
" role="progressbar" style="width: {{server['stats']['mem_percent']}}%" aria-valuenow="0" aria-valuemin="0" aria-valuemax="100"></div>
|
||||||
aria-valuemin="0" aria-valuemax="100"></div>
|
|
||||||
</div>
|
</div>
|
||||||
{{server['stats']['mem_percent']}}% -
|
{{server['stats']['mem_percent']}}% -
|
||||||
|
|
||||||
@ -554,8 +506,7 @@
|
|||||||
data['lang']) }} <br />
|
data['lang']) }} <br />
|
||||||
|
|
||||||
{% if server['stats']['desc'] != 'False' %}
|
{% if server['stats']['desc'] != 'False' %}
|
||||||
<div id="desc_id"
|
<div id="desc_id" style="overflow-wrap: break-word !important; max-width: 85px !important; overflow: scroll;">
|
||||||
style="overflow-wrap: break-word !important; max-width: 85px !important; overflow: scroll;">
|
|
||||||
{{ server['stats']['desc'] }}</div> <br />
|
{{ server['stats']['desc'] }}</div> <br />
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
||||||
@ -796,6 +747,7 @@
|
|||||||
/* Update Motd */
|
/* Update Motd */
|
||||||
let motd = "";
|
let motd = "";
|
||||||
if (server.desc) {
|
if (server.desc) {
|
||||||
|
m_motd = `<span id="m_input_motd_` + server.id + `" class="input_motd">` + server.desc + `</span>`;
|
||||||
motd = `<span id="input_motd_` + server.id + `" class="input_motd">` + server.desc + `</span>`;
|
motd = `<span id="input_motd_` + server.id + `" class="input_motd">` + server.desc + `</span>`;
|
||||||
m_server_infos = server_infos + '<div id="desc_id" style="word-wrap: break-word; overflow: auto;">' + motd + '</div>' + "<br />";
|
m_server_infos = server_infos + '<div id="desc_id" style="word-wrap: break-word; overflow: auto;">' + motd + '</div>' + "<br />";
|
||||||
server_infos = server_infos + '<div id="desc_id" style="word-wrap: break-word; max-width: 85px !important; overflow: auto;">' + motd + '</div>' + "<br />";
|
server_infos = server_infos + '<div id="desc_id" style="word-wrap: break-word; max-width: 85px !important; overflow: auto;">' + motd + '</div>' + "<br />";
|
||||||
@ -1040,23 +992,28 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
function sendOrder(id_string) {
|
async function sendOrder(id_string) {
|
||||||
const token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
|
let res = await fetch(`/api/v2/users/@me`, {
|
||||||
$.ajax({
|
method: 'PATCH',
|
||||||
type: "PATCH",
|
headers: {
|
||||||
headers: { 'X-XSRFToken': token },
|
'X-XSRFToken': token
|
||||||
url: `/api/v2/users/@me`,
|
},
|
||||||
data: JSON.stringify({
|
body: JSON.stringify({
|
||||||
server_order: id_string,
|
server_order: id_string,
|
||||||
}),
|
}),
|
||||||
success: function (data) {
|
});
|
||||||
console.log("got response:");
|
let responseData = await res.json();
|
||||||
console.log(data);
|
if (responseData.status === "ok") {
|
||||||
},
|
return
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// Inits the sortable
|
// Inits the sortable
|
||||||
$("table#servers_table tbody")
|
$("table#servers_table tbody")
|
||||||
.sortable({
|
.sortable({
|
||||||
|
@ -1,106 +1,103 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<!-- Required meta tags -->
|
||||||
|
<meta charset="utf-8" />
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
|
||||||
|
<title>Crafty Controller</title>
|
||||||
|
<!-- plugins:css -->
|
||||||
|
<link rel="stylesheet" href="/static/assets/vendors/mdi/css/materialdesignicons.min.css" />
|
||||||
|
<link rel="stylesheet" href="/static/assets/vendors/flag-icon-css/css/flag-icon.min.css" />
|
||||||
|
<link rel="stylesheet" href="/static/assets/vendors/ti-icons/css/themify-icons.css" />
|
||||||
|
<link rel="stylesheet" href="/static/assets/vendors/typicons/typicons.css" />
|
||||||
|
<link rel="stylesheet" href="/static/assets/vendors/css/vendor.bundle.base.css" />
|
||||||
|
|
||||||
<head>
|
<meta name="apple-mobile-web-app-capable" content="yes" />
|
||||||
<!-- Required meta tags -->
|
<meta name="mobile-web-app-capable" content="yes" />
|
||||||
<meta charset="utf-8">
|
<meta name="apple-mobile-web-app-status-bar-style" content="black" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
|
<meta name="apple-mobile-web-app-title" content="Crafty" />
|
||||||
<title>Crafty Controller</title>
|
<link rel="apple-touch-icon" href="../static/assets/images/Crafty_4-0.png" />
|
||||||
<!-- plugins:css -->
|
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/mdi/css/materialdesignicons.min.css">
|
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/flag-icon-css/css/flag-icon.min.css">
|
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/ti-icons/css/themify-icons.css">
|
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/typicons/typicons.css">
|
|
||||||
<link rel="stylesheet" href="/static/assets/vendors/css/vendor.bundle.base.css">
|
|
||||||
|
|
||||||
|
<!-- endinject -->
|
||||||
<meta name="apple-mobile-web-app-capable" content="yes">
|
<!-- Plugin css for this page -->
|
||||||
<meta name="mobile-web-app-capable" content="yes" />
|
<!-- End Plugin css for this page -->
|
||||||
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
<!-- Layout styles -->
|
||||||
<meta name="apple-mobile-web-app-title" content="Crafty">
|
<link rel="stylesheet" href="/static/assets/css/dark/style.css" />
|
||||||
<link rel="apple-touch-icon" href="../static/assets/images/Crafty_4-0.png">
|
<!-- End Layout styles -->
|
||||||
|
<link rel="shortcut icon" type="image/svg+xml" href="/static/assets/images/logo_small.svg" />
|
||||||
|
<link rel="alternate icon" href="/static/assets/images/favicon.png" />
|
||||||
|
</head>
|
||||||
|
<style>
|
||||||
|
.auth.auth-bg-1 {
|
||||||
|
background: url("../../static/assets/images/auth/{% raw data['background'] %}"),
|
||||||
|
url("/static/assets/images/auth/login_1.jpg");
|
||||||
|
background-size: cover;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<body class="dark-theme">
|
||||||
<!-- endinject -->
|
<div class="container-scroller">
|
||||||
<!-- Plugin css for this page -->
|
<div class="container-fluid page-body-wrapper full-page-wrapper">
|
||||||
<!-- End Plugin css for this page -->
|
<div class="content-wrapper d-flex align-items-center auth auth-bg-1 theme-one" >
|
||||||
<!-- Layout styles -->
|
<div class="row w-100">
|
||||||
<link rel="stylesheet" href="/static/assets/css/dark/style.css">
|
<div class="col-lg-4 mx-auto">
|
||||||
<!-- End Layout styles -->
|
<div class="auto-form-wrapper">
|
||||||
<link rel="shortcut icon" type="image/svg+xml" href="/static/assets/images/logo_small.svg">
|
<div class="text-center">
|
||||||
<link rel="alternate icon" href="/static/assets/images/favicon.png" />
|
<img alt="Crafty Logo" src="/static/assets/images/logo_long.svg" /><br /><br />
|
||||||
</head>
|
<div class="col-sm-12 grid-margin stretch-card">
|
||||||
<style>
|
<div class="card card-statistics social-card google-card card-colored" >
|
||||||
.auth.auth-bg-1 {
|
<div class="card-body">
|
||||||
background: url("../../static/assets/images/auth/{% raw data['background'] %}"),
|
<h4 class="platform-name mb-3 mt-4 font-weight-semibold user-name" >
|
||||||
url("/static/assets/images/auth/login_1.jpg");
|
{{ translate('accessDenied', 'accessDenied', data['lang']) }}
|
||||||
background-size: cover;
|
</h4>
|
||||||
}
|
<h5 class="headline font-weight-medium">
|
||||||
</style>
|
{{ translate('accessDenied', 'noAccess', data['lang']) }}
|
||||||
|
</h5>
|
||||||
<body class="dark-theme">
|
<p class="mb-2 comment font-weight-light">
|
||||||
<div class="container-scroller">
|
{{ translate('accessDenied', 'contactAdmin',
|
||||||
<div class="container-fluid page-body-wrapper full-page-wrapper">
|
data['lang']) }}<br /><br />
|
||||||
<div class="content-wrapper d-flex align-items-center auth auth-bg-1 theme-one">
|
<a class="d-inline font-weight-medium" href="https://discord.gg/9VJPhCE" > {{ translate('accessDenied', 'contact', data['lang']) }}</a>
|
||||||
<div class="row w-100">
|
</p>
|
||||||
<div class="col-lg-4 mx-auto">
|
</div>
|
||||||
|
|
||||||
<div class="auto-form-wrapper">
|
|
||||||
<div class="text-center">
|
|
||||||
<img src="/static/assets/images/logo_long.svg"><br /><br />
|
|
||||||
<div class="col-sm-12 grid-margin stretch-card">
|
|
||||||
<div class="card card-statistics social-card google-card card-colored">
|
|
||||||
<div class="card-body">
|
|
||||||
<h4 class="platform-name mb-3 mt-4 font-weight-semibold user-name">{{ translate('accessDenied',
|
|
||||||
'accessDenied', data['lang']) }}</h4>
|
|
||||||
<h5 class="headline font-weight-medium">{{ translate('accessDenied', 'noAccess', data['lang']) }}
|
|
||||||
</h5>
|
|
||||||
<p class="mb-2 comment font-weight-light">
|
|
||||||
{{ translate('accessDenied', 'contactAdmin', data['lang']) }}<br /><br />
|
|
||||||
<a class="d-inline font-weight-medium" href="https://discord.gg/9VJPhCE"> {{
|
|
||||||
translate('accessDenied', 'contact', data['lang']) }}</a>
|
|
||||||
</p>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<!-- content-wrapper ends -->
|
||||||
</div>
|
</div>
|
||||||
<!-- content-wrapper ends -->
|
<!-- page-body-wrapper ends -->
|
||||||
</div>
|
</div>
|
||||||
<!-- page-body-wrapper ends -->
|
<!-- container-scroller -->
|
||||||
</div>
|
<!-- plugins:js -->
|
||||||
<!-- container-scroller -->
|
<script src="/static/assets/vendors/js/vendor.bundle.base.js"></script>
|
||||||
<!-- plugins:js -->
|
<!-- endinject -->
|
||||||
<script src="/static/assets/vendors/js/vendor.bundle.base.js"></script>
|
<!-- inject:js -->
|
||||||
<!-- endinject -->
|
<script src="/static/assets/js/shared/off-canvas.js"></script>
|
||||||
<!-- inject:js -->
|
<script src="/static/assets/js/shared/hoverable-collapse.js"></script>
|
||||||
<script src="/static/assets/js/shared/off-canvas.js"></script>
|
<script src="/static/assets/js/shared/misc.js"></script>
|
||||||
<script src="/static/assets/js/shared/hoverable-collapse.js"></script>
|
<script src="/static/assets/js/shared/settings.js"></script>
|
||||||
<script src="/static/assets/js/shared/misc.js"></script>
|
<script src="/static/assets/js/shared/todolist.js"></script>
|
||||||
<script src="/static/assets/js/shared/settings.js"></script>
|
<!-- endinject -->
|
||||||
<script src="/static/assets/js/shared/todolist.js"></script>
|
<script>
|
||||||
<!-- endinject -->
|
$(document).ready(function () {
|
||||||
<script>
|
let login_opacity_div = document.getElementById("login_opacity");
|
||||||
$(document).ready(function () {
|
let opacity = login_opacity_div.getAttribute("data-value");
|
||||||
let login_opacity_div = document.getElementById('login_opacity');
|
document.getElementById("login-form-background").style.background =
|
||||||
let opacity = login_opacity_div.getAttribute('data-value');
|
"rgb(34, 36, 55, " + opacity / 100 + ")";
|
||||||
document.getElementById('login-form-background').style.background = 'rgb(34, 36, 55, ' + (opacity / 100) + ')';
|
//Register Service worker for mobile app
|
||||||
//Register Service worker for mobile app
|
if ("serviceWorker" in navigator) {
|
||||||
if ('serviceWorker' in navigator) {
|
navigator.serviceWorker
|
||||||
navigator.serviceWorker.register('/static/assets/js/shared/service-worker.js', {scope: '/'})
|
.register("/static/assets/js/shared/service-worker.js", {
|
||||||
.then(function (registration) {
|
scope: "/",
|
||||||
console.error('Service Worker Registered');
|
})
|
||||||
});
|
.then(function (registration) {
|
||||||
}
|
console.log("Service Worker Registered");
|
||||||
});
|
});
|
||||||
|
}
|
||||||
</script>
|
});
|
||||||
</body>
|
</script>
|
||||||
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -79,7 +79,7 @@
|
|||||||
<tbody>
|
<tbody>
|
||||||
{% for user in data['users'] %}
|
{% for user in data['users'] %}
|
||||||
<tr>
|
<tr>
|
||||||
<td><i class="fas fa-user"></i> {{ user.username }}</td>
|
<td><i class="fas fa-user"></i><span id="user_{{user.user_id}}">{{ user.username }}</span></td>
|
||||||
<td>
|
<td>
|
||||||
{% if user.enabled %}
|
{% if user.enabled %}
|
||||||
<span class="text-success">
|
<span class="text-success">
|
||||||
@ -106,7 +106,10 @@
|
|||||||
{% end %}
|
{% end %}
|
||||||
</ul>
|
</ul>
|
||||||
</td>
|
</td>
|
||||||
<td><a href="/panel/edit_user?id={{user.user_id}}"><i class="fas fa-pencil-alt"></i></a></td>
|
<td><span data-translate="{{translate('userConfig', 'userName', data['lang'])}}" data-toggle="tooltip" title="{{ translate('userConfig', 'userName', data['lang'])}}" id="username_{{user.user_id}}" class="edit_user clickable" data-name="{{user.username}}" data-id="{{user.user_id}}"><i class="fa-solid fa-user"></i></span>
|
||||||
|
<span data-translate1="{{translate('userConfig', 'password', data['lang'])}}" data-translate2="{{translate('userConfig', 'repeat', data['lang'])}}" data-toggle="tooltip" title="{{ translate('userConfig', 'password', data['lang'])}}" class="edit_password clickable" data-id="{{user.user_id}}"><i class="fa-solid fa-lock"></i></span>
|
||||||
|
<a data-toggle="tooltip" title="{{ translate('userConfig', 'pageTitle', data['lang'])}}" href="/panel/edit_user?id={{user.user_id}}"><i class="fas fa-pencil-alt"></i></a>
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% end %}
|
{% end %}
|
||||||
{% for user in data['managed_users'] %}
|
{% for user in data['managed_users'] %}
|
||||||
@ -138,7 +141,10 @@
|
|||||||
{% end %}
|
{% end %}
|
||||||
</ul>
|
</ul>
|
||||||
</td>
|
</td>
|
||||||
<td><a href="/panel/edit_user?id={{user.user_id}}"><i class="fas fa-pencil-alt"></i></a></td>
|
<td><span data-translate="{{translate('userConfig', 'userName', data['lang'])}}" data-toggle="tooltip" title="{{ translate('userConfig', 'userName', data['lang'])}}" id="username_{{user.user_id}}" class="edit_user clickable" data-name="{{user.username}}" data-id="{{user.user_id}}"><i class="fa-solid fa-user"></i></span>
|
||||||
|
<span data-translate1="{{translate('userConfig', 'password', data['lang'])}}" data-translate2="{{translate('userConfig', 'repeat', data['lang'])}}" data-toggle="tooltip" title="{{ translate('userConfig', 'password', data['lang'])}}" class="edit_password clickable" data-id="{{user.user_id}}"><i class="fa-solid fa-lock"></i></span>
|
||||||
|
<a data-toggle="tooltip" title="{{ translate('userConfig', 'pageTitle', data['lang'])}}" href="/panel/edit_user?id={{user.user_id}}"><i class="fas fa-pencil-alt"></i></a>
|
||||||
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% end %}
|
{% end %}
|
||||||
</tbody>
|
</tbody>
|
||||||
@ -274,6 +280,12 @@
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
.clickable {
|
||||||
|
color: #007bff;
|
||||||
|
}
|
||||||
|
.clickable:hover {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
.custom-picker {
|
.custom-picker {
|
||||||
border: 1px solid var(--outline);
|
border: 1px solid var(--outline);
|
||||||
}
|
}
|
||||||
@ -312,6 +324,99 @@
|
|||||||
|
|
||||||
{% block js %}
|
{% block js %}
|
||||||
<script>
|
<script>
|
||||||
|
function validateForm() {
|
||||||
|
let password0 = document.getElementById("password0").value;
|
||||||
|
let password1 = document.getElementById("password1").value;
|
||||||
|
if (password0 != password1) {
|
||||||
|
$('.passwords-match').popover('show');
|
||||||
|
$('.popover-body').click(function () {
|
||||||
|
$('.passwords-match').popover("hide");
|
||||||
|
});
|
||||||
|
document.body.scrollTop = 0;
|
||||||
|
document.documentElement.scrollTop = 0;
|
||||||
|
$("#password0").css("outline", "1px solid red");
|
||||||
|
$("#password1").css("outline", "1px solid red");
|
||||||
|
return false;
|
||||||
|
} else {
|
||||||
|
return password1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$(".edit_password").on("click", async function(){
|
||||||
|
const token = getCookie("_xsrf");
|
||||||
|
let user_id = $(this).data('id');
|
||||||
|
bootbox.confirm(`<form class="form" id='infos' action=''>\
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="new_password">${$(this).data("translate1")}</label>
|
||||||
|
<input class="form-control" type='password' id="password0" name='new_password' /></br>\
|
||||||
|
</div>
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="confirm_password">${$(this).data("translate2")}</label>
|
||||||
|
<input class="form-control" type='password' id="password1" name='confirm_password' />\
|
||||||
|
</div>
|
||||||
|
</form>`, async function(result) {
|
||||||
|
if(result){
|
||||||
|
password = validateForm();
|
||||||
|
if (!password){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let res = await fetch(`/api/v2/users/${user_id}`, {
|
||||||
|
method: 'PATCH',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: JSON.stringify({"password": password}),
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
console.log(responseData.data)
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
$(document).on("submit", ".bootbox form", function(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
$(".bootbox .btn-primary").click();
|
||||||
|
});
|
||||||
|
|
||||||
|
$(".edit_user").on("click", function(){
|
||||||
|
const token = getCookie("_xsrf");
|
||||||
|
let username = $(this).data('name');
|
||||||
|
let user_id = $(this).data('id');
|
||||||
|
bootbox.confirm(`<form class="form" id='infos' action=''>\
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="username">${$(this).data("translate")}</label>
|
||||||
|
<input class="form-control" type='text' name='username' id="username_field" value=${username} /><br/>\
|
||||||
|
</div>
|
||||||
|
</form>`, async function(result) {
|
||||||
|
if(result){
|
||||||
|
let new_username = $("#username_field").val();
|
||||||
|
let res = await fetch(`/api/v2/users/${user_id}`, {
|
||||||
|
method: 'PATCH',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: JSON.stringify({"username": new_username}),
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
$(`#user_${user_id}`).html(` ${new_username}`)
|
||||||
|
$(`#username_${user_id}`).data('name', new_username);
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
if (webSocket) {
|
if (webSocket) {
|
||||||
webSocket.on('move_status', function (message) {
|
webSocket.on('move_status', function (message) {
|
||||||
if (message === "done") {
|
if (message === "done") {
|
||||||
@ -326,24 +431,30 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
$("#server-path").submit(function (e) {
|
$("#server-path").submit(async function (e) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
$("#submit-status").html('<i class="fa fa-spinner fa-spin"></i>');
|
$("#submit-status").html('<i class="fa fa-spinner fa-spin"></i>');
|
||||||
|
|
||||||
let path = $("#global_server_path").val();
|
let path = $("#global_server_path").val();
|
||||||
let encoded = encodeURIComponent(path);
|
let res = await fetch(`/api/v2/crafty/config/servers_dir`, {
|
||||||
console.log(path)
|
method: 'PATCH',
|
||||||
$.ajax({
|
headers: {
|
||||||
type: "POST",
|
'X-XSRFToken': token
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
dataType: "text",
|
|
||||||
url: '/ajax/update_server_dir',
|
|
||||||
data: {
|
|
||||||
"server_dir": encoded,
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "new_dir": path }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
return
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
|
@ -49,10 +49,7 @@
|
|||||||
</ul>
|
</ul>
|
||||||
<div class="">
|
<div class="">
|
||||||
<div class="">
|
<div class="">
|
||||||
<form id="role_form" class="forms-sample" method="post" action="{{ '/panel/add_role' if data['new_role'] else '/panel/edit_role' }}">
|
<form id="role_form" class="forms-sample">
|
||||||
{% raw xsrf_form_html() %}
|
|
||||||
<input type="hidden" name="id" value="{{ data['role']['role_id'] }}">
|
|
||||||
<input type="hidden" name="subpage" value="config">
|
|
||||||
|
|
||||||
<div class="card">
|
<div class="card">
|
||||||
<div class="card-header header-sm d-flex justify-content-between align-items-center">
|
<div class="card-header header-sm d-flex justify-content-between align-items-center">
|
||||||
@ -61,7 +58,7 @@
|
|||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label for="role_name">{{ translate('rolesConfig', 'roleName', data['lang']) }} <small class="text-muted ml-1"> - {{ translate('rolesConfig', 'roleDesc', data['lang']) }}</small> </label>
|
<label for="role_name">{{ translate('rolesConfig', 'roleName', data['lang']) }} <small class="text-muted ml-1"> - {{ translate('rolesConfig', 'roleDesc', data['lang']) }}</small> </label>
|
||||||
<input type="text" class="form-control" name="role_name" id="role_name" value="{{ data['role']['role_name'] }}" placeholder="Role Name" >
|
<input type="text" class="form-control" name="name" id="role_name" value="{{ data['role']['role_name'] }}" placeholder="Role Name" >
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<br />
|
<br />
|
||||||
@ -188,11 +185,11 @@
|
|||||||
<tr>
|
<tr>
|
||||||
<td>{{ server['server_name'] }}</td>
|
<td>{{ server['server_name'] }}</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="checkbox" class="" onclick="enable_disable(event)" data-id="{{server['server_id']}}"
|
<input type="checkbox" class="access" onclick="enable_disable(event)" data-id="{{server['server_id']}}"
|
||||||
id="server_{{ server['server_id'] }}_access"
|
id="server_{{ server['server_id'] }}_access"
|
||||||
name="server_{{ server['server_id'] }}_access"
|
name="server_{{ server['server_id'] }}_access"
|
||||||
{{ 'checked' if server['server_id'] in data['role']['servers'] else '' }}
|
{{ 'checked' if server['server_id'] in data['role']['servers'] else '' }}
|
||||||
autocomplete="off" value="1">
|
autocomplete="off" value="1" form="dummy">
|
||||||
</td>
|
</td>
|
||||||
{% for permission in data['permissions_all'] %}
|
{% for permission in data['permissions_all'] %}
|
||||||
{% if server['server_id'] in data['role']['servers'] %}
|
{% if server['server_id'] in data['role']['servers'] %}
|
||||||
@ -201,14 +198,14 @@
|
|||||||
id="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
id="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
||||||
name="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
name="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
||||||
{{ 'checked' if permission in data['permissions_dict'].get(server['server_id'], []) else '' }}
|
{{ 'checked' if permission in data['permissions_dict'].get(server['server_id'], []) else '' }}
|
||||||
autocomplete="off" value="1">
|
autocomplete="off" value="1" form="dummy">
|
||||||
</td>
|
</td>
|
||||||
{% else %}
|
{% else %}
|
||||||
<td>
|
<td>
|
||||||
<input type="checkbox" class="{{server['server_id']}}_perms"
|
<input type="checkbox" class="{{server['server_id']}}_perms"
|
||||||
id="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
id="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
||||||
name="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
name="permission_{{ server['server_id'] }}_{{ permission.name }}"
|
||||||
autocomplete="off" value="1" disabled>
|
autocomplete="off" value="1" disabled form="dummy">
|
||||||
</td>
|
</td>
|
||||||
{% end %}
|
{% end %}
|
||||||
{% end %}
|
{% end %}
|
||||||
@ -284,7 +281,7 @@
|
|||||||
<a class="btn btn-sm btn-danger disabled"><i class="fas fa-trash"></i>{{ translate('rolesConfig', 'delRole', data['lang']) }}</a><br />
|
<a class="btn btn-sm btn-danger disabled"><i class="fas fa-trash"></i>{{ translate('rolesConfig', 'delRole', data['lang']) }}</a><br />
|
||||||
<small>{{ translate('rolesConfig', 'doesNotExist', data['lang']) }}</small>
|
<small>{{ translate('rolesConfig', 'doesNotExist', data['lang']) }}</small>
|
||||||
{% else %}
|
{% else %}
|
||||||
<a href="/panel/remove_role?id={{ data['role']['role_id'] }}" class="btn btn-sm btn-danger"><i class="fas fa-trash"></i>{{ translate('rolesConfig', 'delRole', data['lang']) }}</a>
|
<button onclick="del_role()" class="btn btn-sm btn-danger"><i class="fas fa-trash"></i>{{ translate('rolesConfig', 'delRole', data['lang']) }}</button>
|
||||||
{% end %}
|
{% end %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -341,24 +338,87 @@
|
|||||||
console.log( "ready!" );
|
console.log( "ready!" );
|
||||||
});
|
});
|
||||||
const roleId = new URLSearchParams(document.location.search).get('id');
|
const roleId = new URLSearchParams(document.location.search).get('id');
|
||||||
|
|
||||||
|
function replacer(key, value) {
|
||||||
|
if (key === "permissions"){
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
if (key === "servers" && value.length === 0){
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
if (typeof value == "boolean") {
|
||||||
|
console.log(value);
|
||||||
|
return value
|
||||||
|
} else {
|
||||||
|
return (isNaN(value) ? value : +value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$("#config_form").on("submit", async function (e) {
|
async function del_role(){
|
||||||
|
const token = getCookie("_xsrf")
|
||||||
|
let res = await fetch(`/api/v2/roles/${roleId}`, {
|
||||||
|
method: "DELETE",
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.href = "/panel/panel_config";
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$("#role_form").on("submit", async function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
let configForm = document.getElementById("config_form");
|
let roleForm = document.getElementById("role_form");
|
||||||
|
|
||||||
let formData = new FormData(configForm);
|
let server_ids = $('.access').map(function() {
|
||||||
|
if ($(this).is(':checked')){
|
||||||
|
return $(this).data('id');
|
||||||
|
}
|
||||||
|
}).get();
|
||||||
|
|
||||||
|
let servers = []
|
||||||
|
for(i=0; i < server_ids.length; i++){
|
||||||
|
let arrchecked = $(`.${server_ids[i]}_perms`).map(function() {
|
||||||
|
if(this.checked){
|
||||||
|
return "1";
|
||||||
|
}else{
|
||||||
|
return "0"
|
||||||
|
}
|
||||||
|
}).get();
|
||||||
|
servers.push({"server_id": server_ids[i], "permissions": arrchecked.join("")});
|
||||||
|
}
|
||||||
|
console.log(servers)
|
||||||
|
|
||||||
|
let formData = new FormData(roleForm);
|
||||||
//Create an object from the form data entries
|
//Create an object from the form data entries
|
||||||
let formDataObject = Object.fromEntries(formData.entries());
|
let formDataObject = Object.fromEntries(formData.entries());
|
||||||
let send_object = Object()
|
formDataObject.servers = servers;
|
||||||
send_object.servers = []
|
console.log(formDataObject);
|
||||||
send_object.name = formDataObject.role_name
|
|
||||||
|
//We need to make sure these are sent regardless of whether or not they're checked
|
||||||
|
|
||||||
// Format the plain form data as JSON
|
// Format the plain form data as JSON
|
||||||
let formDataJsonString = JSON.stringify(formDataObject, replacer);
|
let formDataJsonString = JSON.stringify(formDataObject, replacer);
|
||||||
|
|
||||||
let res = await fetch(`/api/v2/roles/${roleId}`, {
|
console.log(formDataJsonString);
|
||||||
method: 'PATCH',
|
|
||||||
|
let url = `/api/v2/roles/`
|
||||||
|
let method = 'POST'
|
||||||
|
if (roleId){
|
||||||
|
url = `/api/v2/roles/${roleId}`
|
||||||
|
method = 'PATCH'
|
||||||
|
}
|
||||||
|
let res = await fetch(url, {
|
||||||
|
method: method,
|
||||||
headers: {
|
headers: {
|
||||||
'X-XSRFToken': token
|
'X-XSRFToken': token
|
||||||
},
|
},
|
||||||
@ -366,14 +426,14 @@
|
|||||||
});
|
});
|
||||||
let responseData = await res.json();
|
let responseData = await res.json();
|
||||||
if (responseData.status === "ok") {
|
if (responseData.status === "ok") {
|
||||||
window.location.reload();
|
window.location.href = "/panel/panel_config";
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
bootbox.alert({
|
bootbox.alert({
|
||||||
title: responseData.error,
|
title: responseData.error,
|
||||||
message: responseData.error_data
|
message: responseData.error_data
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
@ -58,13 +58,11 @@ data['lang']) }}{% end %}
|
|||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-md-6 col-sm-12">
|
<div class="col-md-6 col-sm-12">
|
||||||
{% if data['new_user'] %}
|
{% if data['new_user'] %}
|
||||||
<form id="user_form" class="forms-sample" method="post" action="/panel/add_user">
|
<form id="user_form" class="forms-sample">
|
||||||
{% else %}
|
{% else %}
|
||||||
<form id="user_form" class="forms-sample" method="post" action="/panel/edit_user">
|
<form id="user_form" class="forms-sample">
|
||||||
{% end %}
|
{% end %}
|
||||||
{% raw xsrf_form_html() %}
|
|
||||||
<input type="hidden" name="id" value="{{ data['user']['user_id'] }}">
|
|
||||||
<input type="hidden" name="subpage" value="config">
|
|
||||||
|
|
||||||
|
|
||||||
<div class="card">
|
<div class="card">
|
||||||
@ -73,6 +71,7 @@ data['lang']) }}{% end %}
|
|||||||
data['lang']) }}</h4>
|
data['lang']) }}</h4>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
|
{% if data['new_user'] %}
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="form-label" for="username">{{ translate('userConfig', 'userName', data['lang'])
|
<label class="form-label" for="username">{{ translate('userConfig', 'userName', data['lang'])
|
||||||
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'userNameDesc', data['lang'])
|
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'userNameDesc', data['lang'])
|
||||||
@ -85,7 +84,7 @@ data['lang']) }}{% end %}
|
|||||||
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'leaveBlank', data['lang']) }}
|
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'leaveBlank', data['lang']) }}
|
||||||
</small> </label>
|
</small> </label>
|
||||||
<input type="password" class="form-control" name="password0" id="password0" value=""
|
<input type="password" class="form-control" name="password0" id="password0" value=""
|
||||||
autocomplete="new-password" data-lpignore="true" placeholder="Password">
|
autocomplete="new-password" data-lpignore="true" placeholder="Password" form="dummy">
|
||||||
<span class="passwords-match" ,
|
<span class="passwords-match" ,
|
||||||
data-content="{{ translate('panelConfig', 'match', data['lang']) }}" ,
|
data-content="{{ translate('panelConfig', 'match', data['lang']) }}" ,
|
||||||
data-placement="right"></span>
|
data-placement="right"></span>
|
||||||
@ -95,11 +94,20 @@ data['lang']) }}{% end %}
|
|||||||
<small class="text-muted ml-1"> - {{ translate('userConfig', 'leaveBlank', data['lang'])
|
<small class="text-muted ml-1"> - {{ translate('userConfig', 'leaveBlank', data['lang'])
|
||||||
}}</small> </label>
|
}}</small> </label>
|
||||||
<input type="password" class="form-control" name="password1" id="password1" value=""
|
<input type="password" class="form-control" name="password1" id="password1" value=""
|
||||||
autocomplete="new-password" data-lpignore="true" placeholder="Repeat Password">
|
autocomplete="new-password" data-lpignore="true" placeholder="Repeat Password" form="dummy">
|
||||||
<span class="passwords-match" ,
|
<span class="passwords-match" ,
|
||||||
data-content="{{ translate('panelConfig', 'match', data['lang']) }}" ,
|
data-content="{{ translate('panelConfig', 'match', data['lang']) }}" ,
|
||||||
data-placement="right"></span>
|
data-placement="right"></span>
|
||||||
</div>
|
</div>
|
||||||
|
{% else %}
|
||||||
|
<div class="form-group">
|
||||||
|
<label class="form-label" for="username">{{ translate('userConfig', 'userName', data['lang'])
|
||||||
|
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'userNameDesc', data['lang'])
|
||||||
|
}}</small> </label>
|
||||||
|
<input type="text" class="form-control" name="username" id="username" autocomplete="off"
|
||||||
|
data-lpignore="true" value="{{ data['user']['username'] }}" placeholder="User Name" disabled>
|
||||||
|
</div>
|
||||||
|
{% end %}
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="form-label" for="email">{{ translate('userConfig', 'gravEmail', data['lang'])
|
<label class="form-label" for="email">{{ translate('userConfig', 'gravEmail', data['lang'])
|
||||||
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'gravDesc', data['lang'])
|
}}<small class="text-muted ml-1"> - {{ translate('userConfig', 'gravDesc', data['lang'])
|
||||||
@ -111,7 +119,7 @@ data['lang']) }}{% end %}
|
|||||||
<label class="form-label" for="language">{{ translate('userConfig', 'userLang', data['lang'])
|
<label class="form-label" for="language">{{ translate('userConfig', 'userLang', data['lang'])
|
||||||
}}</label>
|
}}</label>
|
||||||
<select class="form-select form-control form-control-lg select-css" id="language"
|
<select class="form-select form-control form-control-lg select-css" id="language"
|
||||||
name="language" form="user_form">
|
name="lang" form="user_form">
|
||||||
{% for lang in data['languages'] %}
|
{% for lang in data['languages'] %}
|
||||||
{% if not 'incomplete' in lang %}
|
{% if not 'incomplete' in lang %}
|
||||||
<option value="{{lang}}">{{lang}}</option>
|
<option value="{{lang}}">{{lang}}</option>
|
||||||
@ -182,18 +190,18 @@ data['lang']) }}{% end %}
|
|||||||
<td>
|
<td>
|
||||||
{% if role.role_id in data['user']['roles'] %}
|
{% if role.role_id in data['user']['roles'] %}
|
||||||
{% if role.manager == data['exec_user'] or data['superuser'] %}
|
{% if role.manager == data['exec_user'] or data['superuser'] %}
|
||||||
<input type="checkbox" class="form-check-input"
|
<input type="checkbox" class="form-check-input role_check"
|
||||||
id="role_{{ role.role_id }}_membership" name="role_{{ role.role_id }}_membership"
|
id="role_{{ role.role_id }}_membership" name="role_{{ role.role_id }}_membership"
|
||||||
checked="" value="1">
|
checked="" value="{{role.role_id}}" form="dummy">
|
||||||
{% else %}
|
{% else %}
|
||||||
<input type="checkbox" class="form-check-input"
|
<input type="checkbox" class="form-check-input role_check"
|
||||||
id="role_{{ role.role_id }}_membership" name="role_{{ role.role_id }}_membership"
|
id="role_{{ role.role_id }}_membership" name="role_{{ role.role_id }}_membership"
|
||||||
checked="" value="1" disabled>
|
checked="" value="{{role.role_id}}" disabled form="dummy">
|
||||||
{% end %}
|
{% end %}
|
||||||
{% elif data['superuser'] or role.manager == data['exec_user'] %}
|
{% elif data['superuser'] or role.manager == data['exec_user'] %}
|
||||||
<input type="checkbox" class="form-check-input"
|
<input type="checkbox" class="form-check-input role_check"
|
||||||
id="role_{{ role.role_id }}_membership" name="role_{{ role.role_id }}_membership"
|
id="role_{{ role.role_id }}_membership" name="role_{{ role.role_id }}_membership"
|
||||||
value="1">
|
value="{{role.role_id}}" form="dummy">
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
||||||
</td>
|
</td>
|
||||||
@ -219,7 +227,7 @@ data['lang']) }}{% end %}
|
|||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<div class="table-responsive">
|
<div class="table-responsive">
|
||||||
<table class="table table-hover">
|
<table id="permissions" aria-describedby="User Crafty Permissions" class="table table-hover">
|
||||||
<thead>
|
<thead>
|
||||||
<tr class="rounded">
|
<tr class="rounded">
|
||||||
<th>{{ translate('userConfig', 'permName', data['lang']) }}</th>
|
<th>{{ translate('userConfig', 'permName', data['lang']) }}</th>
|
||||||
@ -233,16 +241,16 @@ data['lang']) }}{% end %}
|
|||||||
<td>{{ permission.name }}</td>
|
<td>{{ permission.name }}</td>
|
||||||
<td>
|
<td>
|
||||||
{% if permission in data['permissions_list'] %}
|
{% if permission in data['permissions_list'] %}
|
||||||
<input type="checkbox" class="form-check-input" id="permission_{{ permission.name }}"
|
<input type="checkbox" class="form-check-input perm-name" id="permission_{{ permission.name }}"
|
||||||
name="permission_{{ permission.name }}" checked="" value="1">
|
name="permission_{{ permission.name }}" checked="" value="1" data-perm="{{permission.name}}" form="dummy">
|
||||||
{% else %}
|
{% else %}
|
||||||
<input type="checkbox" class="form-check-input" id="permission_{{ permission.name }}"
|
<input type="checkbox" class="form-check-input perm-name" id="permission_{{ permission.name }}"
|
||||||
name="permission_{{ permission.name }}" value="1">
|
name="permission_{{ permission.name }}" value="1" data-perm="{{permission.name}}" form="dummy">
|
||||||
{% end %}
|
{% end %}
|
||||||
</td>
|
</td>
|
||||||
<td><input type="text" class="form-control" name="quantity_{{ permission.name }}"
|
<td><input type="text" class="form-control" name="quantity_{{ permission.name }}"
|
||||||
id="quantity_{{ permission.name }}"
|
id="quantity_{{ permission.name }}"
|
||||||
value="{{ data['quantity_server'][permission.name] }}"></td>
|
value="{{ data['quantity_server'][permission.name] }}" data-perm="{{permission.name}}" form="dummy"></td>
|
||||||
</tr>
|
</tr>
|
||||||
{% end %}
|
{% end %}
|
||||||
</tbody>
|
</tbody>
|
||||||
@ -287,7 +295,7 @@ data['lang']) }}{% end %}
|
|||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<button class="btn btn-success mr-2" onclick="submit_user(event);"><i class="fas fa-save"></i> {{
|
<button type="submit" class="btn btn-success mr-2"><i class="fas fa-save"></i> {{
|
||||||
translate('panelConfig', 'save', data['lang']) }}</button>
|
translate('panelConfig', 'save', data['lang']) }}</button>
|
||||||
<button type="reset" onclick="location.href='/panel/panel_config'" class="btn btn-light"><i
|
<button type="reset" onclick="location.href='/panel/panel_config'" class="btn btn-light"><i
|
||||||
class="fas fa-undo-alt"></i> {{ translate('panelConfig', 'cancel', data['lang']) }}</button>
|
class="fas fa-undo-alt"></i> {{ translate('panelConfig', 'cancel', data['lang']) }}</button>
|
||||||
@ -363,9 +371,12 @@ data['lang']) }}{% end %}
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
function validateForm() {
|
function validateForm() {
|
||||||
let password0 = document.getElementById("password0").value
|
let password0 = document.getElementById("password0").value;
|
||||||
let password1 = document.getElementById("password1").value
|
let password1 = document.getElementById("password1").value;
|
||||||
if (password0 != password1) {
|
if (password0 === "" && password1 === "" && userId){
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
else if (password0 != password1) {
|
||||||
$('.passwords-match').popover('show');
|
$('.passwords-match').popover('show');
|
||||||
$('.popover-body').click(function () {
|
$('.popover-body').click(function () {
|
||||||
$('.passwords-match').popover("hide");
|
$('.passwords-match').popover("hide");
|
||||||
@ -376,11 +387,133 @@ data['lang']) }}{% end %}
|
|||||||
$("#password1").css("outline", "1px solid red");
|
$("#password1").css("outline", "1px solid red");
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
return true;
|
return password1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const userId = new URLSearchParams(document.location.search).get('id')
|
function replacer(key, value) {
|
||||||
|
if (typeof value == "boolean" || key === "email" || key === "permissions" || key === "roles") {
|
||||||
|
return value
|
||||||
|
} else {
|
||||||
|
console.log(key, value)
|
||||||
|
return (isNaN(value) ? value : +value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$("#user_form").on("submit", async function (e) {
|
||||||
|
const userId = new URLSearchParams(document.location.search).get('id');
|
||||||
|
console.log(userId)
|
||||||
|
e.preventDefault();
|
||||||
|
let password = null;
|
||||||
|
if(!userId){
|
||||||
|
password = validateForm();
|
||||||
|
if (!password){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const token = getCookie("_xsrf")
|
||||||
|
|
||||||
|
let userRes = await fetch(`/api/v2/users/@me`, {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
});
|
||||||
|
let userData = await userRes.json();
|
||||||
|
let superuser = null;
|
||||||
|
if (userData.status === "ok") {
|
||||||
|
superuser = userData.data["superuser"];
|
||||||
|
edit_id = userData.data["user_id"];
|
||||||
|
} else {
|
||||||
|
bootbox.alert({
|
||||||
|
title: userData.error,
|
||||||
|
message: userData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
let userForm = document.getElementById("user_form");
|
||||||
|
|
||||||
|
let disabled_flag = false;
|
||||||
|
let roles = null;
|
||||||
|
if (superuser || userId != edit_id){
|
||||||
|
roles = $('.role_check').map(function() {
|
||||||
|
if ($(this).attr("disabled")){
|
||||||
|
disabled_flag = true;
|
||||||
|
}
|
||||||
|
if ($(this).is(':checked')){
|
||||||
|
return $(this).val();
|
||||||
|
}
|
||||||
|
}).get();
|
||||||
|
let avail_permissions = $('.perm-name').map(function() {
|
||||||
|
return $(this).data("perm");
|
||||||
|
}).get();
|
||||||
|
|
||||||
|
permissions = []
|
||||||
|
for(i=0; i < avail_permissions.length; i++){
|
||||||
|
permissions.push({"name": avail_permissions[i], "quantity": $(`#quantity_${avail_permissions[i]}`).val(), "enabled": $(`#permission_${avail_permissions[i]}`).is(':checked')})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let formData = new FormData(userForm);
|
||||||
|
//Create an object from the form data entries
|
||||||
|
let formDataObject = Object.fromEntries(formData.entries());
|
||||||
|
if(userId){
|
||||||
|
delete formDataObject.username
|
||||||
|
}
|
||||||
|
if (superuser || userId != edit_id){
|
||||||
|
if (!disabled_flag){
|
||||||
|
formDataObject.roles = roles;
|
||||||
|
}
|
||||||
|
if ($("#permissions").length){
|
||||||
|
formDataObject.permissions = permissions;
|
||||||
|
}
|
||||||
|
if(!userId){
|
||||||
|
if(typeof password === "string"){
|
||||||
|
formDataObject.password = password;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
formDataObject.enabled = $("#enabled").is(":checked");
|
||||||
|
if ($("#superuser").is(":enabled")){
|
||||||
|
formDataObject.superuser = $("#superuser").is(":checked");
|
||||||
|
}
|
||||||
|
formDataObject.hints = $("#hints").is(":checked");
|
||||||
|
console.log(formDataObject);
|
||||||
|
|
||||||
|
//We need to make sure these are sent regardless of whether or not they're checked
|
||||||
|
|
||||||
|
// Format the plain form data as JSON
|
||||||
|
let formDataJsonString = JSON.stringify(formDataObject, replacer);
|
||||||
|
|
||||||
|
console.log(formDataJsonString);
|
||||||
|
if (userId){
|
||||||
|
url = `/api/v2/users/${userId}`
|
||||||
|
method = 'PATCH'
|
||||||
|
}else{
|
||||||
|
url = `/api/v2/users/`
|
||||||
|
method = 'POST'
|
||||||
|
}
|
||||||
|
let res = await fetch(url, {
|
||||||
|
method: method,
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: formDataJsonString,
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.href = "/panel/panel_config";
|
||||||
|
} else {
|
||||||
|
if (responseData.hasOwnProperty("error_data")){
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}else{
|
||||||
|
bootbox.alert(responseData.error
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
$(".delete-user").click(function () {
|
$(".delete-user").click(function () {
|
||||||
var file_to_del = $(this).data("file");
|
var file_to_del = $(this).data("file");
|
||||||
|
|
||||||
@ -398,10 +531,26 @@ data['lang']) }}{% end %}
|
|||||||
label: '<i class="fas fa-check"></i> {{ translate("serverBackups", "confirm", data['lang']) }}'
|
label: '<i class="fas fa-check"></i> {{ translate("serverBackups", "confirm", data['lang']) }}'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
callback: function (result) {
|
callback: async function (result) {
|
||||||
console.log(result);
|
console.log(result);
|
||||||
if (result === true) {
|
if (result === true) {
|
||||||
location.href = "/panel/remove_user?id=" + userId;
|
const token = getCookie("_xsrf")
|
||||||
|
let res = await fetch(`/api/v2/users/${userId}`, {
|
||||||
|
method: "DELETE",
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.href = "/panel/panel_config";
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -86,17 +86,14 @@
|
|||||||
apikey.server_permissions }}
|
apikey.server_permissions }}
|
||||||
{{ translate('apiKeys', 'crafty', data['lang']) }} {{
|
{{ translate('apiKeys', 'crafty', data['lang']) }} {{
|
||||||
apikey.crafty_permissions }}</td>
|
apikey.crafty_permissions }}</td>
|
||||||
<td>
|
<td><button class="btn btn-danger delete-api-key"
|
||||||
<button class="btn btn-danger delete-api-key"
|
|
||||||
data-key-id="{{ apikey.token_id }}"
|
data-key-id="{{ apikey.token_id }}"
|
||||||
data-key-name="{{ apikey.name }}">{{
|
data-key-name="{{ apikey.name }}">{{translate('panelConfig',
|
||||||
translate('panelConfig', 'delete', data['lang'])
|
'delete', data['lang'])}}</button>
|
||||||
}}</button>
|
|
||||||
<button class="btn btn-outline-primary get-a-token"
|
<button class="btn btn-outline-primary get-a-token"
|
||||||
data-key-id="{{ apikey.token_id }}"
|
data-key-id="{{ apikey.token_id }}"
|
||||||
data-key-name="{{ apikey.name }}">{{
|
data-key-name="{{ apikey.name }}">{{translate('apiKeys',
|
||||||
translate('apiKeys', 'getToken', data['lang']) }}
|
'getToken', data['lang'])}}</button>
|
||||||
</button>
|
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% end %}
|
{% end %}
|
||||||
@ -115,10 +112,7 @@
|
|||||||
'createNew', data['lang']) }}</h4>
|
'createNew', data['lang']) }}</h4>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<form id="user_form" class="forms-sample" method="post"
|
<form id="user_api_form" class="forms-sample">
|
||||||
action="/panel/edit_user_apikeys">
|
|
||||||
{% raw xsrf_form_html() %}
|
|
||||||
<input type="hidden" name="id" value="{{ data['user']['user_id'] }}">
|
|
||||||
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
<label class="form-label" for="username">{{ translate('apiKeys', 'name',
|
<label class="form-label" for="username">{{ translate('apiKeys', 'name',
|
||||||
@ -142,7 +136,7 @@
|
|||||||
}}</label>
|
}}</label>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="checkbox" class=""
|
<input type="checkbox" class="server_perm"
|
||||||
id="permission_{{ permission.name }}"
|
id="permission_{{ permission.name }}"
|
||||||
name="permission_{{ permission.name }}" value="1">
|
name="permission_{{ permission.name }}" value="1">
|
||||||
</td>
|
</td>
|
||||||
@ -154,7 +148,7 @@
|
|||||||
}}</label>
|
}}</label>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<input type="checkbox" class=""
|
<input type="checkbox" class="crafty_perm"
|
||||||
id="permission_{{ permission.name }}"
|
id="permission_{{ permission.name }}"
|
||||||
name="permission_{{ permission.name }}" value="1">
|
name="permission_{{ permission.name }}" value="1">
|
||||||
</td>
|
</td>
|
||||||
@ -201,56 +195,122 @@
|
|||||||
var r = document.cookie.match("\\b" + name + "=([^;]*)\\b");
|
var r = document.cookie.match("\\b" + name + "=([^;]*)\\b");
|
||||||
return r ? r[1] : undefined;
|
return r ? r[1] : undefined;
|
||||||
}
|
}
|
||||||
|
const userId = new URLSearchParams(document.location.search).get('id')
|
||||||
|
$(document).ready(function () {
|
||||||
|
$("#user_api_form").on("submit", async function (e) {
|
||||||
|
e.preventDefault();
|
||||||
|
const token = getCookie("_xsrf")
|
||||||
|
let apiForm = document.getElementById("user_api_form");
|
||||||
|
|
||||||
|
let formData = new FormData(apiForm);
|
||||||
|
|
||||||
|
//Create an object from the form data entries
|
||||||
|
let formDataObject = Object.fromEntries(formData.entries());
|
||||||
|
//We need to make sure these are sent regardless of whether or not they're checked
|
||||||
|
formDataObject.disabled_language_files = $('#lang_select').val();
|
||||||
|
$('#user_api_form input[type="checkbox"]:checked').each(function () {
|
||||||
|
if ($(this).val() == 'True') {
|
||||||
|
formDataObject[this.name] = true;
|
||||||
|
} else {
|
||||||
|
formDataObject[this.name] = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
let server_permissions = $('.server_perm').map(function () {
|
||||||
|
if (this.checked) {
|
||||||
|
return "1";
|
||||||
|
} else {
|
||||||
|
return "0"
|
||||||
|
}
|
||||||
|
}).get();
|
||||||
|
server_permissions = server_permissions.join("");
|
||||||
|
|
||||||
|
let crafty_permissions = $('.crafty_perm').map(function () {
|
||||||
|
if (this.checked) {
|
||||||
|
return "1";
|
||||||
|
} else {
|
||||||
|
return "0"
|
||||||
|
}
|
||||||
|
}).get();
|
||||||
|
crafty_permissions = crafty_permissions.join("");
|
||||||
|
console.log(server_permissions);
|
||||||
|
console.log(crafty_permissions);
|
||||||
|
console.log(formDataObject);
|
||||||
|
// Format the plain form data as JSON
|
||||||
|
let formDataJsonString = JSON.stringify({
|
||||||
|
"name": formDataObject.name,
|
||||||
|
"server_permissions_mask": server_permissions,
|
||||||
|
"crafty_permissions_mask": crafty_permissions,
|
||||||
|
"superuser": $("#superuser").prop('checked'),
|
||||||
|
});
|
||||||
|
console.log(formDataJsonString);
|
||||||
|
|
||||||
|
let res = await fetch(`/api/v2/users/${userId}/key/`, {
|
||||||
|
method: 'PATCH',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: formDataJsonString,
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
location.reload();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
console.log("ready!");
|
console.log("ready!");
|
||||||
$('.delete-api-key').click(function () {
|
$('.delete-api-key').click(async function () {
|
||||||
var keyId = $(this).data("key-id");
|
let keyId = $(this).data("key-id");
|
||||||
var keyName = $(this).data("key-name");
|
let token = getCookie("_xsrf");
|
||||||
bootbox.confirm({
|
let res = await fetch(`/api/v2/users/${userId}/key/${keyId}`, {
|
||||||
title: `Remove API key ${keyName}?`,
|
method: 'DELETE',
|
||||||
message: "Do you want to delete this API key? This cannot be undone.",
|
headers: {
|
||||||
buttons: {
|
'X-XSRFToken': token
|
||||||
cancel: {
|
|
||||||
label: '<i class="fas fa-times"></i> {{ translate("panelConfig", "cancel", data['lang']) }}'
|
|
||||||
},
|
|
||||||
confirm: {
|
|
||||||
label: '<i class="fas fa-check"></i> {{ translate("serverBackups", "confirm", data['lang']) }}'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
callback: function (result) {
|
|
||||||
if (result) {
|
|
||||||
var token = getCookie("_xsrf")
|
|
||||||
$.ajax({
|
|
||||||
type: "DELETE",
|
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
url: '/panel/remove_apikey?id=' + keyId,
|
|
||||||
success: function (data) {
|
|
||||||
location.reload();
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
})
|
|
||||||
$('.get-a-token').click(function () {
|
|
||||||
var keyId = $(this).data("key-id");
|
|
||||||
var keyName = $(this).data("key-name");
|
|
||||||
var token = getCookie("_xsrf")
|
|
||||||
$.ajax({
|
|
||||||
type: "POST",
|
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
url: '/panel/get_token?id=' + keyId,
|
|
||||||
success: function (data) {
|
|
||||||
bootbox.alert({
|
|
||||||
title: `API token for ${keyName}`,
|
|
||||||
message: `Here is an API token for ${keyName}:\n<pre style="white-space: pre-wrap;color:white;word-break:break-all;background: grey;border-radius: 5px;">${data}</pre>`
|
|
||||||
});
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
})
|
let responseData = await res.json();
|
||||||
});
|
if (responseData.status === "ok") {
|
||||||
|
location.reload()
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
$('.get-a-token').click(async function () {
|
||||||
|
let keyId = $(this).data("key-id");
|
||||||
|
let keyName = $(this).data("key-name");
|
||||||
|
let token = getCookie("_xsrf");
|
||||||
|
let res = await fetch(`/api/v2/users/${userId}/key/${keyId}`, {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
bootbox.alert({
|
||||||
|
title: `API token for ${keyName}`,
|
||||||
|
message: `Here is an API token for ${keyName}:\n<pre style="white-space: pre-wrap;color:white;word-break:break-all;background: grey;border-radius: 5px;">${responseData.data}</pre>`
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
@ -31,6 +31,9 @@
|
|||||||
<a class="dropdown-item {% if data['active_link'] == 'admin_controls' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=admin_controls" role="tab" aria-selected="true"><i class="fas fa-users"></i> {{ translate('serverDetails', 'playerControls', data['lang']) }}</a>
|
<a class="dropdown-item {% if data['active_link'] == 'admin_controls' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=admin_controls" role="tab" aria-selected="true"><i class="fas fa-users"></i> {{ translate('serverDetails', 'playerControls', data['lang']) }}</a>
|
||||||
{% end %}
|
{% end %}
|
||||||
<a class="dropdown-item {% if data['active_link'] == 'metrics' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=metrics" role="tab" aria-selected="true"><i class="fa-solid fa-chart-line"></i> {{ translate('serverDetails', 'metrics', data['lang']) }}</a>
|
<a class="dropdown-item {% if data['active_link'] == 'metrics' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=metrics" role="tab" aria-selected="true"><i class="fa-solid fa-chart-line"></i> {{ translate('serverDetails', 'metrics', data['lang']) }}</a>
|
||||||
|
{% if data['permissions']['Config'] in data['user_permissions'] %}
|
||||||
|
<a class="dropdown-item {% if data['active_link'] == 'webhooks' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=webhooks" role="tab" aria-selected="true"><i class="fa-regular fa-bell"></i>{{ translate('webhooks', 'webhooks', data['lang']) }}</a>
|
||||||
|
{% end %}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
@ -53,4 +53,10 @@
|
|||||||
<a class="nav-link {% if data['active_link'] == 'metrics' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=metrics" role="tab" aria-selected="true">
|
<a class="nav-link {% if data['active_link'] == 'metrics' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=metrics" role="tab" aria-selected="true">
|
||||||
<i class="fa-solid fa-chart-line"></i>{{ translate('serverDetails', 'metrics', data['lang']) }}</a>
|
<i class="fa-solid fa-chart-line"></i>{{ translate('serverDetails', 'metrics', data['lang']) }}</a>
|
||||||
</li>
|
</li>
|
||||||
|
{% if data['permissions']['Config'] in data['user_permissions'] %}
|
||||||
|
<li class="nav-item term-nav-item">
|
||||||
|
<a class="nav-link {% if data['active_link'] == 'webhooks' %}active{% end %}" href="/panel/server_detail?id={{ data['server_stats']['server_id']['server_id'] }}&subpage=webhooks" role="tab" aria-selected="true">
|
||||||
|
<i class="fa-regular fa-bell"></i>{{ translate('webhooks', 'webhooks', data['lang']) }}</a>
|
||||||
|
</li>
|
||||||
|
{% end %}
|
||||||
</ul>
|
</ul>
|
@ -87,7 +87,7 @@
|
|||||||
|
|
||||||
async function send_command_to_server(command) {
|
async function send_command_to_server(command) {
|
||||||
console.log(command)
|
console.log(command)
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
console.log('sending command: ' + command)
|
console.log('sending command: ' + command)
|
||||||
|
|
||||||
let res = await fetch(`/api/v2/servers/${serverId}/stdin`, {
|
let res = await fetch(`/api/v2/servers/${serverId}/stdin`, {
|
||||||
|
@ -44,30 +44,25 @@
|
|||||||
<div class="col-md-6 col-sm-12">
|
<div class="col-md-6 col-sm-12">
|
||||||
<br>
|
<br>
|
||||||
<br>
|
<br>
|
||||||
<form class="forms-sample" method="post" action="/panel/server_backup">
|
{% if data['backing_up'] %}
|
||||||
{% raw xsrf_form_html() %}
|
<div class="progress" style="height: 15px;">
|
||||||
<input type="hidden" name="id" value="{{ data['server_stats']['server_id']['server_id'] }}">
|
<div class="progress-bar progress-bar-striped progress-bar-animated" id="backup_progress_bar"
|
||||||
<input type="hidden" name="subpage" value="backup">
|
role="progressbar" style="width:{{data['backup_stats']['percent']}}%;"
|
||||||
|
aria-valuenow="{{data['backup_stats']['percent']}}" aria-valuemin="0" aria-valuemax="100">{{
|
||||||
|
data['backup_stats']['percent'] }}%</div>
|
||||||
|
</div>
|
||||||
|
<p>Backing up <i class="fas fa-spin fa-spinner"></i> <span
|
||||||
|
id="total_files">{{data['server_stats']['world_size']}}</span></p>
|
||||||
|
{% end %}
|
||||||
|
|
||||||
|
<br>
|
||||||
{% if data['backing_up'] %}
|
{% if not data['backing_up'] %}
|
||||||
<div class="progress" style="height: 15px;">
|
<div id="backup_button" class="form-group">
|
||||||
<div class="progress-bar progress-bar-striped progress-bar-animated" id="backup_progress_bar"
|
<button class="btn btn-primary" id="backup_now_button">{{ translate('serverBackups', 'backupNow',
|
||||||
role="progressbar" style="width:{{data['backup_stats']['percent']}}%;"
|
data['lang']) }}</button>
|
||||||
aria-valuenow="{{data['backup_stats']['percent']}}" aria-valuemin="0" aria-valuemax="100">{{
|
</div>
|
||||||
data['backup_stats']['percent'] }}%</div>
|
{% end %}
|
||||||
</div>
|
<form id="backup-form" class="forms-sample">
|
||||||
<p>Backing up <i class="fas fa-spin fa-spinner"></i> <span
|
|
||||||
id="total_files">{{data['server_stats']['world_size']}}</span></p>
|
|
||||||
{% end %}
|
|
||||||
|
|
||||||
<br>
|
|
||||||
{% if not data['backing_up'] %}
|
|
||||||
<div id="backup_button" class="form-group">
|
|
||||||
<button class="btn btn-primary" id="backup_now_button">{{ translate('serverBackups', 'backupNow',
|
|
||||||
data['lang']) }}</button>
|
|
||||||
</div>
|
|
||||||
{% end %}
|
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
{% if data['super_user'] %}
|
{% if data['super_user'] %}
|
||||||
<label for="server_name">{{ translate('serverBackups', 'storageLocation', data['lang']) }} <small
|
<label for="server_name">{{ translate('serverBackups', 'storageLocation', data['lang']) }} <small
|
||||||
@ -149,8 +144,6 @@
|
|||||||
data-server_path="{{ data['server_stats']['server_id']['path']}}" type="button">{{
|
data-server_path="{{ data['server_stats']['server_id']['path']}}" type="button">{{
|
||||||
translate('serverBackups', 'clickExclude', data['lang']) }}</button>
|
translate('serverBackups', 'clickExclude', data['lang']) }}</button>
|
||||||
</div>
|
</div>
|
||||||
<input type="number" class="form-control" name="changed" id="changed" value="0"
|
|
||||||
style="visibility: hidden;"></input>
|
|
||||||
<div class="modal fade" id="dir_select" tabindex="-1" role="dialog" aria-labelledby="dir_select"
|
<div class="modal fade" id="dir_select" tabindex="-1" role="dialog" aria-labelledby="dir_select"
|
||||||
aria-hidden="true">
|
aria-hidden="true">
|
||||||
<div class="modal-dialog" role="document">
|
<div class="modal-dialog" role="document">
|
||||||
@ -175,10 +168,8 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
<button type="button" id="modal-cancel" class="btn btn-secondary" data-dismiss="modal">{{
|
<button type="button" id="modal-cancel" class="btn btn-secondary" data-dismiss="modal"><i class="fa-solid fa-xmark"></i></button>
|
||||||
translate('serverBackups', 'cancel', data['lang']) }}</button>
|
<button type="button" id="modal-okay" data-dismiss="modal" class="btn btn-primary"><i class="fa-solid fa-thumbs-up"></i></button>
|
||||||
<button type="button" id="modal-okay" data-dismiss="modal" class="btn btn-primary">{{
|
|
||||||
translate('serverWizard', 'save', data['lang']) }}</button>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -316,66 +307,74 @@
|
|||||||
return r ? r[1] : undefined;
|
return r ? r[1] : undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
function backup_started() {
|
async function backup_started() {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
document.getElementById('backup_button').style.visibility = 'hidden';
|
let res = await fetch(`/api/v2/servers/${server_id}/action/backup_server`, {
|
||||||
var dialog = bootbox.dialog({
|
method: 'POST',
|
||||||
message: "{{ translate('serverBackups', 'backupTask', data['lang']) }}",
|
headers: {
|
||||||
closeButton: false
|
'X-XSRFToken': token
|
||||||
});
|
}
|
||||||
$.ajax({
|
});
|
||||||
type: "POST",
|
let responseData = await res.json();
|
||||||
headers: { 'X-XSRFToken': token },
|
if (responseData.status === "ok") {
|
||||||
url: `/api/v2/servers/${server_id}/action/backup_server`,
|
console.log(responseData);
|
||||||
success: function (data) {
|
$("#backup_button").html(`<div class="progress" style="height: 15px;">
|
||||||
return;
|
<div class="progress-bar progress-bar-striped progress-bar-animated" id="backup_progress_bar"
|
||||||
},
|
role="progressbar" style="width:{{data['backup_stats']['percent']}}%;"
|
||||||
});
|
aria-valuenow="{{data['backup_stats']['percent']}}" aria-valuemin="0" aria-valuemax="100">{{
|
||||||
|
data['backup_stats']['percent'] }}%</div>
|
||||||
|
</div>
|
||||||
|
<p>Backing up <i class="fas fa-spin fa-spinner"></i> <span
|
||||||
|
id="total_files">{{data['server_stats']['world_size']}}</span></p>`);
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
async function del_backup(filename, id) {
|
||||||
function del_backup(filename, id) {
|
const token = getCookie("_xsrf")
|
||||||
var token = getCookie("_xsrf")
|
let contents = JSON.stringify({"filename": filename})
|
||||||
|
let res = await fetch(`/api/v2/servers/${id}/backups/backup/`, {
|
||||||
data_to_send = { file_name: filename }
|
method: 'DELETE',
|
||||||
|
headers: {
|
||||||
console.log('Sending Command to delete backup: ' + filename)
|
'token': token,
|
||||||
$.ajax({
|
|
||||||
type: "DELETE",
|
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
url: '/ajax/del_backup?server_id=' + id,
|
|
||||||
data: {
|
|
||||||
file_path: filename,
|
|
||||||
id: id
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
location.reload();
|
|
||||||
},
|
},
|
||||||
|
body: contents
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.reload();
|
||||||
|
}else{
|
||||||
|
bootbox.alert({"title": responseData.status,
|
||||||
|
"message": responseData.error})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function restore_backup(filename, id) {
|
async function restore_backup(filename, id) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
|
let contents = JSON.stringify({"filename": filename})
|
||||||
var dialog = bootbox.dialog({
|
var dialog = bootbox.dialog({
|
||||||
message: "<i class='fa fa-spin fa-spinner'></i> {{ translate('serverBackups', 'restoring', data['lang']) }}",
|
message: "<i class='fa fa-spin fa-spinner'></i> {{ translate('serverBackups', 'restoring', data['lang']) }}",
|
||||||
closeButton: false
|
closeButton: false
|
||||||
});
|
});
|
||||||
|
let res = await fetch(`/api/v2/servers/${id}/backups/backup/`, {
|
||||||
console.log('Sending Command to restore backup: ' + filename)
|
method: 'POST',
|
||||||
$.ajax({
|
headers: {
|
||||||
type: "POST",
|
'token': token,
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
url: '/ajax/restore_backup?server_id=' + id,
|
|
||||||
data: {
|
|
||||||
zip_file: filename,
|
|
||||||
id: id
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
setTimeout(function () {
|
|
||||||
location.href = ('/panel/dashboard');
|
|
||||||
}, 15000);
|
|
||||||
},
|
},
|
||||||
|
body: contents
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.href = "/panel/dashboard";
|
||||||
|
}else{
|
||||||
|
bootbox.alert({"title": responseData.status,
|
||||||
|
"message": responseData.error})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$("#before-check").on("click", function () {
|
$("#before-check").on("click", function () {
|
||||||
@ -395,7 +394,66 @@
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
function replacer(key, value) {
|
||||||
|
if (key != "backup_before" && key != "backup_after") {
|
||||||
|
if (typeof value == "boolean" || key === "executable_update_url") {
|
||||||
|
return value
|
||||||
|
} else {
|
||||||
|
return (isNaN(value) ? value : +value);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
|
$("#backup-form").on("submit", async function (e) {
|
||||||
|
e.preventDefault();
|
||||||
|
const token = getCookie("_xsrf")
|
||||||
|
let backupForm = document.getElementById("backup-form");
|
||||||
|
|
||||||
|
let formData = new FormData(backupForm);
|
||||||
|
//Remove checks that we don't need in form data.
|
||||||
|
formData.delete("after-check");
|
||||||
|
formData.delete("before-check");
|
||||||
|
//Create an object from the form data entries
|
||||||
|
let formDataObject = Object.fromEntries(formData.entries());
|
||||||
|
//We need to make sure these are sent regardless of whether or not they're checked
|
||||||
|
formDataObject.compress = $("#compress").prop('checked');
|
||||||
|
formDataObject.shutdown = $("#shutdown").prop('checked');
|
||||||
|
let excluded = [];
|
||||||
|
$('input.excluded:checkbox:checked').each(function () {
|
||||||
|
excluded.push($(this).val());
|
||||||
|
});
|
||||||
|
if ($("#root_files_button").hasClass("clicked")){
|
||||||
|
formDataObject.exclusions = excluded;
|
||||||
|
}
|
||||||
|
console.log(excluded);
|
||||||
|
console.log(formDataObject);
|
||||||
|
// Format the plain form data as JSON
|
||||||
|
let formDataJsonString = JSON.stringify(formDataObject, replacer);
|
||||||
|
|
||||||
|
console.log(formDataJsonString);
|
||||||
|
|
||||||
|
let res = await fetch(`/api/v2/servers/${server_id}/backups/`, {
|
||||||
|
method: 'PATCH',
|
||||||
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: formDataJsonString,
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
window.location.reload();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.error,
|
||||||
|
message: responseData.error_data
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if ($('#backup_path').val() == '') {
|
if ($('#backup_path').val() == '') {
|
||||||
console.log('true')
|
console.log('true')
|
||||||
@ -457,7 +515,7 @@
|
|||||||
console.log(result);
|
console.log(result);
|
||||||
if (result == true) {
|
if (result == true) {
|
||||||
var full_path = backup_path + '/' + file_to_del;
|
var full_path = backup_path + '/' + file_to_del;
|
||||||
del_backup(full_path, server_id);
|
del_backup(file_to_del, server_id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -505,27 +563,15 @@
|
|||||||
return;
|
return;
|
||||||
} else {
|
} else {
|
||||||
document.getElementById('root_files_button').classList.add('clicked');
|
document.getElementById('root_files_button').classList.add('clicked');
|
||||||
document.getElementById("changed").value = 1;
|
|
||||||
}
|
}
|
||||||
path = $("#root_files_button").data('server_path')
|
path = $("#root_files_button").data('server_path')
|
||||||
console.log($("#root_files_button").data('server_path'))
|
console.log($("#root_files_button").data('server_path'))
|
||||||
var token = getCookie("_xsrf");
|
const token = getCookie("_xsrf");
|
||||||
var dialog = bootbox.dialog({
|
var dialog = bootbox.dialog({
|
||||||
message: '<p class="text-center mb-0"><i class="fa fa-spin fa-cog"></i> Please wait while we gather your files...</p>',
|
message: '<p class="text-center mb-0"><i class="fa fa-spin fa-cog"></i> Please wait while we gather your files...</p>',
|
||||||
closeButton: false
|
closeButton: false
|
||||||
});
|
});
|
||||||
|
|
||||||
$.ajax({
|
|
||||||
type: "POST",
|
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
url: '/ajax/backup_select?id=' + server_id + '&path=' + path,
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
bootbox.alert("You must input a path before selecting this button");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if (webSocket) {
|
|
||||||
webSocket.on('send_temp_path', function (data) {
|
|
||||||
setTimeout(function () {
|
setTimeout(function () {
|
||||||
var x = document.querySelector('.bootbox');
|
var x = document.querySelector('.bootbox');
|
||||||
if (x) {
|
if (x) {
|
||||||
@ -535,13 +581,15 @@
|
|||||||
if (x) {
|
if (x) {
|
||||||
x.remove()
|
x.remove()
|
||||||
}
|
}
|
||||||
document.getElementById('main-tree-input').setAttribute('value', data.path)
|
document.getElementById('main-tree-input').setAttribute('value', path)
|
||||||
getTreeView(data.path);
|
getTreeView(path);
|
||||||
show_file_tree();
|
show_file_tree();
|
||||||
|
|
||||||
}, 5000);
|
}, 5000);
|
||||||
});
|
} else {
|
||||||
}
|
bootbox.alert("You must input a path before selecting this button");
|
||||||
|
}
|
||||||
|
});
|
||||||
if (webSocket) {
|
if (webSocket) {
|
||||||
webSocket.on('backup_status', function (backup) {
|
webSocket.on('backup_status', function (backup) {
|
||||||
if (backup.percent >= 100) {
|
if (backup.percent >= 100) {
|
||||||
@ -558,68 +606,82 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function getTreeView(path) {
|
function getDirView(event){
|
||||||
path = path
|
let path = event.target.parentElement.getAttribute("data-path");
|
||||||
|
if (document.getElementById(path).classList.contains('clicked')) {
|
||||||
|
return;
|
||||||
|
}else{
|
||||||
|
getTreeView(path);
|
||||||
|
}
|
||||||
|
|
||||||
$.ajax({
|
}
|
||||||
type: "GET",
|
async function getTreeView(path){
|
||||||
url: '/ajax/get_backup_tree?id=' + server_id + '&path=' + path,
|
console.log(path)
|
||||||
dataType: 'text',
|
const token = getCookie("_xsrf");
|
||||||
success: function (data) {
|
let res = await fetch(`/api/v2/servers/${server_id}/files`, {
|
||||||
console.log("got response:");
|
method: 'POST',
|
||||||
console.log(data);
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
|
},
|
||||||
|
body: JSON.stringify({"page": "backups", "path": path}),
|
||||||
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
console.log(responseData);
|
||||||
|
process_tree_response(responseData);
|
||||||
|
|
||||||
dataArr = data.split('\n');
|
} else {
|
||||||
serverDir = dataArr.shift(); // Remove & return first element (server directory)
|
|
||||||
text = dataArr.join('\n');
|
|
||||||
|
|
||||||
try {
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function process_tree_response(response) {
|
||||||
|
let path = response.data.root_path.path;
|
||||||
|
let text = `<ul class="tree-nested d-block" id="${path}ul">`;
|
||||||
|
Object.entries(response.data).forEach(([key, value]) => {
|
||||||
|
if (key === "root_path" || key === "db_stats"){
|
||||||
|
//continue is not valid in for each. Return acts as a continue.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let checked = ""
|
||||||
|
let dpath = value.path;
|
||||||
|
let filename = key;
|
||||||
|
if (value.dir){
|
||||||
|
if (value.excluded){
|
||||||
|
checked = "checked"
|
||||||
|
}
|
||||||
|
text += `<li class="tree-item" data-path="${dpath}">
|
||||||
|
\n<div id="${dpath}" data-path="${dpath}" data-name="${filename}" class="tree-caret tree-ctx-item tree-folder">
|
||||||
|
<input type="checkbox" class="checkBoxClass excluded" value="${dpath}" ${checked}>
|
||||||
|
<span id="${dpath}span" class="files-tree-title" data-path="${dpath}" data-name="${filename}" onclick="getDirView(event)">
|
||||||
|
<i style="color: var(--info);" class="far fa-folder"></i>
|
||||||
|
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
||||||
|
<strong>${filename}</strong>
|
||||||
|
</span>
|
||||||
|
</input></div><li>`
|
||||||
|
}else{
|
||||||
|
text += `<li
|
||||||
|
class="d-block tree-ctx-item tree-file"
|
||||||
|
data-path="${dpath}"
|
||||||
|
data-name="${filename}"
|
||||||
|
onclick=""><input type='checkbox' class="checkBoxClass excluded" name='root_path' value="${dpath}" ${checked}><span style="margin-right: 6px;">
|
||||||
|
<i class="far fa-file"></i></span></input>${filename}</li>`
|
||||||
|
}
|
||||||
|
});
|
||||||
|
text += `</ul>`;
|
||||||
|
if(response.data.root_path.top){
|
||||||
|
try {
|
||||||
document.getElementById('main-tree-div').innerHTML += text;
|
document.getElementById('main-tree-div').innerHTML += text;
|
||||||
document.getElementById('main-tree').parentElement.classList.add("clicked");
|
document.getElementById('main-tree').parentElement.classList.add("clicked");
|
||||||
} catch {
|
} catch {
|
||||||
document.getElementById('files-tree').innerHTML = text;
|
document.getElementById('files-tree').innerHTML = text;
|
||||||
}
|
}
|
||||||
|
}else{
|
||||||
|
try {
|
||||||
document.getElementsByClassName('files-tree-title')[0].setAttribute('data-path', serverDir);
|
|
||||||
document.getElementsByClassName('files-tree-title')[0].setAttribute('data-name', 'Files');
|
|
||||||
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
function getToggleMain(event) {
|
|
||||||
path = event.target.parentElement.getAttribute('data-path');
|
|
||||||
document.getElementById("files-tree").classList.toggle("d-block");
|
|
||||||
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
|
||||||
document.getElementById(path + "span").classList.toggle("tree-caret");
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
function getDirView(event) {
|
|
||||||
path = event.target.parentElement.getAttribute('data-path');
|
|
||||||
|
|
||||||
if (document.getElementById(path).classList.contains('clicked')) {
|
|
||||||
|
|
||||||
var toggler = document.getElementById(path + "span");
|
|
||||||
|
|
||||||
if (toggler.classList.contains('files-tree-title')) {
|
|
||||||
document.getElementById(path + "ul").classList.toggle("d-block");
|
|
||||||
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
$.ajax({
|
|
||||||
type: "GET",
|
|
||||||
url: '/ajax/get_backup_dir?id=' + server_id + '&path=' + path,
|
|
||||||
dataType: 'text',
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
|
|
||||||
dataArr = data.split('\n');
|
|
||||||
serverDir = dataArr.shift(); // Remove & return first element (server directory)
|
|
||||||
text = dataArr.join('\n');
|
|
||||||
|
|
||||||
try {
|
|
||||||
document.getElementById(path + "span").classList.add('tree-caret-down');
|
document.getElementById(path + "span").classList.add('tree-caret-down');
|
||||||
document.getElementById(path).innerHTML += text;
|
document.getElementById(path).innerHTML += text;
|
||||||
document.getElementById(path).classList.add("clicked");
|
document.getElementById(path).classList.add("clicked");
|
||||||
@ -627,7 +689,7 @@
|
|||||||
console.log("Bad")
|
console.log("Bad")
|
||||||
}
|
}
|
||||||
|
|
||||||
var toggler = document.getElementById(path);
|
var toggler = document.getElementById(path + "span");
|
||||||
|
|
||||||
if (toggler.classList.contains('files-tree-title')) {
|
if (toggler.classList.contains('files-tree-title')) {
|
||||||
document.getElementById(path + "span").addEventListener("click", function caretListener() {
|
document.getElementById(path + "span").addEventListener("click", function caretListener() {
|
||||||
@ -635,10 +697,15 @@
|
|||||||
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getToggleMain(event) {
|
||||||
|
path = event.target.parentElement.getAttribute('data-path');
|
||||||
|
document.getElementById("files-tree").classList.toggle("d-block");
|
||||||
|
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
||||||
|
document.getElementById(path + "span").classList.toggle("tree-caret");
|
||||||
|
}
|
||||||
function show_file_tree() {
|
function show_file_tree() {
|
||||||
$("#dir_select").modal();
|
$("#dir_select").modal();
|
||||||
}
|
}
|
||||||
|
@ -136,7 +136,7 @@
|
|||||||
<label for="server_port">{{ translate('serverConfig', 'serverPort', data['lang']) }} <small class="text-muted ml-1"> - {{ translate('serverConfig', 'serverPortDesc', data['lang']) }}
|
<label for="server_port">{{ translate('serverConfig', 'serverPort', data['lang']) }} <small class="text-muted ml-1"> - {{ translate('serverConfig', 'serverPortDesc', data['lang']) }}
|
||||||
</small> </label>
|
</small> </label>
|
||||||
<input type="number" class="form-control" name="server_port" id="server_port" value="{{ data['server_stats']['server_id']['server_port'] }}" step="1" max="65566" min="1" required>
|
<input type="number" class="form-control" name="server_port" id="server_port" value="{{ data['server_stats']['server_id']['server_port'] }}" step="1" max="65566" min="1" required>
|
||||||
<span data-html="true" class="port-hint text-center" title="<i class='fal fa-exclamation-triangle'></i> " , data-content="{{
|
<span data-html="true" class="port-hint text-center" title="<i class='fa-solid fa-triangle-exclamation'></i> " , data-content="{{
|
||||||
translate('serverConfig', 'statsHint1' , data['lang'])}} <br> <br> <strong>{{ translate('serverConfig', 'statsHint2', data['lang'])}}</strong>" , data-placement="right"></span>
|
translate('serverConfig', 'statsHint1' , data['lang'])}} <br> <br> <strong>{{ translate('serverConfig', 'statsHint2', data['lang'])}}</strong>" , data-placement="right"></span>
|
||||||
</div>
|
</div>
|
||||||
{% end %}
|
{% end %}
|
||||||
@ -304,7 +304,7 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
function deleteServerE(callback) {
|
function deleteServerE(callback) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "DELETE",
|
type: "DELETE",
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: { 'X-XSRFToken': token },
|
||||||
@ -318,7 +318,7 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
function deleteServerFilesE(path, callback) {
|
function deleteServerFilesE(path, callback) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "DELETE",
|
type: "DELETE",
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: { 'X-XSRFToken': token },
|
||||||
@ -334,7 +334,7 @@
|
|||||||
|
|
||||||
function send_command(serverId, command) {
|
function send_command(serverId, command) {
|
||||||
//<!-- this getCookie function is in base.html-->
|
//<!-- this getCookie function is in base.html-->
|
||||||
var token = getCookie("_xsrf");
|
const token = getCookie("_xsrf");
|
||||||
if (command == "update_executable") {
|
if (command == "update_executable") {
|
||||||
document.getElementById("update-spinner").style.visibility = "visible";
|
document.getElementById("update-spinner").style.visibility = "visible";
|
||||||
}
|
}
|
||||||
@ -460,7 +460,7 @@
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
setTimeout(function () { window.location = '/panel/dashboard'; }, 5000);
|
setTimeout(function () { window.location = '/panel/dashboard'; }, 5000);
|
||||||
bootbox.dialog({
|
bootbox.dialog({
|
||||||
backdrop: true,
|
backdrop: true,
|
||||||
@ -549,7 +549,7 @@
|
|||||||
});
|
});
|
||||||
$("#config_form").on("submit", async function (e) {
|
$("#config_form").on("submit", async function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
let configForm = document.getElementById("config_form");
|
let configForm = document.getElementById("config_form");
|
||||||
|
|
||||||
let formData = new FormData(configForm);
|
let formData = new FormData(configForm);
|
||||||
@ -576,7 +576,7 @@
|
|||||||
});
|
});
|
||||||
let responseData = await res.json();
|
let responseData = await res.json();
|
||||||
if (responseData.status === "ok") {
|
if (responseData.status === "ok") {
|
||||||
window.location.reload();
|
location.reload(true);
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
bootbox.alert({
|
bootbox.alert({
|
||||||
|
@ -67,7 +67,7 @@
|
|||||||
translate('serverFiles', 'download', data['lang']) }}</a>
|
translate('serverFiles', 'download', data['lang']) }}</a>
|
||||||
<a onclick="deleteFileE(event)" href="javascript:void(0)" id="deleteFile" href="#"
|
<a onclick="deleteFileE(event)" href="javascript:void(0)" id="deleteFile" href="#"
|
||||||
style="color: red">{{ translate('serverFiles', 'delete', data['lang']) }}</a>
|
style="color: red">{{ translate('serverFiles', 'delete', data['lang']) }}</a>
|
||||||
<a onclick="deleteDirE(event)" href="javascript:void(0)" id="deleteDir" href="#" style="color: red">{{
|
<a onclick="deleteFileE(event)" href="javascript:void(0)" id="deleteDir" href="#" style="color: red">{{
|
||||||
translate('serverFiles', 'delete', data['lang']) }}</a>
|
translate('serverFiles', 'delete', data['lang']) }}</a>
|
||||||
<a href="javascript:void(0)" class="closebtn" style="color: var(--info);"
|
<a href="javascript:void(0)" class="closebtn" style="color: var(--info);"
|
||||||
onclick="document.getElementById('files-tree-nav').style.display = 'none';">{{
|
onclick="document.getElementById('files-tree-nav').style.display = 'none';">{{
|
||||||
@ -156,6 +156,9 @@
|
|||||||
right: 35px;
|
right: 35px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.tree-file:hover{
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
<ul class="tree-view">
|
<ul class="tree-view">
|
||||||
<li>
|
<li>
|
||||||
@ -398,32 +401,36 @@
|
|||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
let filePath = '', serverFileContent = '';
|
let path = '', serverFileContent = '';
|
||||||
|
|
||||||
function clickOnFile(event) {
|
async function clickOnFile(event) {
|
||||||
filePath = event.target.getAttribute('data-path');
|
const token = getCookie("_xsrf");
|
||||||
$.ajax({
|
path = event.target.getAttribute('data-path');
|
||||||
type: 'GET',
|
let res = await fetch(`/api/v2/servers/${serverId}/files`, {
|
||||||
url: "/files/get_file?id=" + serverId + "&file_path=" + encodeURIComponent(filePath),
|
method: 'POST',
|
||||||
dataType: 'text',
|
headers: {
|
||||||
success: function (data) {
|
'X-XSRFToken': token
|
||||||
console.log('Got File Contents From Server');
|
|
||||||
json = JSON.parse(data)
|
|
||||||
if (json.error) {
|
|
||||||
$('#editorParent').toggle(false) // hide
|
|
||||||
$('#fileError').toggle(true) // show
|
|
||||||
$('#fileError').text("{{ translate('serverFiles', 'fileReadError', data['lang']) }}: " + json.error) // show error
|
|
||||||
editor.blur()
|
|
||||||
} else {
|
|
||||||
$('#editorParent').toggle(true) // show
|
|
||||||
$('#fileError').toggle(false) // hide
|
|
||||||
setFileName(event.target.innerText);
|
|
||||||
editor.session.setValue(json.content);
|
|
||||||
serverFileContent = json.content;
|
|
||||||
setSaveStatus(true);
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "page": "files", "path": path }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
console.log(responseData)
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
console.log('Got File Contents From Server');
|
||||||
|
$('#editorParent').toggle(true) // show
|
||||||
|
$('#fileError').toggle(false) // hide
|
||||||
|
setFileName(event.target.innerText);
|
||||||
|
editor.session.setValue(responseData.data);
|
||||||
|
serverFileContent = responseData.data;
|
||||||
|
setSaveStatus(true);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function setFileName(name) {
|
function setFileName(name) {
|
||||||
@ -577,124 +584,141 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
async function save() {
|
||||||
function save() {
|
|
||||||
let text = editor.session.getValue();
|
let text = editor.session.getValue();
|
||||||
|
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/servers/${serverId}/files`, {
|
||||||
type: "PUT",
|
method: 'PATCH',
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: {
|
||||||
url: "/files/save_file?id=" + serverId,
|
'X-XSRFToken': token
|
||||||
data: {
|
|
||||||
file_contents: text,
|
|
||||||
file_path: filePath
|
|
||||||
},
|
},
|
||||||
success: (data) => {
|
body: JSON.stringify({ "path": path, "contents": text }),
|
||||||
serverFileContent = text;
|
|
||||||
setSaveStatus(true)
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
serverFileContent = text;
|
||||||
|
setSaveStatus(true)
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function createFile(parent, name, callback) {
|
async function createFile(parent, name, callback) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/servers/${serverId}/files/create/`, {
|
||||||
type: "POST",
|
method: 'PUT',
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: {
|
||||||
url: "/files/create_file?id=" + serverId,
|
'X-XSRFToken': token
|
||||||
data: {
|
|
||||||
file_parent: parent,
|
|
||||||
file_name: name
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
callback();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "parent": parent, "name": name, "directory": false }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
getTreeView($('#root_dir').data('path'));
|
||||||
|
setTreeViewContext();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function createDir(parent, name, callback) {
|
|
||||||
var token = getCookie("_xsrf")
|
async function createDir(parent, name, callback) {
|
||||||
$.ajax({
|
const token = getCookie("_xsrf")
|
||||||
type: "POST",
|
let res = await fetch(`/api/v2/servers/${serverId}/files/create/`, {
|
||||||
headers: { 'X-XSRFToken': token },
|
method: 'PUT',
|
||||||
url: "/files/create_dir?id=" + serverId,
|
headers: {
|
||||||
data: {
|
'X-XSRFToken': token
|
||||||
dir_parent: parent,
|
|
||||||
dir_name: name
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
callback();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "parent": parent, "name": name, "directory": true }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
getTreeView($('#root_dir').data('path'));
|
||||||
|
setTreeViewContext();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renameItem(path, name, callback) {
|
async function renameItem(path, name, callback) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/servers/${serverId}/files/create/`, {
|
||||||
type: "PATCH",
|
method: 'PATCH',
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: {
|
||||||
url: "/files/rename_file?id=" + serverId,
|
'X-XSRFToken': token
|
||||||
data: {
|
|
||||||
item_path: path,
|
|
||||||
new_item_name: name
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
callback();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "path": path, "new_name": name }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
getTreeView($('#root_dir').data('path'));
|
||||||
|
setTreeViewContext();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function deleteFile(path, callback) {
|
async function deleteItem(path, el, callback) {
|
||||||
console.log('Deleting: ' + path)
|
const token = getCookie("_xsrf");
|
||||||
var token = getCookie("_xsrf")
|
let res = await fetch(`/api/v2/servers/${serverId}/files`, {
|
||||||
$.ajax({
|
method: 'DELETE',
|
||||||
type: "DELETE",
|
headers: {
|
||||||
headers: { 'X-XSRFToken': token },
|
'X-XSRFToken': token
|
||||||
url: "/files/del_file?id=" + serverId,
|
|
||||||
data: {
|
|
||||||
file_path: path
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
callback();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "filename": path }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
el = document.getElementById(path + "li");
|
||||||
|
$(el).remove();
|
||||||
|
document.getElementById('files-tree-nav').style.display = 'none';
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function deleteDir(path, callback) {
|
async function unZip(path, callback) {
|
||||||
var token = getCookie("_xsrf")
|
const token = getCookie("_xsrf")
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/servers/${serverId}/files/zip/`, {
|
||||||
type: "DELETE",
|
method: 'POST',
|
||||||
headers: { 'X-XSRFToken': token },
|
headers: {
|
||||||
url: "/files/del_dir?id=" + serverId,
|
'X-XSRFToken': token
|
||||||
data: {
|
|
||||||
dir_path: path
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
callback();
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "folder": path }),
|
||||||
});
|
});
|
||||||
}
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
getTreeView($('#root_dir').data('path'));
|
||||||
|
setTreeViewContext();
|
||||||
|
} else {
|
||||||
|
|
||||||
function unZip(path, callback) {
|
bootbox.alert({
|
||||||
console.log('path: ', path)
|
title: responseData.status,
|
||||||
var token = getCookie("_xsrf")
|
message: responseData.error
|
||||||
$.ajax({
|
});
|
||||||
type: "POST",
|
}
|
||||||
headers: { 'X-XSRFToken': token },
|
|
||||||
url: "/files/unzip_file?id=" + serverId,
|
|
||||||
data: {
|
|
||||||
path: path
|
|
||||||
},
|
|
||||||
success: function (data) {
|
|
||||||
window.location.href = "/panel/server_detail?id=" + serverId + "&subpage=files";
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async function sendFile(file, path, serverId, left, i, onProgress) {
|
async function sendFile(file, path, serverId, left, i, onProgress) {
|
||||||
@ -882,36 +906,104 @@
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function getTreeView(event) {
|
function getDirView(event) {
|
||||||
const path = $('#root_dir').data('path');;
|
let path = event.target.parentElement.getAttribute("data-path");
|
||||||
|
if (document.getElementById(path).classList.contains('clicked')) {
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
getTreeView(path);
|
||||||
|
}
|
||||||
|
|
||||||
$.ajax({
|
}
|
||||||
type: "GET",
|
async function getTreeView(path) {
|
||||||
url: "/files/get_tree?id=" + serverId + "&path=" + path,
|
const token = getCookie("_xsrf");
|
||||||
dataType: 'text',
|
let res = await fetch(`/api/v2/servers/${serverId}/files`, {
|
||||||
success: function (data) {
|
method: 'POST',
|
||||||
console.log("got response:");
|
headers: {
|
||||||
|
'X-XSRFToken': token
|
||||||
dataArr = data.split('\n');
|
|
||||||
serverDir = dataArr.shift(); // Remove & return first element (server directory)
|
|
||||||
text = dataArr.join('\n');
|
|
||||||
|
|
||||||
try {
|
|
||||||
document.getElementById(path).innerHTML += text;
|
|
||||||
event.target.parentElement.classList.add("clicked");
|
|
||||||
} catch {
|
|
||||||
document.getElementById('files-tree').innerHTML = text;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
document.getElementsByClassName('files-tree-title')[0].setAttribute('data-path', serverDir);
|
|
||||||
document.getElementsByClassName('files-tree-title')[0].setAttribute('data-name', 'Files');
|
|
||||||
|
|
||||||
setTimeout(function () { setTreeViewContext() }, 1000);
|
|
||||||
},
|
},
|
||||||
|
body: JSON.stringify({ "page": "files", "path": path }),
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
console.log(responseData);
|
||||||
|
process_tree_response(responseData);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function process_tree_response(response) {
|
||||||
|
let path = response.data.root_path.path;
|
||||||
|
let text = ``;
|
||||||
|
if (!response.data.root_path.top) {
|
||||||
|
text = `<ul class="tree-nested d-block" id="${path}ul">`;
|
||||||
|
}
|
||||||
|
Object.entries(response.data).forEach(([key, value]) => {
|
||||||
|
if (key === "root_path" || key === "db_stats") {
|
||||||
|
//continue is not valid in for each. Return acts as a continue.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let checked = ""
|
||||||
|
let dpath = value.path;
|
||||||
|
let filename = key;
|
||||||
|
if (value.dir) {
|
||||||
|
if (value.excluded) {
|
||||||
|
checked = "checked"
|
||||||
|
}
|
||||||
|
text += `<li class="tree-item" id="${dpath}li" data-path="${dpath}">
|
||||||
|
\n<div id="${dpath}" data-path="${dpath}" data-name="${filename}" class="tree-caret tree-ctx-item tree-folder">
|
||||||
|
<input type="checkbox" class="checkBoxClass d-none file-check" name="root_path" value="${dpath}" ${checked}>
|
||||||
|
<span id="${dpath}span" class="files-tree-title" data-path="${dpath}" data-name="${filename}" onclick="getDirView(event)">
|
||||||
|
<i style="color: var(--info);" class="far fa-folder"></i>
|
||||||
|
<i style="color: var(--info);" class="far fa-folder-open"></i>
|
||||||
|
${filename}
|
||||||
|
</span>
|
||||||
|
</input></div></li>`
|
||||||
|
} else {
|
||||||
|
text += `<li
|
||||||
|
class="d-block tree-ctx-item tree-file"
|
||||||
|
data-path="${dpath}"
|
||||||
|
data-name="${filename}"
|
||||||
|
onclick="clickOnFile(event)" id="${dpath}li"><input type='checkbox' class="checkBoxClass d-none file-check" name='root_path' value="${dpath}" ${checked}><span style="margin-right: 6px;">
|
||||||
|
<i class="far fa-file"></i></span></input>${filename}</li>`
|
||||||
|
}
|
||||||
|
});
|
||||||
|
if (!response.data.root_path.top) {
|
||||||
|
text += `</ul>`;
|
||||||
|
}
|
||||||
|
if (response.data.root_path.top) {
|
||||||
|
try {
|
||||||
|
document.getElementById('main-tree-div').innerHTML += text;
|
||||||
|
document.getElementById('main-tree').parentElement.classList.add("clicked");
|
||||||
|
} catch {
|
||||||
|
document.getElementById('files-tree').innerHTML = text;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
document.getElementById(path + "span").classList.add('tree-caret-down');
|
||||||
|
document.getElementById(path).innerHTML += text;
|
||||||
|
document.getElementById(path).classList.add("clicked");
|
||||||
|
} catch {
|
||||||
|
console.log("Bad")
|
||||||
|
}
|
||||||
|
|
||||||
|
var toggler = document.getElementById(path + "span");
|
||||||
|
|
||||||
|
if (toggler.classList.contains('files-tree-title')) {
|
||||||
|
document.getElementById(path + "span").addEventListener("click", function caretListener() {
|
||||||
|
document.getElementById(path + "ul").classList.toggle("d-block");
|
||||||
|
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
setTimeout(function () { setTreeViewContext() }, 1000);
|
||||||
|
}
|
||||||
function getToggleMain(event) {
|
function getToggleMain(event) {
|
||||||
path = event.target.parentElement.getAttribute('data-path');
|
path = event.target.parentElement.getAttribute('data-path');
|
||||||
document.getElementById("files-tree").classList.toggle("d-block");
|
document.getElementById("files-tree").classList.toggle("d-block");
|
||||||
@ -919,53 +1011,6 @@
|
|||||||
document.getElementById(path + "span").classList.toggle("tree-caret");
|
document.getElementById(path + "span").classList.toggle("tree-caret");
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDirView(event) {
|
|
||||||
let path = event.target.parentElement.getAttribute('data-path');
|
|
||||||
|
|
||||||
if (document.getElementById(path).classList.contains('clicked')) {
|
|
||||||
|
|
||||||
var toggler = document.getElementById(path + "span");
|
|
||||||
|
|
||||||
if (toggler.classList.contains('files-tree-title')) {
|
|
||||||
document.getElementById(path + "ul").classList.toggle("d-block");
|
|
||||||
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
} else {
|
|
||||||
$.ajax({
|
|
||||||
type: "GET",
|
|
||||||
url: "/files/get_dir?id=" + serverId + "&path=" + path,
|
|
||||||
dataType: 'text',
|
|
||||||
success: function (data) {
|
|
||||||
console.log("got response:");
|
|
||||||
|
|
||||||
dataArr = data.split('\n');
|
|
||||||
serverDir = dataArr.shift(); // Remove & return first element (server directory)
|
|
||||||
text = dataArr.join('\n');
|
|
||||||
|
|
||||||
try {
|
|
||||||
document.getElementById(path + "span").classList.add('tree-caret-down');
|
|
||||||
document.getElementById(path).innerHTML += text;
|
|
||||||
document.getElementById(path).classList.add("clicked");
|
|
||||||
} catch {
|
|
||||||
console.log("Bad")
|
|
||||||
}
|
|
||||||
|
|
||||||
setTimeout(function () { setTreeViewContext() }, 1000);
|
|
||||||
|
|
||||||
var toggler = document.getElementById(path);
|
|
||||||
|
|
||||||
if (toggler.classList.contains('files-tree-title')) {
|
|
||||||
document.getElementById(path + "span").addEventListener("click", function caretListener() {
|
|
||||||
document.getElementById(path + "ul").classList.toggle("d-block");
|
|
||||||
document.getElementById(path + "span").classList.toggle("tree-caret-down");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function setTreeViewContext() {
|
function setTreeViewContext() {
|
||||||
var treeItems = Array.from(document.getElementsByClassName('tree-ctx-item'));
|
var treeItems = Array.from(document.getElementsByClassName('tree-ctx-item'));
|
||||||
|
|
||||||
@ -1134,45 +1179,12 @@
|
|||||||
},
|
},
|
||||||
callback: function (result) {
|
callback: function (result) {
|
||||||
if (!result) return;
|
if (!result) return;
|
||||||
deleteFile(path, function () {
|
deleteItem(path);
|
||||||
el = document.getElementById(path + "li");
|
|
||||||
$(el).remove();
|
|
||||||
document.getElementById('files-tree-nav').style.display = 'none';
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function deleteDirE(event) {
|
getTreeView($('#root_dir').data('path'));
|
||||||
path = event.target.parentElement.getAttribute('data-path');
|
|
||||||
name = event.target.parentElement.getAttribute('data-name');
|
|
||||||
bootbox.confirm({
|
|
||||||
size: "",
|
|
||||||
title: "{% raw translate('serverFiles', 'deleteItemQuestion', data['lang']) %}",
|
|
||||||
closeButton: false,
|
|
||||||
message: "{% raw translate('serverFiles', 'deleteItemQuestionMessage', data['lang']) %}",
|
|
||||||
buttons: {
|
|
||||||
confirm: {
|
|
||||||
label: "{{ translate('serverFiles', 'yesDelete', data['lang']) }}",
|
|
||||||
className: 'btn-danger'
|
|
||||||
},
|
|
||||||
cancel: {
|
|
||||||
label: "{{ translate('serverFiles', 'noDelete', data['lang']) }}",
|
|
||||||
className: 'btn-link'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
callback: function (result) {
|
|
||||||
if (!result) return;
|
|
||||||
deleteDir(path, function () {
|
|
||||||
el = document.getElementById(path + "li");
|
|
||||||
$(el).remove();
|
|
||||||
document.getElementById('files-tree-nav').style.display = 'none';
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
getTreeView();
|
|
||||||
setTreeViewContext();
|
setTreeViewContext();
|
||||||
|
|
||||||
function setKeyboard(target) {
|
function setKeyboard(target) {
|
||||||
|
@ -77,8 +77,8 @@
|
|||||||
{% block js %}
|
{% block js %}
|
||||||
<script>
|
<script>
|
||||||
// ##### Log Filter Block #####
|
// ##### Log Filter Block #####
|
||||||
var lines = [];
|
let lines = [];
|
||||||
var words = [];
|
let words = [];
|
||||||
if (localStorage.getItem("words")) {
|
if (localStorage.getItem("words")) {
|
||||||
try {
|
try {
|
||||||
words = JSON.parse(localStorage.getItem("words"));
|
words = JSON.parse(localStorage.getItem("words"));
|
||||||
@ -188,27 +188,40 @@
|
|||||||
|
|
||||||
// Populate logs and filter if present
|
// Populate logs and filter if present
|
||||||
const serverId = new URLSearchParams(document.location.search).get('id')
|
const serverId = new URLSearchParams(document.location.search).get('id')
|
||||||
function get_server_log() {
|
async function get_server_log() {
|
||||||
|
const token = getCookie("_xsrf")
|
||||||
|
let colors = true;
|
||||||
if (!$("#stop_scroll").is(':checked')) {
|
if (!$("#stop_scroll").is(':checked')) {
|
||||||
$.ajax({
|
let res = await fetch(`/api/v2/servers/${serverId}/logs?colors=${colors}`, {
|
||||||
type: 'GET',
|
method: 'GET',
|
||||||
url: '/ajax/server_log?id=' + serverId + '&full=1',
|
headers: {
|
||||||
dataType: 'text',
|
'X-XSRFToken': token
|
||||||
success: function (data) {
|
|
||||||
console.log('Got Log From Server')
|
|
||||||
$('#virt_console').html(data);
|
|
||||||
scroll();
|
|
||||||
lines = document.querySelectorAll('.box');
|
|
||||||
hideFilteredWords();
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
let responseData = await res.json();
|
||||||
|
let html = ``
|
||||||
|
if (responseData.status === "ok") {
|
||||||
|
for (let value of responseData.data) {
|
||||||
|
html += `<span class='box'>${value}<br /></span>`
|
||||||
|
}
|
||||||
|
console.log('Got Log From Server')
|
||||||
|
$('#virt_console').html(html);
|
||||||
|
scroll();
|
||||||
|
lines = document.querySelectorAll('.box');
|
||||||
|
hideFilteredWords();
|
||||||
|
} else {
|
||||||
|
|
||||||
|
bootbox.alert({
|
||||||
|
title: responseData.status,
|
||||||
|
message: responseData.error
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
console.log("ready!");
|
console.log("ready!");
|
||||||
get_server_log();
|
get_server_log();
|
||||||
populateWords();
|
populateWords();
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
{% end %}
|
{% end %}
|
@ -89,11 +89,11 @@
|
|||||||
const cpu = []
|
const cpu = []
|
||||||
{% for item in data['history_stats'] %}
|
{% for item in data['history_stats'] %}
|
||||||
{% if 'minecraft-java' in data['server_stats']['server_type'] %}
|
{% if 'minecraft-java' in data['server_stats']['server_type'] %}
|
||||||
players.push("{{ item.online }}");
|
players.push("{{ item.get('online') }}");
|
||||||
{% end %}
|
{% end %}
|
||||||
dates.push("{{ item.created.strftime('%Y/%m/%d, %H:%M:%S') }}");
|
dates.push("{{ item.get('created').strftime('%Y/%m/%d, %H:%M:%S') }}");
|
||||||
ram.push("{{ item.mem_percent }}")
|
ram.push("{{ item.get('mem_percent') }}")
|
||||||
cpu.push("{{ item.cpu }}")
|
cpu.push("{{ item.get('cpu') }}")
|
||||||
{% end %}
|
{% end %}
|
||||||
var hist_chart = new Chart(ctxL, {
|
var hist_chart = new Chart(ctxL, {
|
||||||
type: 'line',
|
type: 'line',
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user