mirror of
https://github.com/jokob-sk/NetAlertX.git
synced 2025-12-06 17:15:38 -08:00
Merge pull request #1230 from adamoutler/hardening
Feat: Enterprise-Grade Security Hardening and Build Overhaul
This commit is contained in:
268
.devcontainer/Dockerfile
Executable file → Normal file
268
.devcontainer/Dockerfile
Executable file → Normal file
@@ -1,112 +1,236 @@
|
|||||||
# DO NOT MODIFY THIS FILE DIRECTLY. IT IS AUTO-GENERATED BY .devcontainer/scripts/generate-dockerfile.sh
|
# DO NOT MODIFY THIS FILE DIRECTLY. IT IS AUTO-GENERATED BY .devcontainer/scripts/generate-dockerfile.sh
|
||||||
|
|
||||||
# ---/Dockerfile---
|
# ---/Dockerfile---
|
||||||
|
# The NetAlertX Dockerfile has 3 stages:
|
||||||
|
#
|
||||||
|
# Stage 1. Builder - NetAlertX Requires special tools and packages to build our virtual environment, but
|
||||||
|
# which are not needed in future stages. We build the builder and extract the venv for runner to use as
|
||||||
|
# a base.
|
||||||
|
#
|
||||||
|
# Stage 2. Runner builds the bare minimum requirements to create an operational NetAlertX. The primary
|
||||||
|
# reason for breaking at this stage is it leaves the system in a proper state for devcontainer operation
|
||||||
|
# This image also provides a break-out point for uses who wish to execute the anti-pattern of using a
|
||||||
|
# docker container as a VM for experimentation and various development patterns.
|
||||||
|
#
|
||||||
|
# Stage 3. Hardened removes root, sudoers, folders, permissions, and locks the system down into a read-only
|
||||||
|
# compatible image. While NetAlertX does require some read-write operations, this image can guarantee the
|
||||||
|
# code pushed out by the project is the only code which will run on the system after each container restart.
|
||||||
|
# It reduces the chance of system hijacking and operates with all modern security protocols in place as is
|
||||||
|
# expected from a security appliance.
|
||||||
|
#
|
||||||
|
# This file can be built with `docker compose -f docker-compose.yml up --build --force-recreate`
|
||||||
|
|
||||||
FROM alpine:3.22 AS builder
|
FROM alpine:3.22 AS builder
|
||||||
|
|
||||||
ARG INSTALL_DIR=/app
|
ARG INSTALL_DIR=/app
|
||||||
|
|
||||||
ENV PYTHONUNBUFFERED=1
|
ENV PYTHONUNBUFFERED=1
|
||||||
|
ENV PATH="/opt/venv/bin:$PATH"
|
||||||
|
|
||||||
# Install build dependencies
|
# Install build dependencies
|
||||||
|
COPY requirements.txt /tmp/requirements.txt
|
||||||
RUN apk add --no-cache bash shadow python3 python3-dev gcc musl-dev libffi-dev openssl-dev git \
|
RUN apk add --no-cache bash shadow python3 python3-dev gcc musl-dev libffi-dev openssl-dev git \
|
||||||
&& python -m venv /opt/venv
|
&& python -m venv /opt/venv
|
||||||
|
|
||||||
# Enable venv
|
# Create virtual environment owned by root, but readable by everyone else. This makes it easy to copy
|
||||||
ENV PATH="/opt/venv/bin:$PATH"
|
# into hardened stage without worrying about permissions and keeps image size small. Keeping the commands
|
||||||
|
# together makes for a slightly smaller image size.
|
||||||
|
RUN pip install -r /tmp/requirements.txt && \
|
||||||
|
chmod -R u-rwx,g-rwx /opt
|
||||||
|
|
||||||
|
# second stage is the main runtime stage with just the minimum required to run the application
|
||||||
RUN pip install openwrt-luci-rpc asusrouter asyncio aiohttp graphene flask flask-cors unifi-sm-api tplink-omada-client wakeonlan pycryptodome requests paho-mqtt scapy cron-converter pytz json2table dhcp-leases pyunifi speedtest-cli chardet python-nmap dnspython librouteros yattag zeroconf git+https://github.com/foreign-sub/aiofreepybox.git
|
# The runner is used for both devcontainer, and as a base for the hardened stage.
|
||||||
|
|
||||||
# Append Iliadbox certificate to aiofreepybox
|
|
||||||
|
|
||||||
# second stage
|
|
||||||
FROM alpine:3.22 AS runner
|
FROM alpine:3.22 AS runner
|
||||||
|
|
||||||
ARG INSTALL_DIR=/app
|
ARG INSTALL_DIR=/app
|
||||||
|
|
||||||
COPY --from=builder /opt/venv /opt/venv
|
# NetAlertX app directories
|
||||||
COPY --from=builder /usr/sbin/usermod /usr/sbin/groupmod /usr/sbin/
|
ENV NETALERTX_APP=${INSTALL_DIR}
|
||||||
|
ENV NETALERTX_CONFIG=${NETALERTX_APP}/config
|
||||||
|
ENV NETALERTX_FRONT=${NETALERTX_APP}/front
|
||||||
|
ENV NETALERTX_SERVER=${NETALERTX_APP}/server
|
||||||
|
ENV NETALERTX_API=${NETALERTX_APP}/api
|
||||||
|
ENV NETALERTX_DB=${NETALERTX_APP}/db
|
||||||
|
ENV NETALERTX_DB_FILE=${NETALERTX_DB}/app.db
|
||||||
|
ENV NETALERTX_BACK=${NETALERTX_APP}/back
|
||||||
|
ENV NETALERTX_LOG=${NETALERTX_APP}/log
|
||||||
|
ENV NETALERTX_PLUGINS_LOG=${NETALERTX_LOG}/plugins
|
||||||
|
ENV NETALERTX_CONFIG_FILE=${NETALERTX_CONFIG}/app.conf
|
||||||
|
|
||||||
# Enable venv
|
# NetAlertX log files
|
||||||
ENV PATH="/opt/venv/bin:$PATH"
|
ENV LOG_IP_CHANGES=${NETALERTX_LOG}/IP_changes.log
|
||||||
|
ENV LOG_APP=${NETALERTX_LOG}/app.log
|
||||||
|
ENV LOG_APP_FRONT=${NETALERTX_LOG}/app_front.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_TXT=${NETALERTX_LOG}/report_output.txt
|
||||||
|
ENV LOG_DB_IS_LOCKED=${NETALERTX_LOG}/db_is_locked.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_HTML=${NETALERTX_LOG}/report_output.html
|
||||||
|
ENV LOG_STDERR=${NETALERTX_LOG}/stderr.log
|
||||||
|
ENV LOG_APP_PHP_ERRORS=${NETALERTX_LOG}/app.php_errors.log
|
||||||
|
ENV LOG_EXECUTION_QUEUE=${NETALERTX_LOG}/execution_queue.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_JSON=${NETALERTX_LOG}/report_output.json
|
||||||
|
ENV LOG_STDOUT=${NETALERTX_LOG}/stdout.log
|
||||||
|
ENV LOG_CROND=${NETALERTX_LOG}/crond.log
|
||||||
|
|
||||||
# default port and listen address
|
# System Services configuration files
|
||||||
ENV PORT=20211 LISTEN_ADDR=0.0.0.0
|
ENV SYSTEM_SERVICES=/services
|
||||||
|
ENV SYSTEM_SERVICES_SCRIPTS=${SYSTEM_SERVICES}/scripts
|
||||||
|
ENV SYSTEM_SERVICES_CONFIG=${SYSTEM_SERVICES}/config
|
||||||
|
ENV SYSTEM_NGINX_CONFIG=${SYSTEM_SERVICES_CONFIG}/nginx
|
||||||
|
ENV SYSTEM_NGINX_CONFIG_FILE=${SYSTEM_NGINX_CONFIG}/nginx.conf
|
||||||
|
ENV SYSTEM_SERVICES_PHP_FOLDER=${SYSTEM_SERVICES_CONFIG}/php
|
||||||
|
ENV SYSTEM_SERVICES_PHP_FPM_D=${SYSTEM_SERVICES_PHP_FOLDER}/php-fpm.d
|
||||||
|
ENV SYSTEM_SERVICES_CROND=${SYSTEM_SERVICES_CONFIG}/crond
|
||||||
|
ENV SYSTEM_SERVICES_RUN=${SYSTEM_SERVICES}/run
|
||||||
|
ENV SYSTEM_SERVICES_RUN_TMP=${SYSTEM_SERVICES_RUN}/tmp
|
||||||
|
ENV SYSTEM_SERVICES_RUN_LOG=${SYSTEM_SERVICES_RUN}/logs
|
||||||
|
ENV PHP_FPM_CONFIG_FILE=${SYSTEM_SERVICES_PHP_FOLDER}/php-fpm.conf
|
||||||
|
ENV READ_ONLY_FOLDERS="${NETALERTX_BACK} ${NETALERTX_FRONT} ${NETALERTX_SERVER} ${SYSTEM_SERVICES} \
|
||||||
|
${SYSTEM_SERVICES_CONFIG}"
|
||||||
|
ENV READ_WRITE_FOLDERS="${NETALERTX_CONFIG} ${NETALERTX_DB} ${NETALERTX_API} ${NETALERTX_LOG} \
|
||||||
|
${NETALERTX_PLUGINS_LOG} ${SYSTEM_SERVICES_RUN} ${SYSTEM_SERVICES_RUN_TMP} \
|
||||||
|
${SYSTEM_SERVICES_RUN_LOG}"
|
||||||
|
|
||||||
# needed for s6-overlay
|
#Python environment
|
||||||
ENV S6_CMD_WAIT_FOR_SERVICES_MAXTIME=0
|
ENV PYTHONUNBUFFERED=1
|
||||||
|
ENV VIRTUAL_ENV=/opt/venv
|
||||||
|
ENV VIRTUAL_ENV_BIN=/opt/venv/bin
|
||||||
|
ENV PYTHONPATH=${NETALERTX_APP}:${NETALERTX_SERVER}:${VIRTUAL_ENV}/lib/python3.12/site-packages
|
||||||
|
ENV PATH="${SYSTEM_SERVICES}:${VIRTUAL_ENV_BIN}:$PATH"
|
||||||
|
|
||||||
# ❗ IMPORTANT - if you modify this file modify the /install/install_dependecies.sh file as well ❗
|
# App Environment
|
||||||
|
ENV LISTEN_ADDR=0.0.0.0
|
||||||
RUN apk update --no-cache \
|
ENV PORT=20211
|
||||||
&& apk add --no-cache bash libbsd zip lsblk gettext-envsubst sudo mtr tzdata s6-overlay \
|
ENV NETALERTX_DEBUG=0
|
||||||
&& apk add --no-cache curl arp-scan iproute2 iproute2-ss nmap nmap-scripts traceroute nbtscan net-tools net-snmp-tools bind-tools awake ca-certificates \
|
ENV VENDORSPATH=/app/back/ieee-oui.txt
|
||||||
&& apk add --no-cache sqlite php83 php83-fpm php83-cgi php83-curl php83-sqlite3 php83-session \
|
ENV VENDORSPATH_NEWEST=/services/run/tmp/ieee-oui.txt
|
||||||
&& apk add --no-cache python3 nginx \
|
ENV PYTHONPATHPATH="${NETALERTX_APP}:${VIRTUAL_ENV}/bin:${PATH}"
|
||||||
&& ln -s /usr/bin/awake /usr/bin/wakeonlan \
|
ENV ENVIRONMENT=alpine
|
||||||
&& rm -f /etc/nginx/http.d/default.conf
|
ENV READ_ONLY_USER=readonly READ_ONLY_GROUP=readonly
|
||||||
|
ENV NETALERTX_USER=netalertx NETALERTX_GROUP=netalertx
|
||||||
|
ENV LANG=C.UTF-8
|
||||||
|
|
||||||
|
|
||||||
# Add crontab file
|
RUN apk add --no-cache bash mtr libbsd zip lsblk tzdata curl arp-scan iproute2 iproute2-ss nmap \
|
||||||
COPY --chmod=600 --chown=root:root install/crontab /etc/crontabs/root
|
nmap-scripts traceroute nbtscan net-tools net-snmp-tools bind-tools awake ca-certificates \
|
||||||
|
sqlite php83 php83-fpm php83-cgi php83-curl php83-sqlite3 php83-session python3 envsubst \
|
||||||
|
nginx shadow && \
|
||||||
|
rm -Rf /var/cache/apk/* && \
|
||||||
|
rm -Rf /etc/nginx && \
|
||||||
|
addgroup -g 20211 ${NETALERTX_GROUP} && \
|
||||||
|
adduser -u 20211 -D -h ${NETALERTX_APP} -G ${NETALERTX_GROUP} ${NETALERTX_USER} && \
|
||||||
|
apk del shadow
|
||||||
|
|
||||||
# Start all required services
|
|
||||||
|
|
||||||
HEALTHCHECK --interval=30s --timeout=5s --start-period=15s --retries=2 \
|
|
||||||
CMD curl -sf -o /dev/null ${LISTEN_ADDR}:${PORT}/php/server/query_json.php?file=app_state.json
|
|
||||||
|
|
||||||
ENTRYPOINT ["/init"]
|
# Install application, copy files, set permissions
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} install/production-filesystem/ /
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} --chmod=755 back ${NETALERTX_BACK}
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} --chmod=755 front ${NETALERTX_FRONT}
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} --chmod=755 server ${NETALERTX_SERVER}
|
||||||
|
RUN install -d -o ${NETALERTX_USER} -g ${NETALERTX_GROUP} -m 755 ${NETALERTX_API} \
|
||||||
|
${NETALERTX_LOG} ${SYSTEM_SERVICES_RUN_TMP} ${SYSTEM_SERVICES_RUN_LOG} && \
|
||||||
|
sh -c "find ${NETALERTX_APP} -type f \( -name '*.sh' -o -name 'speedtest-cli' \) \
|
||||||
|
-exec chmod 750 {} \;"
|
||||||
|
|
||||||
|
# Copy the virtualenv from the builder stage
|
||||||
|
COPY --from=builder --chown=20212:20212 ${VIRTUAL_ENV} ${VIRTUAL_ENV}
|
||||||
|
|
||||||
|
|
||||||
|
# Initialize each service with the dockerfiles/init-*.sh scripts, once.
|
||||||
|
# This is done after the copy of the venv to ensure the venv is in place
|
||||||
|
# although it may be quicker to do it before the copy, it keeps the image
|
||||||
|
# layers smaller to do it after.
|
||||||
|
RUN apk add libcap && \
|
||||||
|
setcap cap_net_raw+ep /bin/busybox && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip /usr/bin/nmap && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip /usr/bin/arp-scan && \
|
||||||
|
setcap cap_net_raw,cap_net_admin,cap_net_bind_service+eip /usr/bin/nbtscan && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip /usr/bin/traceroute && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip ${VIRTUAL_ENV_BIN}/scapy && \
|
||||||
|
/bin/sh /build/init-nginx.sh && \
|
||||||
|
/bin/sh /build/init-php-fpm.sh && \
|
||||||
|
/bin/sh /build/init-crond.sh && \
|
||||||
|
/bin/sh /build/init-backend.sh && \
|
||||||
|
rm -rf /build && \
|
||||||
|
apk del libcap
|
||||||
|
|
||||||
|
|
||||||
|
ENTRYPOINT ["/bin/sh","/entrypoint.sh"]
|
||||||
|
|
||||||
|
# Final hardened stage to improve security by setting least possible permissions and removing sudo access.
|
||||||
|
# When complete, if the image is compromised, there's not much that can be done with it.
|
||||||
|
# This stage is separate from Runner stage so that devcontainer can use the Runner stage.
|
||||||
|
FROM runner AS hardened
|
||||||
|
|
||||||
|
ENV UMASK=0077
|
||||||
|
|
||||||
|
# Create readonly user and group with no shell access.
|
||||||
|
# Readonly user marks folders that are created by NetAlertX, but should not be modified.
|
||||||
|
# AI may claim this is stupid, but it's actually least possible permissions as
|
||||||
|
# read-only user cannot login, cannot sudo, has no write permission, and cannot even
|
||||||
|
# read the files it owns. The read-only user is ownership-as-a-lock hardening pattern.
|
||||||
|
RUN addgroup -g 20212 ${READ_ONLY_GROUP} && \
|
||||||
|
adduser -u 20212 -G ${READ_ONLY_GROUP} -D -h /app ${READ_ONLY_USER}
|
||||||
|
|
||||||
|
|
||||||
|
# reduce permissions to minimum necessary for all NetAlertX files and folders
|
||||||
|
# Permissions 005 and 004 are not typos, they enable read-only. Everyone can
|
||||||
|
# read the read-only files, and nobody can write to them, even the readonly user.
|
||||||
|
RUN chown -R ${READ_ONLY_USER}:${READ_ONLY_GROUP} ${READ_ONLY_FOLDERS} && \
|
||||||
|
chmod -R 004 ${READ_ONLY_FOLDERS} && \
|
||||||
|
find ${READ_ONLY_FOLDERS} -type d -exec chmod 005 {} + && \
|
||||||
|
install -d -o ${NETALERTX_USER} -g ${NETALERTX_GROUP} -m 700 ${READ_WRITE_FOLDERS} && \
|
||||||
|
chown -R ${NETALERTX_USER}:${NETALERTX_GROUP} ${READ_WRITE_FOLDERS} && \
|
||||||
|
chmod -R 600 ${READ_WRITE_FOLDERS} && \
|
||||||
|
find ${READ_WRITE_FOLDERS} -type d -exec chmod 700 {} + && \
|
||||||
|
chown ${READ_ONLY_USER}:${READ_ONLY_GROUP} /entrypoint.sh /opt /opt/venv && \
|
||||||
|
chmod 005 /entrypoint.sh ${SYSTEM_SERVICES}/*.sh /app /opt /opt/venv && \
|
||||||
|
apk del apk-tools && \
|
||||||
|
rm -Rf /var /etc/sudoers.d/* /etc/shadow /etc/gshadow /etc/sudoers \
|
||||||
|
/lib/apk /lib/firmware /lib/modules-load.d /lib/sysctl.d /mnt /home/ /root \
|
||||||
|
/srv /media && \
|
||||||
|
sed -i "/^\(${READ_ONLY_USER}\|${NETALERTX_USER}\):/!d" /etc/passwd && \
|
||||||
|
sed -i "/^\(${READ_ONLY_GROUP}\|${NETALERTX_GROUP}\):/!d" /etc/group && \
|
||||||
|
echo -ne '#!/bin/sh\n"$@"\n' > /usr/bin/sudo && chmod +x /usr/bin/sudo
|
||||||
|
|
||||||
|
USER netalertx
|
||||||
|
|
||||||
|
HEALTHCHECK --interval=30s --timeout=10s --start-period=60s --retries=3 \
|
||||||
|
CMD /services/healthcheck.sh
|
||||||
|
|
||||||
|
|
||||||
# ---/resources/devcontainer-Dockerfile---
|
# ---/resources/devcontainer-Dockerfile---
|
||||||
|
|
||||||
# Devcontainer build stage (do not build directly)
|
# Devcontainer build stage (do not build directly)
|
||||||
# This file is combined with the root /Dockerfile by
|
# This file is combined with the root /Dockerfile by
|
||||||
# .devcontainer/scripts/generate-dockerfile.sh
|
# .devcontainer/scripts/generate-configs.sh
|
||||||
# The generator appends this stage to produce .devcontainer/Dockerfile.
|
# The generator appends this stage to produce .devcontainer/Dockerfile.
|
||||||
# Prefer to place dev-only setup here; use setup.sh only for runtime fixes.
|
# Prefer to place dev-only setup here; use setup.sh only for runtime fixes.
|
||||||
|
|
||||||
FROM runner AS devcontainer
|
FROM runner AS netalertx-devcontainer
|
||||||
ENV INSTALL_DIR=/app
|
ENV INSTALL_DIR=/app
|
||||||
|
|
||||||
ENV PYTHONPATH=/workspaces/NetAlertX/test:/workspaces/NetAlertX/server:/app:/app/server:/opt/venv/lib/python3.12/site-packages
|
ENV PYTHONPATH=/workspaces/NetAlertX/test:/workspaces/NetAlertX/server:/app:/app/server:/opt/venv/lib/python3.12/site-packages
|
||||||
|
ENV PATH=/services:${PATH}
|
||||||
|
ENV PHP_INI_SCAN_DIR=/services/config/php/conf.d:/etc/php83/conf.d
|
||||||
|
ENV LISTEN_ADDR=0.0.0.0
|
||||||
|
ENV PORT=20211
|
||||||
|
ENV NETALERTX_DEBUG=1
|
||||||
|
ENV PYDEVD_DISABLE_FILE_VALIDATION=1
|
||||||
|
COPY .devcontainer/resources/devcontainer-overlay/ /
|
||||||
|
|
||||||
# Install common tools, create user, and set up sudo
|
# Install common tools, create user, and set up sudo
|
||||||
RUN apk add --no-cache git nano vim jq php83-pecl-xdebug py3-pip nodejs sudo gpgconf pytest pytest-cov && \
|
RUN apk add --no-cache git nano vim jq php83-pecl-xdebug py3-pip nodejs sudo gpgconf pytest pytest-cov fish shfmt sudo
|
||||||
adduser -D -s /bin/sh netalertx && \
|
|
||||||
addgroup netalertx nginx && \
|
RUN install -d -o netalertx -g netalertx -m 755 /services/php/modules && \
|
||||||
addgroup netalertx www-data && \
|
cp -a /usr/lib/php83/modules/. /services/php/modules/ && \
|
||||||
echo "netalertx ALL=(ALL) NOPASSWD:ALL" > /etc/sudoers.d/90-netalertx && \
|
echo "${NETALERTX_USER} ALL=(ALL) NOPASSWD: ALL" >> /etc/sudoers
|
||||||
chmod 440 /etc/sudoers.d/90-netalertx
|
|
||||||
# Install debugpy in the virtualenv if present, otherwise into system python3
|
# Install debugpy in the virtualenv if present, otherwise into system python3
|
||||||
RUN /bin/sh -c '(/opt/venv/bin/python3 -m pip install --no-cache-dir debugpy) || (python3 -m pip install --no-cache-dir debugpy) || true'
|
RUN /bin/sh -c '(/opt/venv/bin/python3 -m pip install --no-cache-dir debugpy) || (python3 -m pip install --no-cache-dir debugpy) || true' && \
|
||||||
# setup nginx
|
mkdir /workspaces && \
|
||||||
COPY .devcontainer/resources/netalertx-devcontainer.conf /etc/nginx/http.d/netalert-frontend.conf
|
install -d -o netalertx -g netalertx -m 777 /services/run/logs && \
|
||||||
RUN set -e; \
|
install -d -o netalertx -g netalertx -m 777 /app/run/tmp/client_body && \
|
||||||
chown netalertx:nginx /etc/nginx/http.d/netalert-frontend.conf; \
|
sed -i -e 's|:/app:|:/workspaces:|' /etc/passwd && \
|
||||||
install -d -o netalertx -g www-data -m 775 /app; \
|
python -m pip install -U pytest pytest-cov
|
||||||
install -d -o netalertx -g www-data -m 755 /run/nginx; \
|
|
||||||
install -d -o netalertx -g www-data -m 755 /var/lib/nginx/logs; \
|
|
||||||
rm -f /var/lib/nginx/logs/* || true; \
|
|
||||||
for f in error access; do : > /var/lib/nginx/logs/$f.log; done; \
|
|
||||||
install -d -o netalertx -g www-data -m 777 /run/php; \
|
|
||||||
install -d -o netalertx -g www-data -m 775 /var/log/php; \
|
|
||||||
chown -R netalertx:www-data /etc/nginx/http.d; \
|
|
||||||
chmod -R 775 /etc/nginx/http.d; \
|
|
||||||
chown -R netalertx:www-data /var/lib/nginx; \
|
|
||||||
chmod -R 755 /var/lib/nginx && \
|
|
||||||
chown -R netalertx:www-data /var/log/nginx/ && \
|
|
||||||
sed -i '/^user /d' /etc/nginx/nginx.conf; \
|
|
||||||
sed -i 's|^error_log .*|error_log /dev/stderr warn;|' /etc/nginx/nginx.conf; \
|
|
||||||
sed -i 's|^access_log .*|access_log /dev/stdout main;|' /etc/nginx/nginx.conf; \
|
|
||||||
sed -i 's|error_log .*|error_log /dev/stderr warn;|g' /etc/nginx/http.d/*.conf 2>/dev/null || true; \
|
|
||||||
sed -i 's|access_log .*|access_log /dev/stdout main;|g' /etc/nginx/http.d/*.conf 2>/dev/null || true; \
|
|
||||||
mkdir -p /run/openrc; \
|
|
||||||
chown netalertx:nginx /run/openrc/; \
|
|
||||||
rm -Rf /run/openrc/*;
|
|
||||||
|
|
||||||
# setup pytest
|
ENTRYPOINT ["/bin/sh","-c","sleep infinity"]
|
||||||
RUN sudo /opt/venv/bin/python -m pip install -U pytest pytest-cov
|
|
||||||
|
|
||||||
WORKDIR /workspaces/NetAlertX
|
|
||||||
|
|
||||||
|
|
||||||
ENTRYPOINT ["/bin/sh","-c","sleep infinity"]
|
|
||||||
|
|||||||
@@ -1,27 +1,47 @@
|
|||||||
{
|
{
|
||||||
"name": "NetAlertX DevContainer",
|
"name": "NetAlertX DevContainer",
|
||||||
"remoteUser": "netalertx",
|
"remoteUser": "netalertx",
|
||||||
"build": {
|
|
||||||
"dockerfile": "Dockerfile",
|
|
||||||
"context": "..",
|
|
||||||
"target": "devcontainer"
|
|
||||||
},
|
|
||||||
"workspaceFolder": "/workspaces/NetAlertX",
|
"workspaceFolder": "/workspaces/NetAlertX",
|
||||||
"runArgs": [
|
"build": {
|
||||||
"--add-host=host.docker.internal:host-gateway",
|
"dockerfile": "./Dockerfile", // Dockerfile generated by script
|
||||||
"--security-opt", "apparmor=unconfined" // for alowing ramdisk mounts
|
"context": "../", // Context is the root of the repository
|
||||||
],
|
"target": "netalertx-devcontainer"
|
||||||
|
},
|
||||||
"capAdd": [
|
"capAdd": [
|
||||||
"SYS_ADMIN", // For mounting ramdisks
|
"SYS_ADMIN", // For mounting ramdisks
|
||||||
"NET_ADMIN", // For network interface configuration
|
"NET_ADMIN", // For network interface configuration
|
||||||
"NET_RAW" // For raw packet manipulation
|
"NET_RAW" // For raw packet manipulation
|
||||||
],
|
],
|
||||||
|
"runArgs": [
|
||||||
|
"--security-opt",
|
||||||
|
"apparmor=unconfined", // for allowing ramdisk mounts
|
||||||
|
"--add-host=host.docker.internal:host-gateway"
|
||||||
|
|
||||||
|
// Uncomment --network=host to run full NetAlertX scanning capabilities of network scanning in
|
||||||
|
// container. This runs too slowly in a large network to be practical for development purposes.
|
||||||
|
// You can start services such as avahi on the host, in other containers within the network, or
|
||||||
|
// even within this container and connect to them as needed.
|
||||||
|
// "--network=host",
|
||||||
|
],
|
||||||
|
// ATTENTION: If running with --network=host, COMMENT `forwardPorts` OR ELSE THERE WILL BE NO WEBUI!
|
||||||
|
"forwardPorts": [20211, 20212, 5678],
|
||||||
|
"portsAttributes": { // the ports we care about
|
||||||
|
"20211": {
|
||||||
|
"label": "Frontend:Nginx+PHP"
|
||||||
|
},
|
||||||
|
"20212": {
|
||||||
|
"label": "Backend:GraphQL"
|
||||||
|
},
|
||||||
|
"9003": {
|
||||||
|
"label": "PHP Debug:Xdebug"
|
||||||
|
},
|
||||||
|
"5678": {
|
||||||
|
"label": "Python Debug:debugpy"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
"postStartCommand": "${containerWorkspaceFolder}/.devcontainer/scripts/setup.sh",
|
"postStartCommand": "${containerWorkspaceFolder}/.devcontainer/scripts/setup.sh",
|
||||||
|
|
||||||
"customizations": {
|
"customizations": {
|
||||||
"vscode": {
|
"vscode": {
|
||||||
"extensions": [
|
"extensions": [
|
||||||
@@ -33,17 +53,24 @@
|
|||||||
"ms-python.vscode-pylance",
|
"ms-python.vscode-pylance",
|
||||||
"pamaron.pytest-runner",
|
"pamaron.pytest-runner",
|
||||||
"coderabbit.coderabbit-vscode",
|
"coderabbit.coderabbit-vscode",
|
||||||
"ms-python.black-formatter"
|
"ms-python.black-formatter",
|
||||||
]
|
"jeff-hykin.better-dockerfile-syntax",
|
||||||
,
|
"GitHub.codespaces",
|
||||||
|
"ms-azuretools.vscode-containers",
|
||||||
|
"ms-python.vscode-python-envs",
|
||||||
|
"dbaeumer.vscode-eslint",
|
||||||
|
"esbenp.prettier-vscode",
|
||||||
|
"eamodio.gitlens",
|
||||||
|
"alexcvzz.vscode-sqlite",
|
||||||
|
"yzhang.markdown-all-in-one",
|
||||||
|
"mkhl.shfmt"
|
||||||
|
],
|
||||||
"settings": {
|
"settings": {
|
||||||
"terminal.integrated.cwd": "${containerWorkspaceFolder}",
|
"terminal.integrated.cwd": "${containerWorkspaceFolder}",
|
||||||
// Python testing configuration
|
// Python testing configuration
|
||||||
"python.testing.pytestEnabled": true,
|
"python.testing.pytestEnabled": true,
|
||||||
"python.testing.unittestEnabled": false,
|
"python.testing.unittestEnabled": false,
|
||||||
"python.testing.pytestArgs": [
|
"python.testing.pytestArgs": ["test"],
|
||||||
"test"
|
|
||||||
],
|
|
||||||
// Make sure we discover tests and import server correctly
|
// Make sure we discover tests and import server correctly
|
||||||
"python.analysis.extraPaths": [
|
"python.analysis.extraPaths": [
|
||||||
"/workspaces/NetAlertX",
|
"/workspaces/NetAlertX",
|
||||||
@@ -54,26 +81,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"forwardPorts": [5678, 9000, 9003, 20211, 20212],
|
|
||||||
|
|
||||||
"portsAttributes": {
|
"shutdownAction": "stopContainer" // stop container when VSCode is closed
|
||||||
"20211": {
|
}
|
||||||
"label": "Frontend:Nginx+PHP"
|
|
||||||
},
|
|
||||||
"20212": {
|
|
||||||
"label": "Backend:GraphQL"
|
|
||||||
},
|
|
||||||
"9003": {
|
|
||||||
"label": "PHP Debug:Xdebug"
|
|
||||||
},
|
|
||||||
"9000": {
|
|
||||||
"label": "PHP-FPM:FastCGI"
|
|
||||||
},
|
|
||||||
"5678": {
|
|
||||||
"label": "Python Debug:debugpy"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// Optional: ensures compose services are stopped when you close the window
|
|
||||||
"shutdownAction": "stopContainer"
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
zend_extension="xdebug.so"
|
|
||||||
[xdebug]
|
|
||||||
xdebug.mode=develop,debug
|
|
||||||
xdebug.log_level=0
|
|
||||||
xdebug.client_host=host.docker.internal
|
|
||||||
xdebug.client_port=9003
|
|
||||||
xdebug.start_with_request=yes
|
|
||||||
xdebug.discover_client_host=1
|
|
||||||
@@ -1,51 +1,33 @@
|
|||||||
# Devcontainer build stage (do not build directly)
|
# Devcontainer build stage (do not build directly)
|
||||||
# This file is combined with the root /Dockerfile by
|
# This file is combined with the root /Dockerfile by
|
||||||
# .devcontainer/scripts/generate-dockerfile.sh
|
# .devcontainer/scripts/generate-configs.sh
|
||||||
# The generator appends this stage to produce .devcontainer/Dockerfile.
|
# The generator appends this stage to produce .devcontainer/Dockerfile.
|
||||||
# Prefer to place dev-only setup here; use setup.sh only for runtime fixes.
|
# Prefer to place dev-only setup here; use setup.sh only for runtime fixes.
|
||||||
|
|
||||||
FROM runner AS devcontainer
|
FROM runner AS netalertx-devcontainer
|
||||||
ENV INSTALL_DIR=/app
|
ENV INSTALL_DIR=/app
|
||||||
|
|
||||||
ENV PYTHONPATH=/workspaces/NetAlertX/test:/workspaces/NetAlertX/server:/app:/app/server:/opt/venv/lib/python3.12/site-packages
|
ENV PYTHONPATH=/workspaces/NetAlertX/test:/workspaces/NetAlertX/server:/app:/app/server:/opt/venv/lib/python3.12/site-packages
|
||||||
|
ENV PATH=/services:${PATH}
|
||||||
|
ENV PHP_INI_SCAN_DIR=/services/config/php/conf.d:/etc/php83/conf.d
|
||||||
|
ENV LISTEN_ADDR=0.0.0.0
|
||||||
|
ENV PORT=20211
|
||||||
|
ENV NETALERTX_DEBUG=1
|
||||||
|
ENV PYDEVD_DISABLE_FILE_VALIDATION=1
|
||||||
|
COPY .devcontainer/resources/devcontainer-overlay/ /
|
||||||
|
|
||||||
# Install common tools, create user, and set up sudo
|
# Install common tools, create user, and set up sudo
|
||||||
RUN apk add --no-cache git nano vim jq php83-pecl-xdebug py3-pip nodejs sudo gpgconf pytest pytest-cov && \
|
RUN apk add --no-cache git nano vim jq php83-pecl-xdebug py3-pip nodejs sudo gpgconf pytest pytest-cov fish shfmt github-cli
|
||||||
adduser -D -s /bin/sh netalertx && \
|
|
||||||
addgroup netalertx nginx && \
|
RUN install -d -o netalertx -g netalertx -m 755 /services/php/modules && \
|
||||||
addgroup netalertx www-data && \
|
cp -a /usr/lib/php83/modules/. /services/php/modules/ && \
|
||||||
echo "netalertx ALL=(ALL) NOPASSWD:ALL" > /etc/sudoers.d/90-netalertx && \
|
echo "${NETALERTX_USER} ALL=(ALL) NOPASSWD: ALL" >> /etc/sudoers
|
||||||
chmod 440 /etc/sudoers.d/90-netalertx
|
|
||||||
# Install debugpy in the virtualenv if present, otherwise into system python3
|
# Install debugpy in the virtualenv if present, otherwise into system python3
|
||||||
RUN /bin/sh -c '(/opt/venv/bin/python3 -m pip install --no-cache-dir debugpy) || (python3 -m pip install --no-cache-dir debugpy) || true'
|
RUN /bin/sh -c '(/opt/venv/bin/python3 -m pip install --no-cache-dir debugpy) || (python3 -m pip install --no-cache-dir debugpy) || true' && \
|
||||||
# setup nginx
|
mkdir /workspaces && \
|
||||||
COPY .devcontainer/resources/netalertx-devcontainer.conf /etc/nginx/http.d/netalert-frontend.conf
|
install -d -o netalertx -g netalertx -m 777 /services/run/logs && \
|
||||||
RUN set -e; \
|
install -d -o netalertx -g netalertx -m 777 /app/run/tmp/client_body && \
|
||||||
chown netalertx:nginx /etc/nginx/http.d/netalert-frontend.conf; \
|
sed -i -e 's|:/app:|:/workspaces:|' /etc/passwd && \
|
||||||
install -d -o netalertx -g www-data -m 775 /app; \
|
python -m pip install -U pytest pytest-cov
|
||||||
install -d -o netalertx -g www-data -m 755 /run/nginx; \
|
|
||||||
install -d -o netalertx -g www-data -m 755 /var/lib/nginx/logs; \
|
|
||||||
rm -f /var/lib/nginx/logs/* || true; \
|
|
||||||
for f in error access; do : > /var/lib/nginx/logs/$f.log; done; \
|
|
||||||
install -d -o netalertx -g www-data -m 777 /run/php; \
|
|
||||||
install -d -o netalertx -g www-data -m 775 /var/log/php; \
|
|
||||||
chown -R netalertx:www-data /etc/nginx/http.d; \
|
|
||||||
chmod -R 775 /etc/nginx/http.d; \
|
|
||||||
chown -R netalertx:www-data /var/lib/nginx; \
|
|
||||||
chmod -R 755 /var/lib/nginx && \
|
|
||||||
chown -R netalertx:www-data /var/log/nginx/ && \
|
|
||||||
sed -i '/^user /d' /etc/nginx/nginx.conf; \
|
|
||||||
sed -i 's|^error_log .*|error_log /dev/stderr warn;|' /etc/nginx/nginx.conf; \
|
|
||||||
sed -i 's|^access_log .*|access_log /dev/stdout main;|' /etc/nginx/nginx.conf; \
|
|
||||||
sed -i 's|error_log .*|error_log /dev/stderr warn;|g' /etc/nginx/http.d/*.conf 2>/dev/null || true; \
|
|
||||||
sed -i 's|access_log .*|access_log /dev/stdout main;|g' /etc/nginx/http.d/*.conf 2>/dev/null || true; \
|
|
||||||
mkdir -p /run/openrc; \
|
|
||||||
chown netalertx:nginx /run/openrc/; \
|
|
||||||
rm -Rf /run/openrc/*;
|
|
||||||
|
|
||||||
# setup pytest
|
ENTRYPOINT ["/bin/sh","-c","sleep infinity"]
|
||||||
RUN sudo /opt/venv/bin/python -m pip install -U pytest pytest-cov
|
|
||||||
|
|
||||||
WORKDIR /workspaces/NetAlertX
|
|
||||||
|
|
||||||
|
|
||||||
ENTRYPOINT ["/bin/sh","-c","sleep infinity"]
|
|
||||||
|
|||||||
@@ -0,0 +1,118 @@
|
|||||||
|
# DO NOT MODIFY THIS FILE DIRECTLY. IT IS AUTO-GENERATED BY .devcontainer/scripts/generate-configs.sh
|
||||||
|
# Generated from: install/production-filesystem/services/config/nginx/netalertx.conf.template
|
||||||
|
|
||||||
|
# Set number of worker processes automatically based on number of CPU cores.
|
||||||
|
worker_processes auto;
|
||||||
|
|
||||||
|
# Enables the use of JIT for regular expressions to speed-up their processing.
|
||||||
|
pcre_jit on;
|
||||||
|
|
||||||
|
# Configures default error logger.
|
||||||
|
error_log /app/log/nginx-error.log warn;
|
||||||
|
|
||||||
|
events {
|
||||||
|
# The maximum number of simultaneous connections that can be opened by
|
||||||
|
# a worker process.
|
||||||
|
worker_connections 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
http {
|
||||||
|
|
||||||
|
# Mapping of temp paths for various nginx modules.
|
||||||
|
client_body_temp_path /services/run/tmp/client_body;
|
||||||
|
proxy_temp_path /services/run/tmp/proxy;
|
||||||
|
fastcgi_temp_path /services/run/tmp/fastcgi;
|
||||||
|
uwsgi_temp_path /services/run/tmp/uwsgi;
|
||||||
|
scgi_temp_path /services/run/tmp/scgi;
|
||||||
|
|
||||||
|
# Includes mapping of file name extensions to MIME types of responses
|
||||||
|
# and defines the default type.
|
||||||
|
include /services/config/nginx/mime.types;
|
||||||
|
default_type application/octet-stream;
|
||||||
|
|
||||||
|
# Name servers used to resolve names of upstream servers into addresses.
|
||||||
|
# It's also needed when using tcpsocket and udpsocket in Lua modules.
|
||||||
|
#resolver 1.1.1.1 1.0.0.1 [2606:4700:4700::1111] [2606:4700:4700::1001];
|
||||||
|
|
||||||
|
# Don't tell nginx version to the clients. Default is 'on'.
|
||||||
|
server_tokens off;
|
||||||
|
|
||||||
|
# Specifies the maximum accepted body size of a client request, as
|
||||||
|
# indicated by the request header Content-Length. If the stated content
|
||||||
|
# length is greater than this size, then the client receives the HTTP
|
||||||
|
# error code 413. Set to 0 to disable. Default is '1m'.
|
||||||
|
client_max_body_size 1m;
|
||||||
|
|
||||||
|
# Sendfile copies data between one FD and other from within the kernel,
|
||||||
|
# which is more efficient than read() + write(). Default is off.
|
||||||
|
sendfile on;
|
||||||
|
|
||||||
|
# Causes nginx to attempt to send its HTTP response head in one packet,
|
||||||
|
# instead of using partial frames. Default is 'off'.
|
||||||
|
tcp_nopush on;
|
||||||
|
|
||||||
|
|
||||||
|
# Enables the specified protocols. Default is TLSv1 TLSv1.1 TLSv1.2.
|
||||||
|
# TIP: If you're not obligated to support ancient clients, remove TLSv1.1.
|
||||||
|
ssl_protocols TLSv1.2 TLSv1.3;
|
||||||
|
|
||||||
|
# Path of the file with Diffie-Hellman parameters for EDH ciphers.
|
||||||
|
# TIP: Generate with: `openssl dhparam -out /etc/ssl/nginx/dh2048.pem 2048`
|
||||||
|
#ssl_dhparam /etc/ssl/nginx/dh2048.pem;
|
||||||
|
|
||||||
|
# Specifies that our cipher suits should be preferred over client ciphers.
|
||||||
|
# Default is 'off'.
|
||||||
|
ssl_prefer_server_ciphers on;
|
||||||
|
|
||||||
|
# Enables a shared SSL cache with size that can hold around 8000 sessions.
|
||||||
|
# Default is 'none'.
|
||||||
|
ssl_session_cache shared:SSL:2m;
|
||||||
|
|
||||||
|
# Specifies a time during which a client may reuse the session parameters.
|
||||||
|
# Default is '5m'.
|
||||||
|
ssl_session_timeout 1h;
|
||||||
|
|
||||||
|
# Disable TLS session tickets (they are insecure). Default is 'on'.
|
||||||
|
ssl_session_tickets off;
|
||||||
|
|
||||||
|
|
||||||
|
# Enable gzipping of responses.
|
||||||
|
gzip on;
|
||||||
|
|
||||||
|
# Set the Vary HTTP header as defined in the RFC 2616. Default is 'off'.
|
||||||
|
gzip_vary on;
|
||||||
|
|
||||||
|
|
||||||
|
# Specifies the main log format.
|
||||||
|
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
|
||||||
|
'$status $body_bytes_sent "$http_referer" '
|
||||||
|
'"$http_user_agent" "$http_x_forwarded_for"';
|
||||||
|
|
||||||
|
# Sets the path, format, and configuration for a buffered log write.
|
||||||
|
access_log /app/log/nginx-access.log main;
|
||||||
|
|
||||||
|
|
||||||
|
# Virtual host config
|
||||||
|
server {
|
||||||
|
listen 0.0.0.0:20211 default_server;
|
||||||
|
large_client_header_buffers 4 16k;
|
||||||
|
root /app/front;
|
||||||
|
index index.php;
|
||||||
|
add_header X-Forwarded-Prefix "/app" always;
|
||||||
|
|
||||||
|
|
||||||
|
location ~* \.php$ {
|
||||||
|
# Set Cache-Control header to prevent caching on the first load
|
||||||
|
add_header Cache-Control "no-store";
|
||||||
|
fastcgi_pass unix:/services/run/php.sock;
|
||||||
|
include /services/config/nginx/fastcgi_params;
|
||||||
|
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
||||||
|
fastcgi_param SCRIPT_NAME $fastcgi_script_name;
|
||||||
|
|
||||||
|
fastcgi_param PHP_VALUE "xdebug.remote_enable=1";
|
||||||
|
fastcgi_connect_timeout 75;
|
||||||
|
fastcgi_send_timeout 600;
|
||||||
|
fastcgi_read_timeout 600;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
zend_extension="/services/php/modules/xdebug.so"
|
||||||
|
extension_dir="/services/php/modules"
|
||||||
|
|
||||||
|
[xdebug]
|
||||||
|
xdebug.mode=develop,debug
|
||||||
|
xdebug.log=/app/log/xdebug.log
|
||||||
|
xdebug.log_level=7
|
||||||
|
xdebug.client_host=127.0.0.1
|
||||||
|
xdebug.client_port=9003
|
||||||
|
xdebug.start_with_request=yes
|
||||||
|
xdebug.discover_client_host=0
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
-m debugpy --listen 0.0.0.0:5678
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
log_format netalertx '$remote_addr - $remote_user [$time_local] "$request" '
|
|
||||||
'$status $body_bytes_sent "$http_referer" '
|
|
||||||
'"$http_user_agent" "$http_x_forwarded_for"';
|
|
||||||
access_log /var/log/nginx/access.log netalertx flush=1s;
|
|
||||||
error_log /var/log/nginx/error.log warn;
|
|
||||||
|
|
||||||
server {
|
|
||||||
listen 20211 default_server;
|
|
||||||
root /app/front;
|
|
||||||
index index.php;
|
|
||||||
|
|
||||||
add_header X-Forwarded-Prefix "/netalertx" always;
|
|
||||||
proxy_set_header X-Forwarded-Prefix "/netalertx";
|
|
||||||
|
|
||||||
location ~* \.php$ {
|
|
||||||
add_header Cache-Control "no-store";
|
|
||||||
fastcgi_pass 127.0.0.1:9000;
|
|
||||||
include fastcgi_params;
|
|
||||||
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
|
||||||
fastcgi_param SCRIPT_NAME $fastcgi_script_name;
|
|
||||||
fastcgi_param PHP_VALUE "xdebug.remote_enable=1";
|
|
||||||
fastcgi_connect_timeout 75;
|
|
||||||
fastcgi_send_timeout 600;
|
|
||||||
fastcgi_read_timeout 600;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
14
.devcontainer/scripts/confirm-docker-prune.sh
Executable file
14
.devcontainer/scripts/confirm-docker-prune.sh
Executable file
@@ -0,0 +1,14 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
read -r -p "Are you sure you want to destroy your host docker containers and images? Type YES to continue: " reply
|
||||||
|
|
||||||
|
if [[ "${reply}" == "YES" ]]; then
|
||||||
|
docker system prune -af
|
||||||
|
docker builder prune -af
|
||||||
|
else
|
||||||
|
echo "Aborted."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Done."
|
||||||
62
.devcontainer/scripts/generate-configs.sh
Executable file
62
.devcontainer/scripts/generate-configs.sh
Executable file
@@ -0,0 +1,62 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# Generator for .devcontainer/Dockerfile
|
||||||
|
# Combines the root /Dockerfile (with some COPY lines removed) and
|
||||||
|
# the dev-only stage in .devcontainer/resources/devcontainer-Dockerfile.
|
||||||
|
# Run this script after modifying the resource Dockerfile to refresh
|
||||||
|
# the final .devcontainer/Dockerfile used by the devcontainer.
|
||||||
|
|
||||||
|
echo "Generating .devcontainer/Dockerfile"
|
||||||
|
SCRIPT_DIR="$(CDPATH= cd -- "$(dirname -- "$0")" && pwd)"
|
||||||
|
DEVCONTAINER_DIR="${SCRIPT_DIR%/scripts}"
|
||||||
|
ROOT_DIR="${DEVCONTAINER_DIR%/.devcontainer}"
|
||||||
|
|
||||||
|
OUT_FILE="${DEVCONTAINER_DIR}/Dockerfile"
|
||||||
|
|
||||||
|
echo "Adding base Dockerfile from $ROOT_DIR..."
|
||||||
|
|
||||||
|
echo "# DO NOT MODIFY THIS FILE DIRECTLY. IT IS AUTO-GENERATED BY .devcontainer/scripts/generate-configs.sh" > "$OUT_FILE"
|
||||||
|
echo "" >> "$OUT_FILE"
|
||||||
|
echo "# ---/Dockerfile---" >> "$OUT_FILE"
|
||||||
|
|
||||||
|
cat "${ROOT_DIR}/Dockerfile" >> "$OUT_FILE"
|
||||||
|
|
||||||
|
echo "" >> "$OUT_FILE"
|
||||||
|
echo "# ---/resources/devcontainer-Dockerfile---" >> "$OUT_FILE"
|
||||||
|
echo "" >> "$OUT_FILE"
|
||||||
|
|
||||||
|
echo "Adding devcontainer-Dockerfile from $DEVCONTAINER_DIR/resources..."
|
||||||
|
cat "${DEVCONTAINER_DIR}/resources/devcontainer-Dockerfile" >> "$OUT_FILE"
|
||||||
|
|
||||||
|
echo "Generated $OUT_FILE using root dir $ROOT_DIR" >&2
|
||||||
|
|
||||||
|
# Generate devcontainer nginx config from production template
|
||||||
|
echo "Generating devcontainer nginx config"
|
||||||
|
NGINX_TEMPLATE="${ROOT_DIR}/install/production-filesystem/services/config/nginx/netalertx.conf.template"
|
||||||
|
NGINX_OUT="${DEVCONTAINER_DIR}/resources/devcontainer-overlay/services/config/nginx/netalertx.conf.template"
|
||||||
|
|
||||||
|
# Create output directory if it doesn't exist
|
||||||
|
mkdir -p "$(dirname "$NGINX_OUT")"
|
||||||
|
|
||||||
|
# Start with header comment
|
||||||
|
cat > "$NGINX_OUT" << 'EOF'
|
||||||
|
# DO NOT MODIFY THIS FILE DIRECTLY. IT IS AUTO-GENERATED BY .devcontainer/scripts/generate-configs.sh
|
||||||
|
# Generated from: install/production-filesystem/services/config/nginx/netalertx.conf.template
|
||||||
|
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# Process the template: replace listen directive and inject Xdebug params
|
||||||
|
sed 's/${LISTEN_ADDR}:${PORT}/0.0.0.0:20211/g' "$NGINX_TEMPLATE" | \
|
||||||
|
awk '
|
||||||
|
/fastcgi_param SCRIPT_NAME \$fastcgi_script_name;/ {
|
||||||
|
print $0
|
||||||
|
print ""
|
||||||
|
print " fastcgi_param PHP_VALUE \"xdebug.remote_enable=1\";"
|
||||||
|
next
|
||||||
|
}
|
||||||
|
{ print }
|
||||||
|
' >> "$NGINX_OUT"
|
||||||
|
|
||||||
|
echo "Generated $NGINX_OUT from $NGINX_TEMPLATE" >&2
|
||||||
|
|
||||||
|
echo "Done."
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
# Generator for .devcontainer/Dockerfile
|
|
||||||
# Combines the root /Dockerfile (with some COPY lines removed) and
|
|
||||||
# the dev-only stage in .devcontainer/resources/devcontainer-Dockerfile.
|
|
||||||
# Run this script after modifying the resource Dockerfile to refresh
|
|
||||||
# the final .devcontainer/Dockerfile used by the devcontainer.
|
|
||||||
|
|
||||||
# Make a copy of the original Dockerfile to the .devcontainer folder
|
|
||||||
# but remove the COPY . ${INSTALL_DIR}/ command from it. This avoids
|
|
||||||
# overwriting /app (which uses symlinks to the workspace) and preserves
|
|
||||||
# debugging capabilities inside the devcontainer.
|
|
||||||
|
|
||||||
SCRIPT_DIR="$(CDPATH= cd -- "$(dirname -- "$0")" && pwd)"
|
|
||||||
DEVCONTAINER_DIR="${SCRIPT_DIR%/scripts}"
|
|
||||||
ROOT_DIR="${DEVCONTAINER_DIR%/.devcontainer}"
|
|
||||||
|
|
||||||
OUT_FILE="${DEVCONTAINER_DIR}/Dockerfile"
|
|
||||||
|
|
||||||
echo "# DO NOT MODIFY THIS FILE DIRECTLY. IT IS AUTO-GENERATED BY .devcontainer/scripts/generate-dockerfile.sh" > "$OUT_FILE"
|
|
||||||
echo "" >> "$OUT_FILE"
|
|
||||||
echo "# ---/Dockerfile---" >> "$OUT_FILE"
|
|
||||||
|
|
||||||
sed '/${INSTALL_DIR}/d' "${ROOT_DIR}/Dockerfile" >> "$OUT_FILE"
|
|
||||||
|
|
||||||
# sed the line https://github.com/foreign-sub/aiofreepybox.git \\ to remove trailing backslash
|
|
||||||
sed -i '/aiofreepybox.git/ s/ \\$//' "$OUT_FILE"
|
|
||||||
|
|
||||||
# don't cat the file, just copy it in because it doesn't exist at build time
|
|
||||||
sed -i 's|^ RUN cat ${INSTALL_DIR}/install/freebox_certificate.pem >> /opt/venv/lib/python3.12/site-packages/aiofreepybox/freebox_certificates.pem$| COPY install/freebox_certificate.pem /opt/venv/lib/python3.12/site-packages/aiofreepybox/freebox_certificates.pem |' "$OUT_FILE"
|
|
||||||
|
|
||||||
echo "" >> "$OUT_FILE"
|
|
||||||
echo "# ---/resources/devcontainer-Dockerfile---" >> "$OUT_FILE"
|
|
||||||
echo "" >> "$OUT_FILE"
|
|
||||||
|
|
||||||
cat "${DEVCONTAINER_DIR}/resources/devcontainer-Dockerfile" >> "$OUT_FILE"
|
|
||||||
|
|
||||||
echo "Generated $OUT_FILE using root dir $ROOT_DIR" >&2
|
|
||||||
8
.devcontainer/scripts/isDevContainer.sh
Executable file
8
.devcontainer/scripts/isDevContainer.sh
Executable file
@@ -0,0 +1,8 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
if [ ! -d /workspaces/NetAlertX/.devcontainer ]; then
|
||||||
|
echo ---------------------------------------------------
|
||||||
|
echo "This script may only be run inside a devcontainer."
|
||||||
|
echo "Not in a devcontainer, exiting..."
|
||||||
|
echo ---------------------------------------------------
|
||||||
|
exit 255
|
||||||
|
fi
|
||||||
@@ -1,26 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
# Start (or restart) the NetAlertX Python backend under debugpy in background.
|
|
||||||
# This script is invoked by the VS Code task "Restart GraphQL".
|
|
||||||
# It exists to avoid complex inline command chains that were being mangled by the task runner.
|
|
||||||
|
|
||||||
set -e
|
|
||||||
|
|
||||||
LOG_DIR=/app/log
|
|
||||||
APP_DIR=/app/server
|
|
||||||
PY=python3
|
|
||||||
PORT_DEBUG=5678
|
|
||||||
|
|
||||||
# Kill any prior debug/run instances
|
|
||||||
sudo killall python3 2>/dev/null || true
|
|
||||||
sleep 2
|
|
||||||
|
|
||||||
echo ''|tee $LOG_DIR/stdout.log $LOG_DIR/stderr.log $LOG_DIR/app.log
|
|
||||||
|
|
||||||
cd "$APP_DIR"
|
|
||||||
|
|
||||||
# Launch using absolute module path for clarity; rely on cwd for local imports
|
|
||||||
setsid nohup "${PY}" -m debugpy --listen "0.0.0.0:${PORT_DEBUG}" /app/server/__main__.py \
|
|
||||||
1>>"$LOG_DIR/stdout.log" \
|
|
||||||
2>>"$LOG_DIR/stderr.log" &
|
|
||||||
PID=$!
|
|
||||||
sleep 2
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
#! /bin/bash
|
#!/bin/bash
|
||||||
# Runtime setup for devcontainer (executed after container starts).
|
# Runtime setup for devcontainer (executed after container starts).
|
||||||
# Prefer building setup into resources/devcontainer-Dockerfile when possible.
|
# Prefer building setup into resources/devcontainer-Dockerfile when possible.
|
||||||
# Use this script for runtime-only adjustments (permissions, sockets, ownership,
|
# Use this script for runtime-only adjustments (permissions, sockets, ownership,
|
||||||
@@ -10,30 +10,28 @@ id
|
|||||||
export APP_DIR="/app"
|
export APP_DIR="/app"
|
||||||
export APP_COMMAND="/workspaces/NetAlertX/.devcontainer/scripts/restart-backend.sh"
|
export APP_COMMAND="/workspaces/NetAlertX/.devcontainer/scripts/restart-backend.sh"
|
||||||
export PHP_FPM_BIN="/usr/sbin/php-fpm83"
|
export PHP_FPM_BIN="/usr/sbin/php-fpm83"
|
||||||
export NGINX_BIN="/usr/sbin/nginx"
|
|
||||||
export CROND_BIN="/usr/sbin/crond -f"
|
export CROND_BIN="/usr/sbin/crond -f"
|
||||||
|
|
||||||
|
|
||||||
export ALWAYS_FRESH_INSTALL=false
|
export ALWAYS_FRESH_INSTALL=false
|
||||||
export INSTALL_DIR=/app
|
export INSTALL_DIR=/app
|
||||||
export APP_DATA_LOCATION=/app/config
|
|
||||||
export APP_CONFIG_LOCATION=/app/config
|
|
||||||
export LOGS_LOCATION=/app/logs
|
export LOGS_LOCATION=/app/logs
|
||||||
export CONF_FILE="app.conf"
|
export CONF_FILE="app.conf"
|
||||||
export NGINX_CONF_FILE=netalertx.conf
|
|
||||||
export DB_FILE="app.db"
|
export DB_FILE="app.db"
|
||||||
export FULL_FILEDB_PATH="${INSTALL_DIR}/db/${DB_FILE}"
|
export FULL_FILEDB_PATH="${INSTALL_DIR}/db/${DB_FILE}"
|
||||||
export NGINX_CONFIG_FILE="/etc/nginx/http.d/${NGINX_CONF_FILE}"
|
|
||||||
export OUI_FILE="/usr/share/arp-scan/ieee-oui.txt" # Define the path to ieee-oui.txt and ieee-iab.txt
|
export OUI_FILE="/usr/share/arp-scan/ieee-oui.txt" # Define the path to ieee-oui.txt and ieee-iab.txt
|
||||||
export TZ=Europe/Paris
|
export TZ=Europe/Paris
|
||||||
export PORT=20211
|
export PORT=20211
|
||||||
export SOURCE_DIR="/workspaces/NetAlertX"
|
export SOURCE_DIR="/workspaces/NetAlertX"
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
main() {
|
main() {
|
||||||
echo "=== NetAlertX Development Container Setup ==="
|
echo "=== NetAlertX Development Container Setup ==="
|
||||||
|
killall php-fpm83 nginx crond python3 2>/dev/null
|
||||||
|
sleep 1
|
||||||
echo "Setting up ${SOURCE_DIR}..."
|
echo "Setting up ${SOURCE_DIR}..."
|
||||||
|
sudo chown $(id -u):$(id -g) /workspaces
|
||||||
|
sudo chmod 755 /workspaces
|
||||||
configure_source
|
configure_source
|
||||||
|
|
||||||
echo "--- Starting Development Services ---"
|
echo "--- Starting Development Services ---"
|
||||||
@@ -43,158 +41,97 @@ main() {
|
|||||||
start_services
|
start_services
|
||||||
}
|
}
|
||||||
|
|
||||||
# safe_link: create a symlink from source to target, removing existing target if necessary
|
isRamDisk() {
|
||||||
# bypassing the default behavior of symlinking the directory into the target directory if it is a directory
|
if [ -z "$1" ] || [ ! -d "$1" ]; then
|
||||||
safe_link() {
|
echo "Usage: isRamDisk <directory>" >&2
|
||||||
# usage: safe_link <source> <target>
|
return 2
|
||||||
local src="$1"
|
fi
|
||||||
local dst="$2"
|
|
||||||
|
|
||||||
# Ensure parent directory exists
|
local fstype
|
||||||
install -d -m 775 "$(dirname "$dst")" >/dev/null 2>&1 || true
|
fstype=$(df -T "$1" | awk 'NR==2 {print $2}')
|
||||||
|
|
||||||
# If target exists, remove it without dereferencing symlinks
|
if [ "$fstype" = "tmpfs" ] || [ "$fstype" = "ramfs" ]; then
|
||||||
if [ -L "$dst" ] || [ -e "$dst" ]; then
|
return 0 # Success (is a ramdisk)
|
||||||
rm -rf "$dst"
|
else
|
||||||
fi
|
return 1 # Failure (is not a ramdisk)
|
||||||
|
fi
|
||||||
# Create link; -n prevents deref, -f replaces if somehow still exists
|
|
||||||
ln -sfn "$src" "$dst"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# Setup source directory
|
# Setup source directory
|
||||||
configure_source() {
|
configure_source() {
|
||||||
echo "[1/3] Configuring Source..."
|
echo "[1/4] Configuring System..."
|
||||||
echo " -> Linking source to ${INSTALL_DIR}"
|
echo " -> Setting up /services permissions"
|
||||||
echo "Dev">${INSTALL_DIR}/.VERSION
|
sudo chown -R netalertx /services
|
||||||
|
|
||||||
echo " -> Mounting ramdisks for /log and /api"
|
echo "[2/4] Configuring Source..."
|
||||||
sudo mount -t tmpfs -o size=256M tmpfs "${SOURCE_DIR}/log"
|
echo " -> Cleaning up previous instances"
|
||||||
sudo mount -t tmpfs -o size=512M tmpfs "${SOURCE_DIR}/api"
|
|
||||||
safe_link ${SOURCE_DIR}/api ${INSTALL_DIR}/api
|
|
||||||
safe_link ${SOURCE_DIR}/back ${INSTALL_DIR}/back
|
|
||||||
safe_link "${SOURCE_DIR}/config" "${INSTALL_DIR}/config"
|
|
||||||
safe_link "${SOURCE_DIR}/db" "${INSTALL_DIR}/db"
|
|
||||||
if [ ! -f "${SOURCE_DIR}/config/app.conf" ]; then
|
|
||||||
cp ${SOURCE_DIR}/back/app.conf ${INSTALL_DIR}/config/
|
|
||||||
cp ${SOURCE_DIR}/back/app.db ${INSTALL_DIR}/db/
|
|
||||||
fi
|
|
||||||
|
|
||||||
safe_link "${SOURCE_DIR}/docs" "${INSTALL_DIR}/docs"
|
test -e ${NETALERTX_LOG} && sudo umount "${NETALERTX_LOG}" 2>/dev/null || true
|
||||||
safe_link "${SOURCE_DIR}/front" "${INSTALL_DIR}/front"
|
test -e ${NETALERTX_API} && sudo umount "${NETALERTX_API}" 2>/dev/null || true
|
||||||
safe_link "${SOURCE_DIR}/install" "${INSTALL_DIR}/install"
|
test -e ${NETALERTX_APP} && sudo rm -Rf ${NETALERTX_APP}/
|
||||||
safe_link "${SOURCE_DIR}/scripts" "${INSTALL_DIR}/scripts"
|
|
||||||
safe_link "${SOURCE_DIR}/server" "${INSTALL_DIR}/server"
|
|
||||||
safe_link "${SOURCE_DIR}/test" "${INSTALL_DIR}/test"
|
|
||||||
safe_link "${SOURCE_DIR}/log" "${INSTALL_DIR}/log"
|
|
||||||
safe_link "${SOURCE_DIR}/mkdocs.yml" "${INSTALL_DIR}/mkdocs.yml"
|
|
||||||
|
|
||||||
echo " -> Copying static files to ${INSTALL_DIR}"
|
echo " -> Linking source to ${NETALERTX_APP}"
|
||||||
cp -R ${SOURCE_DIR}/CODE_OF_CONDUCT.md ${INSTALL_DIR}/
|
sudo ln -s ${SOURCE_DIR}/ ${NETALERTX_APP}
|
||||||
cp -R ${SOURCE_DIR}/dockerfiles ${INSTALL_DIR}/dockerfiles
|
|
||||||
sudo cp -na "${INSTALL_DIR}/back/${CONF_FILE}" "${INSTALL_DIR}/config/${CONF_FILE}"
|
|
||||||
sudo cp -na "${INSTALL_DIR}/back/${DB_FILE}" "${FULL_FILEDB_PATH}"
|
|
||||||
if [ -e "${INSTALL_DIR}/api/user_notifications.json" ]; then
|
|
||||||
echo " -> Removing existing user_notifications.json"
|
|
||||||
sudo rm "${INSTALL_DIR}"/api/user_notifications.json
|
|
||||||
fi
|
|
||||||
|
|
||||||
|
echo " -> Mounting ramdisks for /log and /api"
|
||||||
|
mkdir -p ${NETALERTX_LOG} ${NETALERTX_API}
|
||||||
|
sudo mount -o uid=$(id -u netalertx),gid=$(id -g netalertx),mode=775 -t tmpfs -o size=256M tmpfs "${NETALERTX_LOG}"
|
||||||
|
sudo mount -o uid=$(id -u netalertx),gid=$(id -g netalertx),mode=775 -t tmpfs -o size=256M tmpfs "${NETALERTX_API}"
|
||||||
|
mkdir -p ${NETALERTX_PLUGINS_LOG}
|
||||||
|
touch ${NETALERTX_PLUGINS_LOG}/.dockerignore ${NETALERTX_API}/.dockerignore
|
||||||
|
# tmpfs mounts configured with netalertx ownership and 775 permissions above
|
||||||
|
|
||||||
|
touch /app/log/nginx_error.log
|
||||||
echo " -> Setting ownership and permissions"
|
echo " -> Empty log"|tee ${INSTALL_DIR}/log/app.log \
|
||||||
sudo find ${INSTALL_DIR}/ -type d -exec chmod 775 {} \;
|
|
||||||
sudo find ${INSTALL_DIR}/ -type f -exec chmod 664 {} \;
|
|
||||||
sudo date +%s > "${INSTALL_DIR}/front/buildtimestamp.txt"
|
|
||||||
sudo chmod 640 "${INSTALL_DIR}/config/${CONF_FILE}" || true
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
echo " -> Setting up log directory"
|
|
||||||
install -d -o netalertx -g www-data -m 777 ${INSTALL_DIR}/log/plugins
|
|
||||||
|
|
||||||
echo " -> Empty log"|tee ${INSTALL_DIR}/log/app.log \
|
|
||||||
${INSTALL_DIR}/log/app_front.log \
|
${INSTALL_DIR}/log/app_front.log \
|
||||||
${INSTALL_DIR}/log/stdout.log
|
${INSTALL_DIR}/log/stdout.log
|
||||||
touch ${INSTALL_DIR}/log/stderr.log \
|
touch ${INSTALL_DIR}/log/stderr.log \
|
||||||
${INSTALL_DIR}/log/execution_queue.log
|
${INSTALL_DIR}/log/execution_queue.log
|
||||||
echo 0>${INSTALL_DIR}/log/db_is_locked.log
|
echo 0 > ${INSTALL_DIR}/log/db_is_locked.log
|
||||||
|
for f in ${INSTALL_DIR}/log/*.log; do
|
||||||
date +%s > /app/front/buildtimestamp.txt
|
sudo chown netalertx:www-data $f
|
||||||
|
sudo chmod 664 $f
|
||||||
killall python &>/dev/null
|
echo "" > $f
|
||||||
sleep 1
|
|
||||||
}
|
|
||||||
|
|
||||||
#
|
|
||||||
|
|
||||||
# start_services: start crond, PHP-FPM, nginx and the application
|
|
||||||
start_services() {
|
|
||||||
echo "[3/3] Starting services..."
|
|
||||||
|
|
||||||
killall nohup &>/dev/null || true
|
|
||||||
|
|
||||||
killall php-fpm83 &>/dev/null || true
|
|
||||||
killall crond &>/dev/null || true
|
|
||||||
# Give the OS a moment to release the php-fpm socket
|
|
||||||
sleep 0.3
|
|
||||||
echo " -> Starting CronD"
|
|
||||||
setsid nohup $CROND_BIN &>/dev/null &
|
|
||||||
|
|
||||||
echo " -> Starting PHP-FPM"
|
|
||||||
setsid nohup $PHP_FPM_BIN &>/dev/null &
|
|
||||||
|
|
||||||
sudo killall nginx &>/dev/null || true
|
|
||||||
# Wait for the previous nginx processes to exit and for the port to free up
|
|
||||||
tries=0
|
|
||||||
while ss -ltn | grep -q ":${PORT}[[:space:]]" && [ $tries -lt 10 ]; do
|
|
||||||
echo " -> Waiting for port ${PORT} to free..."
|
|
||||||
sleep 0.2
|
|
||||||
tries=$((tries+1))
|
|
||||||
done
|
done
|
||||||
sleep 0.2
|
|
||||||
echo " -> Starting Nginx"
|
mkdir -p /app/log/plugins
|
||||||
setsid nohup $NGINX_BIN &>/dev/null &
|
sudo chown -R netalertx:www-data ${INSTALL_DIR}
|
||||||
echo " -> Starting Backend ${APP_DIR}/server..."
|
|
||||||
$APP_COMMAND
|
|
||||||
sleep 2
|
while ps ax | grep -v grep | grep python3 > /dev/null; do
|
||||||
|
killall python3 &>/dev/null
|
||||||
|
sleep 0.2
|
||||||
|
done
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# configure_php: configure PHP-FPM and enable dev debug options
|
# configure_php: configure PHP-FPM and enable dev debug options
|
||||||
configure_php() {
|
configure_php() {
|
||||||
echo "[2/3] Configuring PHP-FPM..."
|
echo "[3/4] Configuring PHP-FPM..."
|
||||||
sudo killall php-fpm83 &>/dev/null || true
|
sudo chown -R netalertx:netalertx ${SYSTEM_SERVICES_RUN} 2>/dev/null || true
|
||||||
install -d -o nginx -g www-data /run/php/ &>/dev/null
|
|
||||||
sudo sed -i "/^;pid/c\pid = /run/php/php8.3-fpm.pid" /etc/php83/php-fpm.conf
|
|
||||||
sudo sed -i 's|^listen = .*|listen = 127.0.0.1:9000|' /etc/php83/php-fpm.d/www.conf
|
|
||||||
sudo sed -i 's|fastcgi_pass .*|fastcgi_pass 127.0.0.1:9000;|' /etc/nginx/http.d/*.conf
|
|
||||||
|
|
||||||
#increase max child process count to 10
|
|
||||||
sudo sed -i -e 's/pm.max_children = 5/pm.max_children = 10/' /etc/php83/php-fpm.d/www.conf
|
|
||||||
|
|
||||||
# find any line in php-fmp that starts with either ;error_log or error_log = and replace it with error_log = /app/log/app.php_errors.log
|
|
||||||
sudo sed -i '/^;*error_log\s*=/c\error_log = /app/log/app.php_errors.log' /etc/php83/php-fpm.conf
|
|
||||||
# If the line was not found, append it to the end of the file
|
|
||||||
if ! grep -q '^error_log\s*=' /etc/php83/php-fpm.conf; then
|
|
||||||
echo 'error_log = /app/log/app.php_errors.log' | sudo tee -a /etc/php83/php-fpm.conf
|
|
||||||
fi
|
|
||||||
|
|
||||||
sudo mkdir -p /etc/php83/conf.d
|
|
||||||
sudo cp /workspaces/NetAlertX/.devcontainer/resources/99-xdebug.ini /etc/php83/conf.d/99-xdebug.ini
|
|
||||||
|
|
||||||
sudo rm -R /var/log/php83 &>/dev/null || true
|
|
||||||
install -d -o netalertx -g www-data -m 755 var/log/php83;
|
|
||||||
|
|
||||||
sudo chmod 644 /etc/php83/conf.d/99-xdebug.ini || true
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
# (duplicate start_services removed)
|
# start_services: start crond, PHP-FPM, nginx and the application
|
||||||
|
start_services() {
|
||||||
|
echo "[4/4] Starting services"
|
||||||
|
|
||||||
|
sudo chmod +x /entrypoint.sh
|
||||||
|
setsid bash /entrypoint.sh&
|
||||||
|
sleep 1
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
sudo chmod 755 /app/
|
||||||
echo "$(git rev-parse --short=8 HEAD)">/app/.VERSION
|
echo "Development $(git rev-parse --short=8 HEAD)"| sudo tee /app/.VERSION
|
||||||
# Run the main function
|
# Run the main function
|
||||||
main
|
main
|
||||||
|
|
||||||
|
# create a services readme file
|
||||||
|
echo "This folder is auto-generated by the container and devcontainer setup.sh script." > /services/README.md
|
||||||
|
echo "Any changes here will be lost on rebuild. To make permanent changes, edit files in .devcontainer or production filesystem and rebuild the container." >> /services/README.md
|
||||||
|
echo "Only make temporary/test changes in this folder, then perform a rebuild to reset." >> /services/README.md
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,40 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
# Stream NetAlertX logs to stdout so the Dev Containers output channel shows them.
|
|
||||||
# This script waits briefly for the files to appear and then tails them with -F.
|
|
||||||
|
|
||||||
LOG_FILES="/app/log/app.log /app/log/db_is_locked.log /app/log/execution_queue.log /app/log/app_front.log /app/log/app.php_errors.log /app/log/IP_changes.log /app/stderr.log /app/stdout.log"
|
|
||||||
|
|
||||||
wait_for_files() {
|
|
||||||
# Wait up to ~10s for at least one of the files to exist
|
|
||||||
attempts=0
|
|
||||||
while [ $attempts -lt 20 ]; do
|
|
||||||
for f in $LOG_FILES; do
|
|
||||||
if [ -f "$f" ]; then
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
attempts=$((attempts+1))
|
|
||||||
sleep 0.5
|
|
||||||
done
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
if wait_for_files; then
|
|
||||||
echo "Starting log stream for:"
|
|
||||||
for f in $LOG_FILES; do
|
|
||||||
[ -f "$f" ] && echo " $f"
|
|
||||||
done
|
|
||||||
|
|
||||||
# Use tail -F where available. If tail -F isn't supported, tail -f is used as fallback.
|
|
||||||
# Some minimal images may have busybox tail without -F; this handles both.
|
|
||||||
if tail --version >/dev/null 2>&1; then
|
|
||||||
# GNU tail supports -F
|
|
||||||
tail -n +1 -F $LOG_FILES
|
|
||||||
else
|
|
||||||
# Fallback to -f for busybox; will exit if files rotate or do not exist initially
|
|
||||||
tail -n +1 -f $LOG_FILES
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "No log files appeared after wait; exiting stream script."
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
zend_extension=xdebug.so
|
|
||||||
xdebug.mode=debug
|
|
||||||
xdebug.start_with_request=trigger
|
|
||||||
xdebug.trigger_value=VSCODE
|
|
||||||
xdebug.client_host=host.docker.internal
|
|
||||||
xdebug.client_port=9003
|
|
||||||
xdebug.log=/var/log/xdebug.log
|
|
||||||
xdebug.log_level=7
|
|
||||||
xdebug.idekey=VSCODE
|
|
||||||
xdebug.discover_client_host=true
|
|
||||||
xdebug.max_nesting_level=512
|
|
||||||
@@ -2,7 +2,6 @@
|
|||||||
.env
|
.env
|
||||||
.git
|
.git
|
||||||
.github
|
.github
|
||||||
.gitignore
|
|
||||||
docker-compose.yml
|
docker-compose.yml
|
||||||
Dockerfile
|
Dockerfile
|
||||||
Dockerfile.debian
|
Dockerfile.debian
|
||||||
|
|||||||
24
.github/copilot-instructions.md
vendored
24
.github/copilot-instructions.md
vendored
@@ -1,6 +1,7 @@
|
|||||||
This is NetAlertX — network monitoring & alerting.
|
# NetAlertX AI Assistant Instructions
|
||||||
|
This is NetAlertX — network monitoring & alerting. NetAlertX provides Network inventory, awareness, insight, categorization, intruder and presence detection. This is a heavily community-driven project, welcoming of all contributions.
|
||||||
|
|
||||||
Purpose: Guide AI assistants to follow NetAlertX architecture, conventions, and safety practices. Be concise, opinionated, and prefer existing helpers/settings over new code or hardcoded values.
|
You are expected to be concise, opinionated, and biased toward security and simplicity.
|
||||||
|
|
||||||
## Architecture (what runs where)
|
## Architecture (what runs where)
|
||||||
- Backend (Python): main loop + GraphQL/REST endpoints orchestrate scans, plugins, workflows, notifications, and JSON export.
|
- Backend (Python): main loop + GraphQL/REST endpoints orchestrate scans, plugins, workflows, notifications, and JSON export.
|
||||||
@@ -29,9 +30,9 @@ Backend loop phases (see `server/__main__.py` and `server/plugin.py`): `once`, `
|
|||||||
* other: Miscellaneous plugins. Runs at various times. Data source: self / Template.
|
* other: Miscellaneous plugins. Runs at various times. Data source: self / Template.
|
||||||
|
|
||||||
### Plugin logging & outputs
|
### Plugin logging & outputs
|
||||||
- Always log via `mylog()` like other plugins do (no `print()`). Example: `mylog('verbose', [f'[{pluginName}] In script'])`.
|
- Use logging as shown in other plugins.
|
||||||
- Collect results with `Plugin_Objects.add_object(...)` during processing and call `plugin_objects.write_result_file()` exactly once at the end of the script.
|
- Collect results with `Plugin_Objects.add_object(...)` during processing and call `plugin_objects.write_result_file()` exactly once at the end of the script.
|
||||||
- Prefer to log a brief summary before writing (e.g., total objects added) to aid troubleshooting; keep logs concise at `verbose` level unless debugging.
|
- Prefer to log a brief summary before writing (e.g., total objects added) to aid troubleshooting; keep logs concise at `info` level and use `verbose` or `debug` for extra context.
|
||||||
|
|
||||||
- Do not write ad‑hoc files for results; the only consumable output is `last_result.<PREF>.log` generated by `Plugin_Objects`.
|
- Do not write ad‑hoc files for results; the only consumable output is `last_result.<PREF>.log` generated by `Plugin_Objects`.
|
||||||
## API/Endpoints quick map
|
## API/Endpoints quick map
|
||||||
@@ -58,5 +59,16 @@ Backend loop phases (see `server/__main__.py` and `server/plugin.py`): `once`, `
|
|||||||
- Docs: `docs/PLUGINS_DEV.md`, `docs/SETTINGS_SYSTEM.md`, `docs/API_*.md`, `docs/DEBUG_*.md`
|
- Docs: `docs/PLUGINS_DEV.md`, `docs/SETTINGS_SYSTEM.md`, `docs/API_*.md`, `docs/DEBUG_*.md`
|
||||||
- Logs: backend `/app/log/app.log`, plugin logs under `/app/log/plugins/`, nginx/php logs under `/var/log/*`
|
- Logs: backend `/app/log/app.log`, plugin logs under `/app/log/plugins/`, nginx/php logs under `/var/log/*`
|
||||||
|
|
||||||
Assistant expectations
|
## Assistant expectations:
|
||||||
- Reference concrete files/paths. Use existing helpers/settings. Keep changes idempotent and safe. Offer a quick validation step (log line, API hit, or JSON export) for anything you add.
|
- Be concise, opinionated, and biased toward security and simplicity.
|
||||||
|
- Reference concrete files/paths/environmental variables.
|
||||||
|
- Use existing helpers/settings.
|
||||||
|
- Offer a quick validation step (log line, API hit, or JSON export) for anything you add.
|
||||||
|
- Be blunt about risks and when you offer suggestions ensure they're also blunt,
|
||||||
|
- Ask for confirmation before making changes that run code or change multiple files.
|
||||||
|
- Make statements actionable and specific; propose exact edits.
|
||||||
|
- Request confirmation before applying changes that affect more than a single, clearly scoped line or file.
|
||||||
|
- Ask the user to debug something for an actionable value if you're unsure.
|
||||||
|
- Be sure to offer choices when appropriate.
|
||||||
|
- Always understand the intent of the user's request and undo/redo as needed.
|
||||||
|
- Above all, use the simplest possible code that meets the need so it can be easily audited and maintained.
|
||||||
|
|||||||
10
.gitignore
vendored
10
.gitignore
vendored
@@ -1,6 +1,15 @@
|
|||||||
.vscode
|
.vscode
|
||||||
|
.dotnet
|
||||||
|
.vscode-server
|
||||||
|
.gitconfig
|
||||||
|
.*CommandMarker
|
||||||
|
deviceid
|
||||||
.DS_Store
|
.DS_Store
|
||||||
|
.cache
|
||||||
|
nohup.out
|
||||||
config/*
|
config/*
|
||||||
|
.ash_history
|
||||||
|
.VERSION
|
||||||
config/pialert.conf
|
config/pialert.conf
|
||||||
config/app.conf
|
config/app.conf
|
||||||
db/*
|
db/*
|
||||||
@@ -8,6 +17,7 @@ db/pialert.db
|
|||||||
db/app.db
|
db/app.db
|
||||||
front/log/*
|
front/log/*
|
||||||
/log/*
|
/log/*
|
||||||
|
/log/plugins/*
|
||||||
front/api/*
|
front/api/*
|
||||||
/api/*
|
/api/*
|
||||||
**/plugins/**/*.log
|
**/plugins/**/*.log
|
||||||
|
|||||||
23
.venv_import_check.py
Normal file
23
.venv_import_check.py
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import sys, importlib
|
||||||
|
mods = [
|
||||||
|
'json', 'simplejson',
|
||||||
|
'httplib', 'http.client',
|
||||||
|
'urllib2', 'urllib.request',
|
||||||
|
'Queue', 'queue',
|
||||||
|
'cStringIO', 'StringIO', 'io',
|
||||||
|
'md5', 'hashlib',
|
||||||
|
'ssl'
|
||||||
|
]
|
||||||
|
print('PYTHON_EXE:' + sys.executable)
|
||||||
|
print('PYTHON_VER:' + sys.version.replace('\n', ' '))
|
||||||
|
for m in mods:
|
||||||
|
try:
|
||||||
|
mod = importlib.import_module(m)
|
||||||
|
ver = getattr(mod, '__version__', None)
|
||||||
|
if ver is None:
|
||||||
|
# try common attributes
|
||||||
|
ver = getattr(mod, 'version', None)
|
||||||
|
info = (' version=' + str(ver)) if ver is not None else ''
|
||||||
|
print('OK %s%s' % (m, info))
|
||||||
|
except Exception as e:
|
||||||
|
print('MISSING %s %s: %s' % (m, e.__class__.__name__, e))
|
||||||
10
.vscode/settings.json
vendored
10
.vscode/settings.json
vendored
@@ -10,4 +10,14 @@
|
|||||||
"python.defaultInterpreterPath": "/opt/venv/bin/python",
|
"python.defaultInterpreterPath": "/opt/venv/bin/python",
|
||||||
// Let the Python extension invoke pytest via the interpreter; avoid hardcoded paths
|
// Let the Python extension invoke pytest via the interpreter; avoid hardcoded paths
|
||||||
// Removed python.testing.pytestPath and legacy pytest.command overrides
|
// Removed python.testing.pytestPath and legacy pytest.command overrides
|
||||||
|
|
||||||
|
"terminal.integrated.defaultProfile.linux": "fish",
|
||||||
|
"terminal.integrated.profiles.linux": {
|
||||||
|
"fish": {
|
||||||
|
"path": "/usr/bin/fish"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
,
|
||||||
|
// Fallback for older VS Code versions or schema validators that don't accept custom profiles
|
||||||
|
"terminal.integrated.shell.linux": "/usr/bin/fish"
|
||||||
}
|
}
|
||||||
98
.vscode/tasks.json
vendored
98
.vscode/tasks.json
vendored
@@ -2,9 +2,9 @@
|
|||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
"tasks": [
|
"tasks": [
|
||||||
{
|
{
|
||||||
"label": "Generate Dockerfile",
|
"label": "[Any POSIX] Generate Devcontainer Configs",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "${workspaceFolder:NetAlertX}/.devcontainer/scripts/generate-dockerfile.sh",
|
"command": ".devcontainer/scripts/generate-configs.sh",
|
||||||
"presentation": {
|
"presentation": {
|
||||||
"echo": true,
|
"echo": true,
|
||||||
"reveal": "always",
|
"reveal": "always",
|
||||||
@@ -16,24 +16,45 @@
|
|||||||
"kind": "build",
|
"kind": "build",
|
||||||
"isDefault": false
|
"isDefault": false
|
||||||
},
|
},
|
||||||
"options": {
|
|
||||||
"cwd": "${workspaceFolder:NetAlertX}"
|
|
||||||
},
|
|
||||||
"icon": {
|
"icon": {
|
||||||
"id": "tools",
|
"id": "tools",
|
||||||
"color": "terminal.ansiYellow"
|
"color": "terminal.ansiYellow"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "Re-Run Startup Script",
|
"label": "[Any] Docker system and build Prune",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "${workspaceFolder:NetAlertX}/.devcontainer/scripts/setup.sh",
|
"command": ".devcontainer/scripts/confirm-docker-prune.sh",
|
||||||
"presentation": {
|
"presentation": {
|
||||||
"echo": true,
|
"echo": true,
|
||||||
"reveal": "always",
|
"reveal": "always",
|
||||||
"panel": "shared",
|
"panel": "shared",
|
||||||
"showReuseMessage": false
|
"showReuseMessage": false
|
||||||
},
|
},
|
||||||
|
"problemMatcher": [],
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": false
|
||||||
|
},
|
||||||
|
"icon": {
|
||||||
|
"id": "trash",
|
||||||
|
"color": "terminal.ansiRed"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "[Dev Container] Re-Run Startup Script",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "./isDevContainer.sh || exit 1;/workspaces/NetAlertX/.devcontainer/scripts/setup.sh",
|
||||||
|
"options": {
|
||||||
|
"cwd": "/workspaces/NetAlertX/.devcontainer/scripts"
|
||||||
|
},
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false
|
||||||
|
},
|
||||||
|
|
||||||
"problemMatcher": [],
|
"problemMatcher": [],
|
||||||
"icon": {
|
"icon": {
|
||||||
"id": "beaker",
|
"id": "beaker",
|
||||||
@@ -41,9 +62,12 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "Start Backend (Python)",
|
"label": "[Dev Container] Start Backend (Python)",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "/workspaces/NetAlertX/.devcontainer/scripts/restart-backend.sh",
|
"command": "./isDevContainer.sh || exit 1; /services/start-backend.sh",
|
||||||
|
"options": {
|
||||||
|
"cwd": "/workspaces/NetAlertX/.devcontainer/scripts"
|
||||||
|
},
|
||||||
"presentation": {
|
"presentation": {
|
||||||
"echo": true,
|
"echo": true,
|
||||||
"reveal": "always",
|
"reveal": "always",
|
||||||
@@ -58,9 +82,12 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "Start Frontend (nginx and PHP-FPM)",
|
"label": "[Dev Container] Start CronD (Scheduler)",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "killall php-fpm83 nginx 2>/dev/null || true; sleep 1; php-fpm83 & nginx",
|
"command": "./isDevContainer.sh || exit 1; /services/start-crond.sh",
|
||||||
|
"options": {
|
||||||
|
"cwd": "/workspaces/NetAlertX/.devcontainer/scripts"
|
||||||
|
},
|
||||||
"presentation": {
|
"presentation": {
|
||||||
"echo": true,
|
"echo": true,
|
||||||
"reveal": "always",
|
"reveal": "always",
|
||||||
@@ -75,9 +102,33 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"label": "Stop Frontend & Backend Services",
|
"label": "[Dev Container] Start Frontend (nginx and PHP-FPM)",
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
"command": "pkill -f 'php-fpm83|nginx|crond|python3' || true",
|
"command": "./isDevContainer.sh || exit 1; /services/start-php-fpm.sh & /services/start-nginx.sh &",
|
||||||
|
"options": {
|
||||||
|
"cwd": "/workspaces/NetAlertX/.devcontainer/scripts"
|
||||||
|
|
||||||
|
},
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": false
|
||||||
|
},
|
||||||
|
"problemMatcher": [],
|
||||||
|
"icon": {
|
||||||
|
"id": "debug-restart",
|
||||||
|
"color": "terminal.ansiGreen"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "[Dev Container] Stop Frontend & Backend Services",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "./isDevContainer.sh || exit 1; pkill -f 'php-fpm83|nginx|crond|python3' || true",
|
||||||
|
"options": {
|
||||||
|
"cwd": "/workspaces/NetAlertX/.devcontainer/scripts"
|
||||||
|
},
|
||||||
"presentation": {
|
"presentation": {
|
||||||
"echo": true,
|
"echo": true,
|
||||||
"reveal": "always",
|
"reveal": "always",
|
||||||
@@ -89,6 +140,25 @@
|
|||||||
"id": "debug-stop",
|
"id": "debug-stop",
|
||||||
"color": "terminal.ansiRed"
|
"color": "terminal.ansiRed"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "[Dev Container] List NetAlertX Ports",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "list-ports.sh",
|
||||||
|
"options": {
|
||||||
|
"cwd": "/workspaces/NetAlertX/.devcontainer/scripts"
|
||||||
|
},
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false
|
||||||
|
},
|
||||||
|
"problemMatcher": [],
|
||||||
|
"icon": {
|
||||||
|
"id": "output",
|
||||||
|
"color": "terminal.ansiBlue"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
210
Dockerfile
210
Dockerfile
@@ -1,63 +1,197 @@
|
|||||||
|
# The NetAlertX Dockerfile has 3 stages:
|
||||||
|
#
|
||||||
|
# Stage 1. Builder - NetAlertX Requires special tools and packages to build our virtual environment, but
|
||||||
|
# which are not needed in future stages. We build the builder and extract the venv for runner to use as
|
||||||
|
# a base.
|
||||||
|
#
|
||||||
|
# Stage 2. Runner builds the bare minimum requirements to create an operational NetAlertX. The primary
|
||||||
|
# reason for breaking at this stage is it leaves the system in a proper state for devcontainer operation
|
||||||
|
# This image also provides a break-out point for uses who wish to execute the anti-pattern of using a
|
||||||
|
# docker container as a VM for experimentation and various development patterns.
|
||||||
|
#
|
||||||
|
# Stage 3. Hardened removes root, sudoers, folders, permissions, and locks the system down into a read-only
|
||||||
|
# compatible image. While NetAlertX does require some read-write operations, this image can guarantee the
|
||||||
|
# code pushed out by the project is the only code which will run on the system after each container restart.
|
||||||
|
# It reduces the chance of system hijacking and operates with all modern security protocols in place as is
|
||||||
|
# expected from a security appliance.
|
||||||
|
#
|
||||||
|
# This file can be built with `docker compose -f docker-compose.yml up --build --force-recreate`
|
||||||
|
|
||||||
FROM alpine:3.22 AS builder
|
FROM alpine:3.22 AS builder
|
||||||
|
|
||||||
ARG INSTALL_DIR=/app
|
ARG INSTALL_DIR=/app
|
||||||
|
|
||||||
ENV PYTHONUNBUFFERED=1
|
ENV PYTHONUNBUFFERED=1
|
||||||
|
ENV PATH="/opt/venv/bin:$PATH"
|
||||||
|
|
||||||
# Install build dependencies
|
# Install build dependencies
|
||||||
|
COPY requirements.txt /tmp/requirements.txt
|
||||||
RUN apk add --no-cache bash shadow python3 python3-dev gcc musl-dev libffi-dev openssl-dev git \
|
RUN apk add --no-cache bash shadow python3 python3-dev gcc musl-dev libffi-dev openssl-dev git \
|
||||||
&& python -m venv /opt/venv
|
&& python -m venv /opt/venv
|
||||||
|
|
||||||
# Enable venv
|
# Create virtual environment owned by root, but readable by everyone else. This makes it easy to copy
|
||||||
ENV PATH="/opt/venv/bin:$PATH"
|
# into hardened stage without worrying about permissions and keeps image size small. Keeping the commands
|
||||||
|
# together makes for a slightly smaller image size.
|
||||||
|
RUN pip install -r /tmp/requirements.txt && \
|
||||||
|
chmod -R u-rwx,g-rwx /opt
|
||||||
|
|
||||||
COPY . ${INSTALL_DIR}/
|
# second stage is the main runtime stage with just the minimum required to run the application
|
||||||
|
# The runner is used for both devcontainer, and as a base for the hardened stage.
|
||||||
RUN pip install openwrt-luci-rpc asusrouter asyncio aiohttp graphene flask flask-cors unifi-sm-api tplink-omada-client wakeonlan pycryptodome requests paho-mqtt scapy cron-converter pytz json2table dhcp-leases pyunifi speedtest-cli chardet python-nmap dnspython librouteros yattag zeroconf git+https://github.com/foreign-sub/aiofreepybox.git \
|
|
||||||
&& bash -c "find ${INSTALL_DIR} -type d -exec chmod 750 {} \;" \
|
|
||||||
&& bash -c "find ${INSTALL_DIR} -type f -exec chmod 640 {} \;" \
|
|
||||||
&& bash -c "find ${INSTALL_DIR} -type f \( -name '*.sh' -o -name '*.py' -o -name 'speedtest-cli' \) -exec chmod 750 {} \;"
|
|
||||||
|
|
||||||
# Append Iliadbox certificate to aiofreepybox
|
|
||||||
RUN cat ${INSTALL_DIR}/install/freebox_certificate.pem >> /opt/venv/lib/python3.12/site-packages/aiofreepybox/freebox_certificates.pem
|
|
||||||
|
|
||||||
# second stage
|
|
||||||
FROM alpine:3.22 AS runner
|
FROM alpine:3.22 AS runner
|
||||||
|
|
||||||
ARG INSTALL_DIR=/app
|
ARG INSTALL_DIR=/app
|
||||||
|
|
||||||
COPY --from=builder /opt/venv /opt/venv
|
# NetAlertX app directories
|
||||||
COPY --from=builder /usr/sbin/usermod /usr/sbin/groupmod /usr/sbin/
|
ENV NETALERTX_APP=${INSTALL_DIR}
|
||||||
|
ENV NETALERTX_CONFIG=${NETALERTX_APP}/config
|
||||||
|
ENV NETALERTX_FRONT=${NETALERTX_APP}/front
|
||||||
|
ENV NETALERTX_SERVER=${NETALERTX_APP}/server
|
||||||
|
ENV NETALERTX_API=${NETALERTX_APP}/api
|
||||||
|
ENV NETALERTX_DB=${NETALERTX_APP}/db
|
||||||
|
ENV NETALERTX_DB_FILE=${NETALERTX_DB}/app.db
|
||||||
|
ENV NETALERTX_BACK=${NETALERTX_APP}/back
|
||||||
|
ENV NETALERTX_LOG=${NETALERTX_APP}/log
|
||||||
|
ENV NETALERTX_PLUGINS_LOG=${NETALERTX_LOG}/plugins
|
||||||
|
ENV NETALERTX_CONFIG_FILE=${NETALERTX_CONFIG}/app.conf
|
||||||
|
|
||||||
# Enable venv
|
# NetAlertX log files
|
||||||
ENV PATH="/opt/venv/bin:$PATH"
|
ENV LOG_IP_CHANGES=${NETALERTX_LOG}/IP_changes.log
|
||||||
|
ENV LOG_APP=${NETALERTX_LOG}/app.log
|
||||||
|
ENV LOG_APP_FRONT=${NETALERTX_LOG}/app_front.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_TXT=${NETALERTX_LOG}/report_output.txt
|
||||||
|
ENV LOG_DB_IS_LOCKED=${NETALERTX_LOG}/db_is_locked.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_HTML=${NETALERTX_LOG}/report_output.html
|
||||||
|
ENV LOG_STDERR=${NETALERTX_LOG}/stderr.log
|
||||||
|
ENV LOG_APP_PHP_ERRORS=${NETALERTX_LOG}/app.php_errors.log
|
||||||
|
ENV LOG_EXECUTION_QUEUE=${NETALERTX_LOG}/execution_queue.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_JSON=${NETALERTX_LOG}/report_output.json
|
||||||
|
ENV LOG_STDOUT=${NETALERTX_LOG}/stdout.log
|
||||||
|
ENV LOG_CROND=${NETALERTX_LOG}/crond.log
|
||||||
|
|
||||||
# default port and listen address
|
# System Services configuration files
|
||||||
ENV PORT=20211 LISTEN_ADDR=0.0.0.0
|
ENV SYSTEM_SERVICES=/services
|
||||||
|
ENV SYSTEM_SERVICES_SCRIPTS=${SYSTEM_SERVICES}/scripts
|
||||||
|
ENV SYSTEM_SERVICES_CONFIG=${SYSTEM_SERVICES}/config
|
||||||
|
ENV SYSTEM_NGINX_CONFIG=${SYSTEM_SERVICES_CONFIG}/nginx
|
||||||
|
ENV SYSTEM_NGINX_CONFIG_FILE=${SYSTEM_NGINX_CONFIG}/nginx.conf
|
||||||
|
ENV SYSTEM_SERVICES_PHP_FOLDER=${SYSTEM_SERVICES_CONFIG}/php
|
||||||
|
ENV SYSTEM_SERVICES_PHP_FPM_D=${SYSTEM_SERVICES_PHP_FOLDER}/php-fpm.d
|
||||||
|
ENV SYSTEM_SERVICES_CROND=${SYSTEM_SERVICES_CONFIG}/crond
|
||||||
|
ENV SYSTEM_SERVICES_RUN=${SYSTEM_SERVICES}/run
|
||||||
|
ENV SYSTEM_SERVICES_RUN_TMP=${SYSTEM_SERVICES_RUN}/tmp
|
||||||
|
ENV SYSTEM_SERVICES_RUN_LOG=${SYSTEM_SERVICES_RUN}/logs
|
||||||
|
ENV PHP_FPM_CONFIG_FILE=${SYSTEM_SERVICES_PHP_FOLDER}/php-fpm.conf
|
||||||
|
ENV READ_ONLY_FOLDERS="${NETALERTX_BACK} ${NETALERTX_FRONT} ${NETALERTX_SERVER} ${SYSTEM_SERVICES} \
|
||||||
|
${SYSTEM_SERVICES_CONFIG}"
|
||||||
|
ENV READ_WRITE_FOLDERS="${NETALERTX_CONFIG} ${NETALERTX_DB} ${NETALERTX_API} ${NETALERTX_LOG} \
|
||||||
|
${NETALERTX_PLUGINS_LOG} ${SYSTEM_SERVICES_RUN} ${SYSTEM_SERVICES_RUN_TMP} \
|
||||||
|
${SYSTEM_SERVICES_RUN_LOG}"
|
||||||
|
|
||||||
# needed for s6-overlay
|
#Python environment
|
||||||
ENV S6_CMD_WAIT_FOR_SERVICES_MAXTIME=0
|
ENV PYTHONUNBUFFERED=1
|
||||||
|
ENV VIRTUAL_ENV=/opt/venv
|
||||||
|
ENV VIRTUAL_ENV_BIN=/opt/venv/bin
|
||||||
|
ENV PYTHONPATH=${NETALERTX_APP}:${NETALERTX_SERVER}:${VIRTUAL_ENV}/lib/python3.12/site-packages
|
||||||
|
ENV PATH="${SYSTEM_SERVICES}:${VIRTUAL_ENV_BIN}:$PATH"
|
||||||
|
|
||||||
# ❗ IMPORTANT - if you modify this file modify the /install/install_dependecies.sh file as well ❗
|
# App Environment
|
||||||
|
ENV LISTEN_ADDR=0.0.0.0
|
||||||
|
ENV PORT=20211
|
||||||
|
ENV NETALERTX_DEBUG=0
|
||||||
|
ENV VENDORSPATH=/app/back/ieee-oui.txt
|
||||||
|
ENV VENDORSPATH_NEWEST=/services/run/tmp/ieee-oui.txt
|
||||||
|
ENV ENVIRONMENT=alpine
|
||||||
|
ENV READ_ONLY_USER=readonly READ_ONLY_GROUP=readonly
|
||||||
|
ENV NETALERTX_USER=netalertx NETALERTX_GROUP=netalertx
|
||||||
|
ENV LANG=C.UTF-8
|
||||||
|
|
||||||
RUN apk update --no-cache \
|
|
||||||
&& apk add --no-cache bash libbsd zip lsblk gettext-envsubst sudo mtr tzdata s6-overlay \
|
|
||||||
&& apk add --no-cache curl arp-scan iproute2 iproute2-ss nmap nmap-scripts traceroute nbtscan net-tools net-snmp-tools bind-tools awake ca-certificates \
|
|
||||||
&& apk add --no-cache sqlite php83 php83-fpm php83-cgi php83-curl php83-sqlite3 php83-session \
|
|
||||||
&& apk add --no-cache python3 nginx \
|
|
||||||
&& ln -s /usr/bin/awake /usr/bin/wakeonlan \
|
|
||||||
&& bash -c "install -d -m 750 -o nginx -g www-data ${INSTALL_DIR} ${INSTALL_DIR}" \
|
|
||||||
&& rm -f /etc/nginx/http.d/default.conf
|
|
||||||
|
|
||||||
COPY --from=builder --chown=nginx:www-data ${INSTALL_DIR}/ ${INSTALL_DIR}/
|
RUN apk add --no-cache bash mtr libbsd zip lsblk tzdata curl arp-scan iproute2 iproute2-ss nmap \
|
||||||
|
nmap-scripts traceroute nbtscan net-tools net-snmp-tools bind-tools awake ca-certificates \
|
||||||
|
sqlite php83 php83-fpm php83-cgi php83-curl php83-sqlite3 php83-session python3 envsubst \
|
||||||
|
nginx shadow && \
|
||||||
|
rm -Rf /var/cache/apk/* && \
|
||||||
|
rm -Rf /etc/nginx && \
|
||||||
|
addgroup -g 20211 ${NETALERTX_GROUP} && \
|
||||||
|
adduser -u 20211 -D -h ${NETALERTX_APP} -G ${NETALERTX_GROUP} ${NETALERTX_USER} && \
|
||||||
|
apk del shadow
|
||||||
|
|
||||||
# Add crontab file
|
|
||||||
COPY --chmod=600 --chown=root:root install/crontab /etc/crontabs/root
|
|
||||||
|
|
||||||
# Start all required services
|
|
||||||
RUN ${INSTALL_DIR}/dockerfiles/start.sh
|
|
||||||
|
|
||||||
HEALTHCHECK --interval=30s --timeout=5s --start-period=15s --retries=2 \
|
# Install application, copy files, set permissions
|
||||||
CMD curl -sf -o /dev/null ${LISTEN_ADDR}:${PORT}/php/server/query_json.php?file=app_state.json
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} install/production-filesystem/ /
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} --chmod=755 back ${NETALERTX_BACK}
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} --chmod=755 front ${NETALERTX_FRONT}
|
||||||
|
COPY --chown=${NETALERTX_USER}:${NETALERTX_GROUP} --chmod=755 server ${NETALERTX_SERVER}
|
||||||
|
RUN install -d -o ${NETALERTX_USER} -g ${NETALERTX_GROUP} -m 755 ${NETALERTX_API} \
|
||||||
|
${NETALERTX_LOG} ${SYSTEM_SERVICES_RUN_TMP} ${SYSTEM_SERVICES_RUN_LOG} && \
|
||||||
|
sh -c "find ${NETALERTX_APP} -type f \( -name '*.sh' -o -name 'speedtest-cli' \) \
|
||||||
|
-exec chmod 750 {} \;"
|
||||||
|
|
||||||
|
# Copy the virtualenv from the builder stage
|
||||||
|
COPY --from=builder --chown=20212:20212 ${VIRTUAL_ENV} ${VIRTUAL_ENV}
|
||||||
|
|
||||||
|
|
||||||
|
# Initialize each service with the dockerfiles/init-*.sh scripts, once.
|
||||||
|
# This is done after the copy of the venv to ensure the venv is in place
|
||||||
|
# although it may be quicker to do it before the copy, it keeps the image
|
||||||
|
# layers smaller to do it after.
|
||||||
|
RUN apk add libcap && \
|
||||||
|
setcap cap_net_raw+ep /bin/busybox && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip /usr/bin/nmap && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip /usr/bin/arp-scan && \
|
||||||
|
setcap cap_net_raw,cap_net_admin,cap_net_bind_service+eip /usr/bin/nbtscan && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip /usr/bin/traceroute && \
|
||||||
|
setcap cap_net_raw,cap_net_admin+eip $(readlink -f ${VIRTUAL_ENV_BIN}/python) && \
|
||||||
|
/bin/sh /build/init-nginx.sh && \
|
||||||
|
/bin/sh /build/init-php-fpm.sh && \
|
||||||
|
/bin/sh /build/init-crond.sh && \
|
||||||
|
/bin/sh /build/init-backend.sh && \
|
||||||
|
rm -rf /build && \
|
||||||
|
apk del libcap && \
|
||||||
|
date +%s > ${NETALERTX_FRONT}/buildtimestamp.txt
|
||||||
|
|
||||||
|
|
||||||
|
ENTRYPOINT ["/bin/sh","/entrypoint.sh"]
|
||||||
|
|
||||||
|
# Final hardened stage to improve security by setting least possible permissions and removing sudo access.
|
||||||
|
# When complete, if the image is compromised, there's not much that can be done with it.
|
||||||
|
# This stage is separate from Runner stage so that devcontainer can use the Runner stage.
|
||||||
|
FROM runner AS hardened
|
||||||
|
|
||||||
|
ENV UMASK=0077
|
||||||
|
|
||||||
|
# Create readonly user and group with no shell access.
|
||||||
|
# Readonly user marks folders that are created by NetAlertX, but should not be modified.
|
||||||
|
# AI may claim this is stupid, but it's actually least possible permissions as
|
||||||
|
# read-only user cannot login, cannot sudo, has no write permission, and cannot even
|
||||||
|
# read the files it owns. The read-only user is ownership-as-a-lock hardening pattern.
|
||||||
|
RUN addgroup -g 20212 ${READ_ONLY_GROUP} && \
|
||||||
|
adduser -u 20212 -G ${READ_ONLY_GROUP} -D -h /app ${READ_ONLY_USER}
|
||||||
|
|
||||||
|
|
||||||
|
# reduce permissions to minimum necessary for all NetAlertX files and folders
|
||||||
|
# Permissions 005 and 004 are not typos, they enable read-only. Everyone can
|
||||||
|
# read the read-only files, and nobody can write to them, even the readonly user.
|
||||||
|
RUN chown -R ${READ_ONLY_USER}:${READ_ONLY_GROUP} ${READ_ONLY_FOLDERS} && \
|
||||||
|
chmod -R 004 ${READ_ONLY_FOLDERS} && \
|
||||||
|
find ${READ_ONLY_FOLDERS} -type d -exec chmod 005 {} + && \
|
||||||
|
install -d -o ${NETALERTX_USER} -g ${NETALERTX_GROUP} -m 700 ${READ_WRITE_FOLDERS} && \
|
||||||
|
chown -R ${NETALERTX_USER}:${NETALERTX_GROUP} ${READ_WRITE_FOLDERS} && \
|
||||||
|
chmod -R 600 ${READ_WRITE_FOLDERS} && \
|
||||||
|
find ${READ_WRITE_FOLDERS} -type d -exec chmod 700 {} + && \
|
||||||
|
chown ${READ_ONLY_USER}:${READ_ONLY_GROUP} /entrypoint.sh /opt /opt/venv && \
|
||||||
|
chmod 005 /entrypoint.sh ${SYSTEM_SERVICES}/*.sh /app /opt /opt/venv && \
|
||||||
|
apk del apk-tools && \
|
||||||
|
rm -Rf /var /etc/sudoers.d/* /etc/shadow /etc/gshadow /etc/sudoers \
|
||||||
|
/lib/apk /lib/firmware /lib/modules-load.d /lib/sysctl.d /mnt /home/ /root \
|
||||||
|
/srv /media && \
|
||||||
|
sed -i "/^\(${READ_ONLY_USER}\|${NETALERTX_USER}\):/!d" /etc/passwd && \
|
||||||
|
sed -i "/^\(${READ_ONLY_GROUP}\|${NETALERTX_GROUP}\):/!d" /etc/group && \
|
||||||
|
echo -ne '#!/bin/sh\n"$@"\n' > /usr/bin/sudo && chmod +x /usr/bin/sudo
|
||||||
|
|
||||||
|
USER netalertx
|
||||||
|
|
||||||
|
HEALTHCHECK --interval=30s --timeout=10s --start-period=60s --retries=3 \
|
||||||
|
CMD /services/healthcheck.sh
|
||||||
|
|
||||||
ENTRYPOINT ["/init"]
|
|
||||||
|
|||||||
@@ -1,53 +1,169 @@
|
|||||||
|
# Warning - use of this unhardened image is not recommended for production use.
|
||||||
|
# This image is provided for backward compatibility, development and testing purposes only.
|
||||||
|
# For production use, please use the hardened image built with Alpine. This image attempts to
|
||||||
|
# treat a container as an operating system, which is an anti-pattern and a common source of
|
||||||
|
# security issues.
|
||||||
|
#
|
||||||
|
# The default Dockerfile/docker-compose image contains the following security improvements
|
||||||
|
# over the Debian image:
|
||||||
|
# - read-only filesystem
|
||||||
|
# - no sudo access
|
||||||
|
# - least possible permissions on all files and folders
|
||||||
|
# - Root user has all permissions revoked and is unused
|
||||||
|
# - Secure umask applied so files are owner-only by default
|
||||||
|
# - non-privileged user runs the application
|
||||||
|
# - no shell access for non-privileged users
|
||||||
|
# - no unnecessary packages or services
|
||||||
|
# - reduced capabilities
|
||||||
|
# - tmpfs for writable folders
|
||||||
|
# - healthcheck
|
||||||
|
# - no package managers
|
||||||
|
# - no compilers or build tools
|
||||||
|
# - no systemd, uses lightweight init system
|
||||||
|
# - no persistent storage except for config and db volumes
|
||||||
|
# - minimal image size due to segmented build stages
|
||||||
|
# - minimal base image (Alpine Linux)
|
||||||
|
# - minimal python environment (venv, no pip)
|
||||||
|
# - minimal stripped web server
|
||||||
|
# - minimal stripped php environment
|
||||||
|
# - minimal services (nginx, php-fpm, crond, no unnecessary services or service managers)
|
||||||
|
# - minimal users and groups (netalertx and readonly only, no others)
|
||||||
|
# - minimal permissions (read-only for most files and folders, write-only for necessary folders)
|
||||||
|
# - minimal capabilities (NET_ADMIN and NET_RAW only, no others)
|
||||||
|
# - minimal environment variables (only necessary ones, no others)
|
||||||
|
# - minimal entrypoint (only necessary commands, no others)
|
||||||
|
# - Uses the same base image as the development environmnment (Alpine Linux)
|
||||||
|
# - Uses the same services as the development environment (nginx, php-fpm, crond)
|
||||||
|
# - Uses the same environment variables as the development environment (only necessary ones, no others)
|
||||||
|
# - Uses the same file and folder structure as the development environment (only necessary ones, no others)
|
||||||
|
# NetAlertX is designed to be run as an unattended network security monitoring appliance, which means it
|
||||||
|
# should be able to operate without human intervention. Overall, the hardened image is designed to be as
|
||||||
|
# secure as possible while still being functional and is recommended because you cannot attack a surface
|
||||||
|
# that isn't there.
|
||||||
|
|
||||||
|
|
||||||
FROM debian:bookworm-slim
|
FROM debian:bookworm-slim
|
||||||
|
|
||||||
# default UID and GID
|
|
||||||
ENV USER=pi USER_ID=1000 USER_GID=1000 PORT=20211
|
|
||||||
#TZ=Europe/London
|
#TZ=Europe/London
|
||||||
|
|
||||||
|
# NetAlertX app directories
|
||||||
|
ENV INSTALL_DIR=/app
|
||||||
|
ENV NETALERTX_APP=${INSTALL_DIR}
|
||||||
|
ENV NETALERTX_CONFIG=${NETALERTX_APP}/config
|
||||||
|
ENV NETALERTX_FRONT=${NETALERTX_APP}/front
|
||||||
|
ENV NETALERTX_SERVER=${NETALERTX_APP}/server
|
||||||
|
ENV NETALERTX_API=${NETALERTX_APP}/api
|
||||||
|
ENV NETALERTX_DB=${NETALERTX_APP}/db
|
||||||
|
ENV NETALERTX_DB_FILE=${NETALERTX_DB}/app.db
|
||||||
|
ENV NETALERTX_BACK=${NETALERTX_APP}/back
|
||||||
|
ENV NETALERTX_LOG=${NETALERTX_APP}/log
|
||||||
|
ENV NETALERTX_PLUGINS_LOG=${NETALERTX_LOG}/plugins
|
||||||
|
|
||||||
|
# NetAlertX log files
|
||||||
|
ENV LOG_IP_CHANGES=${NETALERTX_LOG}/IP_changes.log
|
||||||
|
ENV LOG_APP=${NETALERTX_LOG}/app.log
|
||||||
|
ENV LOG_APP_FRONT=${NETALERTX_LOG}/app_front.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_TXT=${NETALERTX_LOG}/report_output.txt
|
||||||
|
ENV LOG_DB_IS_LOCKED=${NETALERTX_LOG}/db_is_locked.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_HTML=${NETALERTX_LOG}/report_output.html
|
||||||
|
ENV LOG_STDERR=${NETALERTX_LOG}/stderr.log
|
||||||
|
ENV LOG_APP_PHP_ERRORS=${NETALERTX_LOG}/app.php_errors.log
|
||||||
|
ENV LOG_EXECUTION_QUEUE=${NETALERTX_LOG}/execution_queue.log
|
||||||
|
ENV LOG_REPORT_OUTPUT_JSON=${NETALERTX_LOG}/report_output.json
|
||||||
|
ENV LOG_STDOUT=${NETALERTX_LOG}/stdout.log
|
||||||
|
ENV LOG_CROND=${NETALERTX_LOG}/crond.log
|
||||||
|
|
||||||
|
# System Services configuration files
|
||||||
|
ENV SYSTEM_SERVICES=/services
|
||||||
|
ENV SYSTEM_SERVICES_CONFIG=${SYSTEM_SERVICES}/config
|
||||||
|
ENV SYSTEM_NGINIX_CONFIG=${SYSTEM_SERVICES_CONFIG}/nginx
|
||||||
|
ENV SYSTEM_NGINX_CONFIG_FILE=${SYSTEM_NGINIX_CONFIG}/nginx.conf
|
||||||
|
ENV NETALERTX_CONFIG_FILE=${NETALERTX_CONFIG}/app.conf
|
||||||
|
ENV SYSTEM_SERVICES_PHP_FOLDER=${SYSTEM_SERVICES_CONFIG}/php
|
||||||
|
ENV SYSTEM_SERVICES_PHP_FPM_D=${SYSTEM_SERVICES_PHP_FOLDER}/php-fpm.d
|
||||||
|
ENV SYSTEM_SERVICES_CROND=${SYSTEM_SERVICES_CONFIG}/crond
|
||||||
|
ENV SYSTEM_SERVICES_RUN=${SYSTEM_SERVICES}/run
|
||||||
|
ENV SYSTEM_SERVICES_RUN_TMP=${SYSTEM_SERVICES_RUN}/tmp
|
||||||
|
ENV SYSTEM_SERVICES_RUN_LOG=${SYSTEM_SERVICES_RUN}/logs
|
||||||
|
ENV PHP_FPM_CONFIG_FILE=${SYSTEM_SERVICES_PHP_FOLDER}/php-fpm.conf
|
||||||
|
|
||||||
|
#Python environment
|
||||||
|
ENV PYTHONPATH=${NETALERTX_SERVER}
|
||||||
|
ENV PYTHONUNBUFFERED=1
|
||||||
|
ENV VIRTUAL_ENV=/opt/venv
|
||||||
|
ENV VIRTUAL_ENV_BIN=/opt/venv/bin
|
||||||
|
ENV PATH="${VIRTUAL_ENV}/bin:${PATH}:/services"
|
||||||
|
ENV VENDORSPATH=/app/back/ieee-oui.txt
|
||||||
|
ENV VENDORSPATH_NEWEST=/services/run/tmp/ieee-oui.txt
|
||||||
|
|
||||||
|
|
||||||
|
# App Environment
|
||||||
|
ENV LISTEN_ADDR=0.0.0.0
|
||||||
|
ENV PORT=20211
|
||||||
|
ENV NETALERTX_DEBUG=0
|
||||||
|
|
||||||
|
#Container environment
|
||||||
|
ENV ENVIRONMENT=debian
|
||||||
|
ENV USER=netalertx
|
||||||
|
ENV USER_ID=1000
|
||||||
|
ENV USER_GID=1000
|
||||||
|
|
||||||
# Todo, figure out why using a workdir instead of full paths don't work
|
# Todo, figure out why using a workdir instead of full paths don't work
|
||||||
# Todo, do we still need all these packages? I can already see sudo which isn't needed
|
# Todo, do we still need all these packages? I can already see sudo which isn't needed
|
||||||
|
|
||||||
RUN apt-get update
|
|
||||||
RUN apt-get install sudo -y
|
|
||||||
|
|
||||||
ARG INSTALL_DIR=/app
|
|
||||||
|
|
||||||
# create pi user and group
|
# create pi user and group
|
||||||
# add root and www-data to pi group so they can r/w files and db
|
# add root and www-data to pi group so they can r/w files and db
|
||||||
RUN groupadd --gid "${USER_GID}" "${USER}" && \
|
RUN groupadd --gid "${USER_GID}" "${USER}" && \
|
||||||
useradd \
|
useradd \
|
||||||
--uid ${USER_ID} \
|
--uid ${USER_ID} \
|
||||||
--gid ${USER_GID} \
|
--gid ${USER_GID} \
|
||||||
--create-home \
|
--create-home \
|
||||||
--shell /bin/bash \
|
--shell /bin/bash \
|
||||||
${USER} && \
|
${USER} && \
|
||||||
usermod -a -G ${USER_GID} root && \
|
usermod -a -G ${USER_GID} root && \
|
||||||
usermod -a -G ${USER_GID} www-data
|
usermod -a -G ${USER_GID} www-data
|
||||||
|
|
||||||
|
COPY --chmod=775 --chown=${USER_ID}:${USER_GID} install/production-filesystem/ /
|
||||||
COPY --chmod=775 --chown=${USER_ID}:${USER_GID} . ${INSTALL_DIR}/
|
COPY --chmod=775 --chown=${USER_ID}:${USER_GID} . ${INSTALL_DIR}/
|
||||||
|
|
||||||
|
|
||||||
# ❗ IMPORTANT - if you modify this file modify the /install/install_dependecies.debian.sh file as well ❗
|
# ❗ IMPORTANT - if you modify this file modify the /install/install_dependecies.debian.sh file as well ❗
|
||||||
|
RUN apt update && apt-get install -y \
|
||||||
|
tini snmp ca-certificates curl libwww-perl arp-scan sudo gettext-base \
|
||||||
|
nginx-light php php-cgi php-fpm php-sqlite3 php-curl sqlite3 dnsutils net-tools \
|
||||||
|
python3 python3-dev iproute2 nmap python3-pip zip git systemctl usbutils traceroute nbtscan openrc \
|
||||||
|
busybox nginx nginx-core mtr python3-venv
|
||||||
|
|
||||||
|
# While php8.3 is in debian bookworm repos, php-fpm is not included so we need to add sury.org repo
|
||||||
RUN apt-get install -y \
|
# (Ondřej Surý maintains php packages for debian. This is temp until debian includes php-fpm in their
|
||||||
tini snmp ca-certificates curl libwww-perl arp-scan perl apt-utils cron sudo \
|
# repos. Likely it will be in Debian Trixie.). This keeps the image up-to-date with the alpine version.
|
||||||
nginx-light php php-cgi php-fpm php-sqlite3 php-curl sqlite3 dnsutils net-tools php-openssl \
|
RUN apt-get install -y --no-install-recommends \
|
||||||
python3 python3-dev iproute2 nmap python3-pip zip systemctl usbutils traceroute nbtscan
|
apt-transport-https \
|
||||||
|
ca-certificates \
|
||||||
# Alternate dependencies
|
lsb-release \
|
||||||
RUN apt-get install nginx nginx-core mtr php-fpm php8.2-fpm php-cli php8.2 php8.2-sqlite3 -y
|
wget && \
|
||||||
RUN phpenmod -v 8.2 sqlite3
|
wget -O /etc/apt/trusted.gpg.d/php.gpg https://packages.sury.org/php/apt.gpg && \
|
||||||
|
echo "deb https://packages.sury.org/php/ $(lsb_release -sc) main" > /etc/apt/sources.list.d/php.list && \
|
||||||
|
apt-get update && \
|
||||||
|
apt-get install -y php8.3-fpm php8.3-cli php8.3-sqlite3 php8.3-common php8.3-curl php8.3-cgi && \
|
||||||
|
ln -s /usr/sbin/php-fpm8.3 /usr/sbin/php-fpm83 # make it compatible with alpine version
|
||||||
|
|
||||||
# Setup virtual python environment and use pip3 to install packages
|
# Setup virtual python environment and use pip3 to install packages
|
||||||
RUN apt-get install -y python3-venv
|
RUN python3 -m venv ${VIRTUAL_ENV} && \
|
||||||
RUN python3 -m venv myenv
|
/bin/bash -c "source ${VIRTUAL_ENV_BIN}/activate && update-alternatives --install /usr/bin/python python /usr/bin/python3 10 && pip3 install -r ${INSTALL_DIR}/requirements.txt"
|
||||||
|
|
||||||
|
# Configure php-fpm
|
||||||
|
RUN chmod -R 755 /services && \
|
||||||
|
chown -R ${USER}:${USER_GID} /services && \
|
||||||
|
sed -i 's/^;listen.mode = .*/listen.mode = 0666/' ${SYSTEM_SERVICES_PHP_FPM_D}/www.conf && \
|
||||||
|
printf "user = %s\ngroup = %s\n" "${USER}" "${USER_GID}" >> /services/config/php/php-fpm.d/www.conf
|
||||||
|
|
||||||
|
|
||||||
RUN /bin/bash -c "source myenv/bin/activate && update-alternatives --install /usr/bin/python python /usr/bin/python3 10 && pip3 install openwrt-luci-rpc asusrouter asyncio aiohttp graphene flask flask-cors unifi-sm-api tplink-omada-client wakeonlan pycryptodome requests paho-mqtt scapy cron-converter pytz json2table dhcp-leases pyunifi speedtest-cli chardet python-nmap dnspython librouteros yattag zeroconf "
|
|
||||||
|
|
||||||
# Create a buildtimestamp.txt to later check if a new version was released
|
# Create a buildtimestamp.txt to later check if a new version was released
|
||||||
RUN date +%s > ${INSTALL_DIR}/front/buildtimestamp.txt
|
RUN date +%s > ${INSTALL_DIR}/front/buildtimestamp.txt
|
||||||
|
USER netalertx:netalertx
|
||||||
CMD ["${INSTALL_DIR}/install/start.debian.sh"]
|
ENTRYPOINT ["/bin/bash","/entrypoint.sh"]
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
411
back/app.sql
Normal file
411
back/app.sql
Normal file
@@ -0,0 +1,411 @@
|
|||||||
|
CREATE TABLE sqlite_stat1(tbl,idx,stat);
|
||||||
|
CREATE TABLE Events (eve_MAC STRING (50) NOT NULL COLLATE NOCASE, eve_IP STRING (50) NOT NULL COLLATE NOCASE, eve_DateTime DATETIME NOT NULL, eve_EventType STRING (30) NOT NULL COLLATE NOCASE, eve_AdditionalInfo STRING (250) DEFAULT (''), eve_PendingAlertEmail BOOLEAN NOT NULL CHECK (eve_PendingAlertEmail IN (0, 1)) DEFAULT (1), eve_PairEventRowid INTEGER);
|
||||||
|
CREATE TABLE Sessions (ses_MAC STRING (50) COLLATE NOCASE, ses_IP STRING (50) COLLATE NOCASE, ses_EventTypeConnection STRING (30) COLLATE NOCASE, ses_DateTimeConnection DATETIME, ses_EventTypeDisconnection STRING (30) COLLATE NOCASE, ses_DateTimeDisconnection DATETIME, ses_StillConnected BOOLEAN, ses_AdditionalInfo STRING (250));
|
||||||
|
CREATE TABLE IF NOT EXISTS "Online_History" (
|
||||||
|
"Index" INTEGER,
|
||||||
|
"Scan_Date" TEXT,
|
||||||
|
"Online_Devices" INTEGER,
|
||||||
|
"Down_Devices" INTEGER,
|
||||||
|
"All_Devices" INTEGER,
|
||||||
|
"Archived_Devices" INTEGER,
|
||||||
|
"Offline_Devices" INTEGER,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE sqlite_sequence(name,seq);
|
||||||
|
CREATE TABLE Devices (
|
||||||
|
devMac STRING (50) PRIMARY KEY NOT NULL COLLATE NOCASE,
|
||||||
|
devName STRING (50) NOT NULL DEFAULT "(unknown)",
|
||||||
|
devOwner STRING (30) DEFAULT "(unknown)" NOT NULL,
|
||||||
|
devType STRING (30),
|
||||||
|
devVendor STRING (250),
|
||||||
|
devFavorite BOOLEAN CHECK (devFavorite IN (0, 1)) DEFAULT (0) NOT NULL,
|
||||||
|
devGroup STRING (10),
|
||||||
|
devComments TEXT,
|
||||||
|
devFirstConnection DATETIME NOT NULL,
|
||||||
|
devLastConnection DATETIME NOT NULL,
|
||||||
|
devLastIP STRING (50) NOT NULL COLLATE NOCASE,
|
||||||
|
devStaticIP BOOLEAN DEFAULT (0) NOT NULL CHECK (devStaticIP IN (0, 1)),
|
||||||
|
devScan INTEGER DEFAULT (1) NOT NULL,
|
||||||
|
devLogEvents BOOLEAN NOT NULL DEFAULT (1) CHECK (devLogEvents IN (0, 1)),
|
||||||
|
devAlertEvents BOOLEAN NOT NULL DEFAULT (1) CHECK (devAlertEvents IN (0, 1)),
|
||||||
|
devAlertDown BOOLEAN NOT NULL DEFAULT (0) CHECK (devAlertDown IN (0, 1)),
|
||||||
|
devSkipRepeated INTEGER DEFAULT 0 NOT NULL,
|
||||||
|
devLastNotification DATETIME,
|
||||||
|
devPresentLastScan BOOLEAN NOT NULL DEFAULT (0) CHECK (devPresentLastScan IN (0, 1)),
|
||||||
|
devIsNew BOOLEAN NOT NULL DEFAULT (1) CHECK (devIsNew IN (0, 1)),
|
||||||
|
devLocation STRING (250) COLLATE NOCASE,
|
||||||
|
devIsArchived BOOLEAN NOT NULL DEFAULT (0) CHECK (devIsArchived IN (0, 1)),
|
||||||
|
devParentMAC TEXT,
|
||||||
|
devParentPort INTEGER,
|
||||||
|
devIcon TEXT,
|
||||||
|
devGUID TEXT,
|
||||||
|
devSite TEXT,
|
||||||
|
devSSID TEXT,
|
||||||
|
devSyncHubNode TEXT,
|
||||||
|
devSourcePlugin TEXT
|
||||||
|
, "devCustomProps" TEXT);
|
||||||
|
CREATE TABLE IF NOT EXISTS "Settings" (
|
||||||
|
"setKey" TEXT,
|
||||||
|
"setName" TEXT,
|
||||||
|
"setDescription" TEXT,
|
||||||
|
"setType" TEXT,
|
||||||
|
"setOptions" TEXT,
|
||||||
|
"setGroup" TEXT,
|
||||||
|
"setValue" TEXT,
|
||||||
|
"setEvents" TEXT,
|
||||||
|
"setOverriddenByEnv" INTEGER
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS "Parameters" (
|
||||||
|
"par_ID" TEXT PRIMARY KEY,
|
||||||
|
"par_Value" TEXT
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_Objects(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Plugin TEXT NOT NULL,
|
||||||
|
Object_PrimaryID TEXT NOT NULL,
|
||||||
|
Object_SecondaryID TEXT NOT NULL,
|
||||||
|
DateTimeCreated TEXT NOT NULL,
|
||||||
|
DateTimeChanged TEXT NOT NULL,
|
||||||
|
Watched_Value1 TEXT NOT NULL,
|
||||||
|
Watched_Value2 TEXT NOT NULL,
|
||||||
|
Watched_Value3 TEXT NOT NULL,
|
||||||
|
Watched_Value4 TEXT NOT NULL,
|
||||||
|
Status TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
UserData TEXT NOT NULL,
|
||||||
|
ForeignKey TEXT NOT NULL,
|
||||||
|
SyncHubNodeName TEXT,
|
||||||
|
"HelpVal1" TEXT,
|
||||||
|
"HelpVal2" TEXT,
|
||||||
|
"HelpVal3" TEXT,
|
||||||
|
"HelpVal4" TEXT,
|
||||||
|
ObjectGUID TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_Events(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Plugin TEXT NOT NULL,
|
||||||
|
Object_PrimaryID TEXT NOT NULL,
|
||||||
|
Object_SecondaryID TEXT NOT NULL,
|
||||||
|
DateTimeCreated TEXT NOT NULL,
|
||||||
|
DateTimeChanged TEXT NOT NULL,
|
||||||
|
Watched_Value1 TEXT NOT NULL,
|
||||||
|
Watched_Value2 TEXT NOT NULL,
|
||||||
|
Watched_Value3 TEXT NOT NULL,
|
||||||
|
Watched_Value4 TEXT NOT NULL,
|
||||||
|
Status TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
UserData TEXT NOT NULL,
|
||||||
|
ForeignKey TEXT NOT NULL,
|
||||||
|
SyncHubNodeName TEXT,
|
||||||
|
"HelpVal1" TEXT,
|
||||||
|
"HelpVal2" TEXT,
|
||||||
|
"HelpVal3" TEXT,
|
||||||
|
"HelpVal4" TEXT, "ObjectGUID" TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_History(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Plugin TEXT NOT NULL,
|
||||||
|
Object_PrimaryID TEXT NOT NULL,
|
||||||
|
Object_SecondaryID TEXT NOT NULL,
|
||||||
|
DateTimeCreated TEXT NOT NULL,
|
||||||
|
DateTimeChanged TEXT NOT NULL,
|
||||||
|
Watched_Value1 TEXT NOT NULL,
|
||||||
|
Watched_Value2 TEXT NOT NULL,
|
||||||
|
Watched_Value3 TEXT NOT NULL,
|
||||||
|
Watched_Value4 TEXT NOT NULL,
|
||||||
|
Status TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
UserData TEXT NOT NULL,
|
||||||
|
ForeignKey TEXT NOT NULL,
|
||||||
|
SyncHubNodeName TEXT,
|
||||||
|
"HelpVal1" TEXT,
|
||||||
|
"HelpVal2" TEXT,
|
||||||
|
"HelpVal3" TEXT,
|
||||||
|
"HelpVal4" TEXT, "ObjectGUID" TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_Language_Strings(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Language_Code TEXT NOT NULL,
|
||||||
|
String_Key TEXT NOT NULL,
|
||||||
|
String_Value TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE CurrentScan (
|
||||||
|
cur_MAC STRING(50) NOT NULL COLLATE NOCASE,
|
||||||
|
cur_IP STRING(50) NOT NULL COLLATE NOCASE,
|
||||||
|
cur_Vendor STRING(250),
|
||||||
|
cur_ScanMethod STRING(10),
|
||||||
|
cur_Name STRING(250),
|
||||||
|
cur_LastQuery STRING(250),
|
||||||
|
cur_DateTime STRING(250),
|
||||||
|
cur_SyncHubNodeName STRING(50),
|
||||||
|
cur_NetworkSite STRING(250),
|
||||||
|
cur_SSID STRING(250),
|
||||||
|
cur_NetworkNodeMAC STRING(250),
|
||||||
|
cur_PORT STRING(250),
|
||||||
|
cur_Type STRING(250),
|
||||||
|
UNIQUE(cur_MAC)
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS "AppEvents" (
|
||||||
|
"Index" INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
"GUID" TEXT UNIQUE,
|
||||||
|
"AppEventProcessed" BOOLEAN,
|
||||||
|
"DateTimeCreated" TEXT,
|
||||||
|
"ObjectType" TEXT,
|
||||||
|
"ObjectGUID" TEXT,
|
||||||
|
"ObjectPlugin" TEXT,
|
||||||
|
"ObjectPrimaryID" TEXT,
|
||||||
|
"ObjectSecondaryID" TEXT,
|
||||||
|
"ObjectForeignKey" TEXT,
|
||||||
|
"ObjectIndex" TEXT,
|
||||||
|
"ObjectIsNew" BOOLEAN,
|
||||||
|
"ObjectIsArchived" BOOLEAN,
|
||||||
|
"ObjectStatusColumn" TEXT,
|
||||||
|
"ObjectStatus" TEXT,
|
||||||
|
"AppEventType" TEXT,
|
||||||
|
"Helper1" TEXT,
|
||||||
|
"Helper2" TEXT,
|
||||||
|
"Helper3" TEXT,
|
||||||
|
"Extra" TEXT
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS "Notifications" (
|
||||||
|
"Index" INTEGER,
|
||||||
|
"GUID" TEXT UNIQUE,
|
||||||
|
"DateTimeCreated" TEXT,
|
||||||
|
"DateTimePushed" TEXT,
|
||||||
|
"Status" TEXT,
|
||||||
|
"JSON" TEXT,
|
||||||
|
"Text" TEXT,
|
||||||
|
"HTML" TEXT,
|
||||||
|
"PublishedVia" TEXT,
|
||||||
|
"Extra" TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE INDEX IDX_eve_DateTime ON Events (eve_DateTime);
|
||||||
|
CREATE INDEX IDX_eve_EventType ON Events (eve_EventType COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_eve_MAC ON Events (eve_MAC COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_eve_PairEventRowid ON Events (eve_PairEventRowid);
|
||||||
|
CREATE INDEX IDX_ses_EventTypeDisconnection ON Sessions (ses_EventTypeDisconnection COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_ses_EventTypeConnection ON Sessions (ses_EventTypeConnection COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_ses_DateTimeDisconnection ON Sessions (ses_DateTimeDisconnection);
|
||||||
|
CREATE INDEX IDX_ses_MAC ON Sessions (ses_MAC COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_ses_DateTimeConnection ON Sessions (ses_DateTimeConnection);
|
||||||
|
CREATE INDEX IDX_dev_PresentLastScan ON Devices (devPresentLastScan);
|
||||||
|
CREATE INDEX IDX_dev_FirstConnection ON Devices (devFirstConnection);
|
||||||
|
CREATE INDEX IDX_dev_AlertDeviceDown ON Devices (devAlertDown);
|
||||||
|
CREATE INDEX IDX_dev_StaticIP ON Devices (devStaticIP);
|
||||||
|
CREATE INDEX IDX_dev_ScanCycle ON Devices (devScan);
|
||||||
|
CREATE INDEX IDX_dev_Favorite ON Devices (devFavorite);
|
||||||
|
CREATE INDEX IDX_dev_LastIP ON Devices (devLastIP);
|
||||||
|
CREATE INDEX IDX_dev_NewDevice ON Devices (devIsNew);
|
||||||
|
CREATE INDEX IDX_dev_Archived ON Devices (devIsArchived);
|
||||||
|
CREATE VIEW Events_Devices AS
|
||||||
|
SELECT *
|
||||||
|
FROM Events
|
||||||
|
LEFT JOIN Devices ON eve_MAC = devMac
|
||||||
|
/* Events_Devices(eve_MAC,eve_IP,eve_DateTime,eve_EventType,eve_AdditionalInfo,eve_PendingAlertEmail,eve_PairEventRowid,devMac,devName,devOwner,devType,devVendor,devFavorite,devGroup,devComments,devFirstConnection,devLastConnection,devLastIP,devStaticIP,devScan,devLogEvents,devAlertEvents,devAlertDown,devSkipRepeated,devLastNotification,devPresentLastScan,devIsNew,devLocation,devIsArchived,devParentMAC,devParentPort,devIcon,devGUID,devSite,devSSID,devSyncHubNode,devSourcePlugin,devCustomProps) */;
|
||||||
|
CREATE VIEW LatestEventsPerMAC AS
|
||||||
|
WITH RankedEvents AS (
|
||||||
|
SELECT
|
||||||
|
e.*,
|
||||||
|
ROW_NUMBER() OVER (PARTITION BY e.eve_MAC ORDER BY e.eve_DateTime DESC) AS row_num
|
||||||
|
FROM Events AS e
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
e.*,
|
||||||
|
d.*,
|
||||||
|
c.*
|
||||||
|
FROM RankedEvents AS e
|
||||||
|
LEFT JOIN Devices AS d ON e.eve_MAC = d.devMac
|
||||||
|
INNER JOIN CurrentScan AS c ON e.eve_MAC = c.cur_MAC
|
||||||
|
WHERE e.row_num = 1
|
||||||
|
/* LatestEventsPerMAC(eve_MAC,eve_IP,eve_DateTime,eve_EventType,eve_AdditionalInfo,eve_PendingAlertEmail,eve_PairEventRowid,row_num,devMac,devName,devOwner,devType,devVendor,devFavorite,devGroup,devComments,devFirstConnection,devLastConnection,devLastIP,devStaticIP,devScan,devLogEvents,devAlertEvents,devAlertDown,devSkipRepeated,devLastNotification,devPresentLastScan,devIsNew,devLocation,devIsArchived,devParentMAC,devParentPort,devIcon,devGUID,devSite,devSSID,devSyncHubNode,devSourcePlugin,devCustomProps,cur_MAC,cur_IP,cur_Vendor,cur_ScanMethod,cur_Name,cur_LastQuery,cur_DateTime,cur_SyncHubNodeName,cur_NetworkSite,cur_SSID,cur_NetworkNodeMAC,cur_PORT,cur_Type) */;
|
||||||
|
CREATE VIEW Sessions_Devices AS SELECT * FROM Sessions LEFT JOIN "Devices" ON ses_MAC = devMac
|
||||||
|
/* Sessions_Devices(ses_MAC,ses_IP,ses_EventTypeConnection,ses_DateTimeConnection,ses_EventTypeDisconnection,ses_DateTimeDisconnection,ses_StillConnected,ses_AdditionalInfo,devMac,devName,devOwner,devType,devVendor,devFavorite,devGroup,devComments,devFirstConnection,devLastConnection,devLastIP,devStaticIP,devScan,devLogEvents,devAlertEvents,devAlertDown,devSkipRepeated,devLastNotification,devPresentLastScan,devIsNew,devLocation,devIsArchived,devParentMAC,devParentPort,devIcon,devGUID,devSite,devSSID,devSyncHubNode,devSourcePlugin,devCustomProps) */;
|
||||||
|
CREATE VIEW Convert_Events_to_Sessions AS SELECT EVE1.eve_MAC,
|
||||||
|
EVE1.eve_IP,
|
||||||
|
EVE1.eve_EventType AS eve_EventTypeConnection,
|
||||||
|
EVE1.eve_DateTime AS eve_DateTimeConnection,
|
||||||
|
CASE WHEN EVE2.eve_EventType IN ('Disconnected', 'Device Down') OR
|
||||||
|
EVE2.eve_EventType IS NULL THEN EVE2.eve_EventType ELSE '<missing event>' END AS eve_EventTypeDisconnection,
|
||||||
|
CASE WHEN EVE2.eve_EventType IN ('Disconnected', 'Device Down') THEN EVE2.eve_DateTime ELSE NULL END AS eve_DateTimeDisconnection,
|
||||||
|
CASE WHEN EVE2.eve_EventType IS NULL THEN 1 ELSE 0 END AS eve_StillConnected,
|
||||||
|
EVE1.eve_AdditionalInfo
|
||||||
|
FROM Events AS EVE1
|
||||||
|
LEFT JOIN
|
||||||
|
Events AS EVE2 ON EVE1.eve_PairEventRowID = EVE2.RowID
|
||||||
|
WHERE EVE1.eve_EventType IN ('New Device', 'Connected','Down Reconnected')
|
||||||
|
UNION
|
||||||
|
SELECT eve_MAC,
|
||||||
|
eve_IP,
|
||||||
|
'<missing event>' AS eve_EventTypeConnection,
|
||||||
|
NULL AS eve_DateTimeConnection,
|
||||||
|
eve_EventType AS eve_EventTypeDisconnection,
|
||||||
|
eve_DateTime AS eve_DateTimeDisconnection,
|
||||||
|
0 AS eve_StillConnected,
|
||||||
|
eve_AdditionalInfo
|
||||||
|
FROM Events AS EVE1
|
||||||
|
WHERE (eve_EventType = 'Device Down' OR
|
||||||
|
eve_EventType = 'Disconnected') AND
|
||||||
|
EVE1.eve_PairEventRowID IS NULL
|
||||||
|
/* Convert_Events_to_Sessions(eve_MAC,eve_IP,eve_EventTypeConnection,eve_DateTimeConnection,eve_EventTypeDisconnection,eve_DateTimeDisconnection,eve_StillConnected,eve_AdditionalInfo) */;
|
||||||
|
CREATE TRIGGER "trg_insert_devices"
|
||||||
|
AFTER INSERT ON "Devices"
|
||||||
|
WHEN NOT EXISTS (
|
||||||
|
SELECT 1 FROM AppEvents
|
||||||
|
WHERE AppEventProcessed = 0
|
||||||
|
AND ObjectType = 'Devices'
|
||||||
|
AND ObjectGUID = NEW.devGUID
|
||||||
|
AND ObjectStatus = CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END
|
||||||
|
AND AppEventType = 'insert'
|
||||||
|
)
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO "AppEvents" (
|
||||||
|
"GUID",
|
||||||
|
"DateTimeCreated",
|
||||||
|
"AppEventProcessed",
|
||||||
|
"ObjectType",
|
||||||
|
"ObjectGUID",
|
||||||
|
"ObjectPrimaryID",
|
||||||
|
"ObjectSecondaryID",
|
||||||
|
"ObjectStatus",
|
||||||
|
"ObjectStatusColumn",
|
||||||
|
"ObjectIsNew",
|
||||||
|
"ObjectIsArchived",
|
||||||
|
"ObjectForeignKey",
|
||||||
|
"ObjectPlugin",
|
||||||
|
"AppEventType"
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
|
||||||
|
lower(
|
||||||
|
hex(randomblob(4)) || '-' || hex(randomblob(2)) || '-' || '4' ||
|
||||||
|
substr(hex( randomblob(2)), 2) || '-' ||
|
||||||
|
substr('AB89', 1 + (abs(random()) % 4) , 1) ||
|
||||||
|
substr(hex(randomblob(2)), 2) || '-' ||
|
||||||
|
hex(randomblob(6))
|
||||||
|
)
|
||||||
|
,
|
||||||
|
DATETIME('now'),
|
||||||
|
FALSE,
|
||||||
|
'Devices',
|
||||||
|
NEW.devGUID, -- ObjectGUID
|
||||||
|
NEW.devMac, -- ObjectPrimaryID
|
||||||
|
NEW.devLastIP, -- ObjectSecondaryID
|
||||||
|
CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END, -- ObjectStatus
|
||||||
|
'devPresentLastScan', -- ObjectStatusColumn
|
||||||
|
NEW.devIsNew, -- ObjectIsNew
|
||||||
|
NEW.devIsArchived, -- ObjectIsArchived
|
||||||
|
NEW.devGUID, -- ObjectForeignKey
|
||||||
|
'DEVICES', -- ObjectForeignKey
|
||||||
|
'insert'
|
||||||
|
);
|
||||||
|
END;
|
||||||
|
CREATE TRIGGER "trg_update_devices"
|
||||||
|
AFTER UPDATE ON "Devices"
|
||||||
|
WHEN NOT EXISTS (
|
||||||
|
SELECT 1 FROM AppEvents
|
||||||
|
WHERE AppEventProcessed = 0
|
||||||
|
AND ObjectType = 'Devices'
|
||||||
|
AND ObjectGUID = NEW.devGUID
|
||||||
|
AND ObjectStatus = CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END
|
||||||
|
AND AppEventType = 'update'
|
||||||
|
)
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO "AppEvents" (
|
||||||
|
"GUID",
|
||||||
|
"DateTimeCreated",
|
||||||
|
"AppEventProcessed",
|
||||||
|
"ObjectType",
|
||||||
|
"ObjectGUID",
|
||||||
|
"ObjectPrimaryID",
|
||||||
|
"ObjectSecondaryID",
|
||||||
|
"ObjectStatus",
|
||||||
|
"ObjectStatusColumn",
|
||||||
|
"ObjectIsNew",
|
||||||
|
"ObjectIsArchived",
|
||||||
|
"ObjectForeignKey",
|
||||||
|
"ObjectPlugin",
|
||||||
|
"AppEventType"
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
|
||||||
|
lower(
|
||||||
|
hex(randomblob(4)) || '-' || hex(randomblob(2)) || '-' || '4' ||
|
||||||
|
substr(hex( randomblob(2)), 2) || '-' ||
|
||||||
|
substr('AB89', 1 + (abs(random()) % 4) , 1) ||
|
||||||
|
substr(hex(randomblob(2)), 2) || '-' ||
|
||||||
|
hex(randomblob(6))
|
||||||
|
)
|
||||||
|
,
|
||||||
|
DATETIME('now'),
|
||||||
|
FALSE,
|
||||||
|
'Devices',
|
||||||
|
NEW.devGUID, -- ObjectGUID
|
||||||
|
NEW.devMac, -- ObjectPrimaryID
|
||||||
|
NEW.devLastIP, -- ObjectSecondaryID
|
||||||
|
CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END, -- ObjectStatus
|
||||||
|
'devPresentLastScan', -- ObjectStatusColumn
|
||||||
|
NEW.devIsNew, -- ObjectIsNew
|
||||||
|
NEW.devIsArchived, -- ObjectIsArchived
|
||||||
|
NEW.devGUID, -- ObjectForeignKey
|
||||||
|
'DEVICES', -- ObjectForeignKey
|
||||||
|
'update'
|
||||||
|
);
|
||||||
|
END;
|
||||||
|
CREATE TRIGGER "trg_delete_devices"
|
||||||
|
AFTER DELETE ON "Devices"
|
||||||
|
WHEN NOT EXISTS (
|
||||||
|
SELECT 1 FROM AppEvents
|
||||||
|
WHERE AppEventProcessed = 0
|
||||||
|
AND ObjectType = 'Devices'
|
||||||
|
AND ObjectGUID = OLD.devGUID
|
||||||
|
AND ObjectStatus = CASE WHEN OLD.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END
|
||||||
|
AND AppEventType = 'delete'
|
||||||
|
)
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO "AppEvents" (
|
||||||
|
"GUID",
|
||||||
|
"DateTimeCreated",
|
||||||
|
"AppEventProcessed",
|
||||||
|
"ObjectType",
|
||||||
|
"ObjectGUID",
|
||||||
|
"ObjectPrimaryID",
|
||||||
|
"ObjectSecondaryID",
|
||||||
|
"ObjectStatus",
|
||||||
|
"ObjectStatusColumn",
|
||||||
|
"ObjectIsNew",
|
||||||
|
"ObjectIsArchived",
|
||||||
|
"ObjectForeignKey",
|
||||||
|
"ObjectPlugin",
|
||||||
|
"AppEventType"
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
|
||||||
|
lower(
|
||||||
|
hex(randomblob(4)) || '-' || hex(randomblob(2)) || '-' || '4' ||
|
||||||
|
substr(hex( randomblob(2)), 2) || '-' ||
|
||||||
|
substr('AB89', 1 + (abs(random()) % 4) , 1) ||
|
||||||
|
substr(hex(randomblob(2)), 2) || '-' ||
|
||||||
|
hex(randomblob(6))
|
||||||
|
)
|
||||||
|
,
|
||||||
|
DATETIME('now'),
|
||||||
|
FALSE,
|
||||||
|
'Devices',
|
||||||
|
OLD.devGUID, -- ObjectGUID
|
||||||
|
OLD.devMac, -- ObjectPrimaryID
|
||||||
|
OLD.devLastIP, -- ObjectSecondaryID
|
||||||
|
CASE WHEN OLD.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END, -- ObjectStatus
|
||||||
|
'devPresentLastScan', -- ObjectStatusColumn
|
||||||
|
OLD.devIsNew, -- ObjectIsNew
|
||||||
|
OLD.devIsArchived, -- ObjectIsArchived
|
||||||
|
OLD.devGUID, -- ObjectForeignKey
|
||||||
|
'DEVICES', -- ObjectForeignKey
|
||||||
|
'delete'
|
||||||
|
);
|
||||||
|
END;
|
||||||
111367
back/ieee-oui.txt
Normal file
111367
back/ieee-oui.txt
Normal file
File diff suppressed because it is too large
Load Diff
2013
back/speedtest-cli
2013
back/speedtest-cli
File diff suppressed because it is too large
Load Diff
@@ -1,82 +1,76 @@
|
|||||||
services:
|
services:
|
||||||
netalertx:
|
netalertx:
|
||||||
privileged: true
|
network_mode: host # Use host networking for ARP scanning and other services
|
||||||
build:
|
build:
|
||||||
dockerfile: Dockerfile
|
context: . # Build context is the current directory
|
||||||
context: .
|
dockerfile: Dockerfile # Specify the Dockerfile to use
|
||||||
cache_from:
|
image: netalertx:latest
|
||||||
- type=registry,ref=docker.io/jokob-sk/netalertx:buildcache
|
container_name: netalertx # The name when you docker contiainer ls
|
||||||
container_name: netalertx
|
read_only: true # Make the container filesystem read-only
|
||||||
network_mode: host
|
cap_drop: # Drop all capabilities for enhanced security
|
||||||
# restart: unless-stopped
|
- ALL
|
||||||
|
cap_add: # Add only the necessary capabilities
|
||||||
|
- NET_ADMIN # Required for ARP scanning
|
||||||
|
- NET_RAW # Required for raw socket operations
|
||||||
|
- NET_BIND_SERVICE # Required to bind to privileged ports (nbtscan)
|
||||||
|
|
||||||
volumes:
|
volumes:
|
||||||
# - ${APP_DATA_LOCATION}/netalertx_dev/config:/app/config
|
- type: volume
|
||||||
- ${APP_DATA_LOCATION}/netalertx/config:/app/config
|
source: netalertx_config
|
||||||
# - ${APP_DATA_LOCATION}/netalertx_dev/db:/app/db
|
target: /app/config
|
||||||
- ${APP_DATA_LOCATION}/netalertx/db:/app/db
|
read_only: false
|
||||||
# (optional) useful for debugging if you have issues setting up the container
|
|
||||||
- ${APP_DATA_LOCATION}/netalertx/log:/app/log
|
- type: volume
|
||||||
# (API: OPTION 1) use for performance
|
source: netalertx_db
|
||||||
- type: tmpfs
|
target: /app/db
|
||||||
target: /app/api
|
read_only: false
|
||||||
# (API: OPTION 2) use when debugging issues
|
|
||||||
# - ${DEV_LOCATION}/api:/app/api
|
- type: bind
|
||||||
# ---------------------------------------------------------------------------
|
source: /etc/localtime
|
||||||
# DELETE START anyone trying to use this file: comment out / delete BELOW lines, they are only for development purposes
|
target: /etc/localtime
|
||||||
- ${APP_DATA_LOCATION}/netalertx/dhcp_samples/dhcp1.leases:/mnt/dhcp1.leases # test data for DCPLSS plugin
|
read_only: true
|
||||||
- ${APP_DATA_LOCATION}/netalertx/dhcp_samples/dhcp2.leases:/mnt/dhcp2.leases # test data for DCPLSS plugin
|
|
||||||
- ${APP_DATA_LOCATION}/netalertx/dhcp_samples/pihole_dhcp_full.leases:/etc/pihole/dhcp.leases # test data for DCPLSS plugin
|
# Use a custom Enterprise-configured nginx config for ldap or other settings
|
||||||
- ${APP_DATA_LOCATION}/netalertx/dhcp_samples/pihole_dhcp_2.leases:/etc/pihole/dhcp2.leases # test data for DCPLSS plugin
|
# - /custom-enterprise.conf:/services/config/nginx/conf.active/netalertx.conf:ro
|
||||||
- ${APP_DATA_LOCATION}/pihole/etc-pihole/pihole-FTL.db:/etc/pihole/pihole-FTL.db # test data for PIHOLE plugin
|
|
||||||
- ${DEV_LOCATION}/mkdocs.yml:/app/mkdocs.yml
|
# Test your plugin on the production container
|
||||||
- ${DEV_LOCATION}/docs:/app/docs
|
# - /path/on/host:/app/front/plugins/custom
|
||||||
- ${DEV_LOCATION}/server:/app/server
|
|
||||||
- ${DEV_LOCATION}/test:/app/test
|
# Retain logs - comment out tmpfs /app/log if you want to retain logs between container restarts
|
||||||
- ${DEV_LOCATION}/dockerfiles:/app/dockerfiles
|
# - /path/on/host/log:/app/log
|
||||||
# - ${APP_DATA_LOCATION}/netalertx/php.ini:/etc/php/8.2/fpm/php.ini
|
|
||||||
- ${DEV_LOCATION}/install:/app/install
|
# Tempfs mounts for writable directories in a read-only container and improve system performance
|
||||||
- ${DEV_LOCATION}/front/css:/app/front/css
|
tmpfs:
|
||||||
- ${DEV_LOCATION}/front/img:/app/front/img
|
# Speed up logging. This can be commented out to retain logs between container restarts
|
||||||
- ${DEV_LOCATION}/back/update_vendors.sh:/app/back/update_vendors.sh
|
- "/app/log:uid=20211,gid=20211,mode=1700,rw,noexec,nosuid,nodev,async,noatime,nodiratime"
|
||||||
- ${DEV_LOCATION}/front/lib:/app/front/lib
|
# Speed up API access as frontend/backend API is very chatty
|
||||||
- ${DEV_LOCATION}/front/js:/app/front/js
|
- "/app/api:uid=20211,gid=20211,mode=1700,rw,noexec,nosuid,nodev,sync,noatime,nodiratime"
|
||||||
- ${DEV_LOCATION}/front/php:/app/front/php
|
# Required for customization of the nginx listen addr/port without rebuilding the container
|
||||||
- ${DEV_LOCATION}/front/deviceDetails.php:/app/front/deviceDetails.php
|
- "/services/config/nginx/conf.active:uid=20211,gid=20211,mode=1700,rw,noexec,nosuid,nodev,async,noatime,nodiratime"
|
||||||
- ${DEV_LOCATION}/front/deviceDetailsEdit.php:/app/front/deviceDetailsEdit.php
|
# /services/config/nginx/conf.d is required for nginx and php to start
|
||||||
- ${DEV_LOCATION}/front/userNotifications.php:/app/front/userNotifications.php
|
- "/services/run:uid=20211,gid=20211,mode=1700,rw,noexec,nosuid,nodev,async,noatime,nodiratime"
|
||||||
- ${DEV_LOCATION}/front/deviceDetailsTools.php:/app/front/deviceDetailsTools.php
|
# /tmp is required by php for session save this should be reworked to /services/run/tmp
|
||||||
- ${DEV_LOCATION}/front/deviceDetailsPresence.php:/app/front/deviceDetailsPresence.php
|
- "/tmp:uid=20211,gid=20211,mode=1700,rw,noexec,nosuid,nodev,async,noatime,nodiratime"
|
||||||
- ${DEV_LOCATION}/front/deviceDetailsSessions.php:/app/front/deviceDetailsSessions.php
|
environment:
|
||||||
- ${DEV_LOCATION}/front/deviceDetailsEvents.php:/app/front/deviceDetailsEvents.php
|
LISTEN_ADDR: 0.0.0.0 # Listen for connections on all interfaces
|
||||||
- ${DEV_LOCATION}/front/devices.php:/app/front/devices.php
|
PORT: 20211 # Application port
|
||||||
- ${DEV_LOCATION}/front/events.php:/app/front/events.php
|
GRAPHQL_PORT: 20212 # GraphQL API port
|
||||||
- ${DEV_LOCATION}/front/plugins.php:/app/front/plugins.php
|
ALWAYS_FRESH_INSTALL: false # Set to true to reset your config and database on each container start
|
||||||
- ${DEV_LOCATION}/front/pluginsCore.php:/app/front/pluginsCore.php
|
NETALERTX_DEBUG: 0 # 0=kill all services and restart if any dies. 1 keeps running dead services.
|
||||||
- ${DEV_LOCATION}/front/index.php:/app/front/index.php
|
|
||||||
- ${DEV_LOCATION}/front/initCheck.php:/app/front/initCheck.php
|
# Resource limits to prevent resource exhaustion
|
||||||
- ${DEV_LOCATION}/front/maintenance.php:/app/front/maintenance.php
|
mem_limit: 2048m
|
||||||
- ${DEV_LOCATION}/front/network.php:/app/front/network.php
|
mem_reservation: 1024m
|
||||||
- ${DEV_LOCATION}/front/presence.php:/app/front/presence.php
|
cpus: 4
|
||||||
- ${DEV_LOCATION}/front/settings.php:/app/front/settings.php
|
pids_limit: 512
|
||||||
- ${DEV_LOCATION}/front/systeminfo.php:/app/front/systeminfo.php
|
logging:
|
||||||
- ${DEV_LOCATION}/front/systeminfoNetwork.php:/app/front/systeminfoNetwork.php
|
driver: "json-file"
|
||||||
- ${DEV_LOCATION}/front/systeminfoServer.php:/app/front/systeminfoServer.php
|
options:
|
||||||
- ${DEV_LOCATION}/front/systeminfoStorage.php:/app/front/systeminfoStorage.php
|
max-size: "10m"
|
||||||
- ${DEV_LOCATION}/front/cloud_services.php:/app/front/cloud_services.php
|
max-file: "3"
|
||||||
- ${DEV_LOCATION}/front/report.php:/app/front/report.php
|
restart: unless-stopped
|
||||||
- ${DEV_LOCATION}/front/workflows.php:/app/front/workflows.php
|
|
||||||
- ${DEV_LOCATION}/front/workflowsCore.php:/app/front/workflowsCore.php
|
volumes:
|
||||||
- ${DEV_LOCATION}/front/appEvents.php:/app/front/appEvents.php
|
netalertx_config:
|
||||||
- ${DEV_LOCATION}/front/appEventsCore.php:/app/front/appEventsCore.php
|
netalertx_db:
|
||||||
- ${DEV_LOCATION}/front/multiEditCore.php:/app/front/multiEditCore.php
|
|
||||||
- ${DEV_LOCATION}/front/plugins:/app/front/plugins
|
|
||||||
# DELETE END anyone trying to use this file: comment out / delete ABOVE lines, they are only for development purposes
|
|
||||||
# ---------------------------------------------------------------------------
|
|
||||||
environment:
|
|
||||||
# - APP_CONF_OVERRIDE={"SCAN_SUBNETS":"['192.168.1.0/24 --interface=eth1']","GRAPHQL_PORT":"20223","UI_theme":"Light"}
|
|
||||||
- TZ=${TZ}
|
|
||||||
- PORT=${PORT}
|
|
||||||
# ❗ DANGER ZONE BELOW - Setting ALWAYS_FRESH_INSTALL=true will delete the content of the /db & /config folders
|
|
||||||
- ALWAYS_FRESH_INSTALL=${ALWAYS_FRESH_INSTALL}
|
|
||||||
# - LOADED_PLUGINS=["DHCPLSS","PIHOLE","ASUSWRT","FREEBOX"]
|
|
||||||
|
|
||||||
|
|||||||
534
docker_build.log
Normal file
534
docker_build.log
Normal file
@@ -0,0 +1,534 @@
|
|||||||
|
#0 building with "default" instance using docker driver
|
||||||
|
|
||||||
|
#1 [internal] load build definition from Dockerfile
|
||||||
|
#1 transferring dockerfile: 5.29kB done
|
||||||
|
#1 DONE 0.1s
|
||||||
|
|
||||||
|
#2 [auth] library/alpine:pull token for registry-1.docker.io
|
||||||
|
#2 DONE 0.0s
|
||||||
|
|
||||||
|
#3 [internal] load metadata for docker.io/library/alpine:3.22
|
||||||
|
#3 DONE 0.4s
|
||||||
|
|
||||||
|
#4 [internal] load .dockerignore
|
||||||
|
#4 transferring context: 216B done
|
||||||
|
#4 DONE 0.1s
|
||||||
|
|
||||||
|
#5 [builder 1/15] FROM docker.io/library/alpine:3.22@sha256:4bcff63911fcb4448bd4fdacec207030997caf25e9bea4045fa6c8c44de311d1
|
||||||
|
#5 CACHED
|
||||||
|
|
||||||
|
#6 [internal] load build context
|
||||||
|
#6 transferring context: 36.76kB 0.0s done
|
||||||
|
#6 DONE 0.1s
|
||||||
|
|
||||||
|
#7 [builder 2/15] RUN apk add --no-cache bash shadow python3 python3-dev gcc musl-dev libffi-dev openssl-dev git && python -m venv /opt/venv
|
||||||
|
#7 0.443 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/main/x86_64/APKINDEX.tar.gz
|
||||||
|
#7 0.688 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/community/x86_64/APKINDEX.tar.gz
|
||||||
|
#7 1.107 (1/52) Upgrading libcrypto3 (3.5.1-r0 -> 3.5.3-r0)
|
||||||
|
#7 1.358 (2/52) Upgrading libssl3 (3.5.1-r0 -> 3.5.3-r0)
|
||||||
|
#7 1.400 (3/52) Installing ncurses-terminfo-base (6.5_p20250503-r0)
|
||||||
|
#7 1.413 (4/52) Installing libncursesw (6.5_p20250503-r0)
|
||||||
|
#7 1.444 (5/52) Installing readline (8.2.13-r1)
|
||||||
|
#7 1.471 (6/52) Installing bash (5.2.37-r0)
|
||||||
|
#7 1.570 Executing bash-5.2.37-r0.post-install
|
||||||
|
#7 1.593 (7/52) Installing libgcc (14.2.0-r6)
|
||||||
|
#7 1.605 (8/52) Installing jansson (2.14.1-r0)
|
||||||
|
#7 1.613 (9/52) Installing libstdc++ (14.2.0-r6)
|
||||||
|
#7 1.705 (10/52) Installing zstd-libs (1.5.7-r0)
|
||||||
|
#7 1.751 (11/52) Installing binutils (2.44-r3)
|
||||||
|
#7 2.041 (12/52) Installing libgomp (14.2.0-r6)
|
||||||
|
#7 2.064 (13/52) Installing libatomic (14.2.0-r6)
|
||||||
|
#7 2.071 (14/52) Installing gmp (6.3.0-r3)
|
||||||
|
#7 2.097 (15/52) Installing isl26 (0.26-r1)
|
||||||
|
#7 2.183 (16/52) Installing mpfr4 (4.2.1_p1-r0)
|
||||||
|
#7 2.219 (17/52) Installing mpc1 (1.3.1-r1)
|
||||||
|
#7 2.231 (18/52) Installing gcc (14.2.0-r6)
|
||||||
|
#7 6.782 (19/52) Installing brotli-libs (1.1.0-r2)
|
||||||
|
#7 6.828 (20/52) Installing c-ares (1.34.5-r0)
|
||||||
|
#7 6.846 (21/52) Installing libunistring (1.3-r0)
|
||||||
|
#7 6.919 (22/52) Installing libidn2 (2.3.7-r0)
|
||||||
|
#7 6.937 (23/52) Installing nghttp2-libs (1.65.0-r0)
|
||||||
|
#7 6.950 (24/52) Installing libpsl (0.21.5-r3)
|
||||||
|
#7 6.960 (25/52) Installing libcurl (8.14.1-r1)
|
||||||
|
#7 7.015 (26/52) Installing libexpat (2.7.2-r0)
|
||||||
|
#7 7.029 (27/52) Installing pcre2 (10.43-r1)
|
||||||
|
#7 7.069 (28/52) Installing git (2.49.1-r0)
|
||||||
|
#7 7.397 (29/52) Installing git-init-template (2.49.1-r0)
|
||||||
|
#7 7.404 (30/52) Installing linux-headers (6.14.2-r0)
|
||||||
|
#7 7.572 (31/52) Installing libffi (3.4.8-r0)
|
||||||
|
#7 7.578 (32/52) Installing pkgconf (2.4.3-r0)
|
||||||
|
#7 7.593 (33/52) Installing libffi-dev (3.4.8-r0)
|
||||||
|
#7 7.607 (34/52) Installing musl-dev (1.2.5-r10)
|
||||||
|
#7 7.961 (35/52) Installing openssl-dev (3.5.3-r0)
|
||||||
|
#7 8.021 (36/52) Installing libbz2 (1.0.8-r6)
|
||||||
|
#7 8.045 (37/52) Installing gdbm (1.24-r0)
|
||||||
|
#7 8.055 (38/52) Installing xz-libs (5.8.1-r0)
|
||||||
|
#7 8.071 (39/52) Installing mpdecimal (4.0.1-r0)
|
||||||
|
#7 8.090 (40/52) Installing libpanelw (6.5_p20250503-r0)
|
||||||
|
#7 8.098 (41/52) Installing sqlite-libs (3.49.2-r1)
|
||||||
|
#7 8.185 (42/52) Installing python3 (3.12.11-r0)
|
||||||
|
#7 8.904 (43/52) Installing python3-pycache-pyc0 (3.12.11-r0)
|
||||||
|
#7 9.292 (44/52) Installing pyc (3.12.11-r0)
|
||||||
|
#7 9.292 (45/52) Installing python3-pyc (3.12.11-r0)
|
||||||
|
#7 9.292 (46/52) Installing python3-dev (3.12.11-r0)
|
||||||
|
#7 10.71 (47/52) Installing libmd (1.1.0-r0)
|
||||||
|
#7 10.72 (48/52) Installing libbsd (0.12.2-r0)
|
||||||
|
#7 10.73 (49/52) Installing skalibs-libs (2.14.4.0-r0)
|
||||||
|
#7 10.75 (50/52) Installing utmps-libs (0.1.3.1-r0)
|
||||||
|
#7 10.76 (51/52) Installing linux-pam (1.7.0-r4)
|
||||||
|
#7 10.82 (52/52) Installing shadow (4.17.3-r0)
|
||||||
|
#7 10.88 Executing busybox-1.37.0-r18.trigger
|
||||||
|
#7 10.90 OK: 274 MiB in 66 packages
|
||||||
|
#7 DONE 14.4s
|
||||||
|
|
||||||
|
#8 [builder 3/15] RUN mkdir -p /app
|
||||||
|
#8 DONE 0.5s
|
||||||
|
|
||||||
|
#9 [builder 4/15] COPY api /app/api
|
||||||
|
#9 DONE 0.3s
|
||||||
|
|
||||||
|
#10 [builder 5/15] COPY back /app/back
|
||||||
|
#10 DONE 0.3s
|
||||||
|
|
||||||
|
#11 [builder 6/15] COPY config /app/config
|
||||||
|
#11 DONE 0.3s
|
||||||
|
|
||||||
|
#12 [builder 7/15] COPY db /app/db
|
||||||
|
#12 DONE 0.3s
|
||||||
|
|
||||||
|
#13 [builder 8/15] COPY dockerfiles /app/dockerfiles
|
||||||
|
#13 DONE 0.3s
|
||||||
|
|
||||||
|
#14 [builder 9/15] COPY front /app/front
|
||||||
|
#14 DONE 0.4s
|
||||||
|
|
||||||
|
#15 [builder 10/15] COPY server /app/server
|
||||||
|
#15 DONE 0.3s
|
||||||
|
|
||||||
|
#16 [builder 11/15] COPY install/crontab /etc/crontabs/root
|
||||||
|
#16 DONE 0.3s
|
||||||
|
|
||||||
|
#17 [builder 12/15] COPY dockerfiles/start* /start*.sh
|
||||||
|
#17 DONE 0.3s
|
||||||
|
|
||||||
|
#18 [builder 13/15] RUN pip install openwrt-luci-rpc asusrouter asyncio aiohttp graphene flask flask-cors unifi-sm-api tplink-omada-client wakeonlan pycryptodome requests paho-mqtt scapy cron-converter pytz json2table dhcp-leases pyunifi speedtest-cli chardet python-nmap dnspython librouteros yattag git+https://github.com/foreign-sub/aiofreepybox.git
|
||||||
|
#18 0.737 Collecting git+https://github.com/foreign-sub/aiofreepybox.git
|
||||||
|
#18 0.737 Cloning https://github.com/foreign-sub/aiofreepybox.git to /tmp/pip-req-build-waf5_npl
|
||||||
|
#18 0.738 Running command git clone --filter=blob:none --quiet https://github.com/foreign-sub/aiofreepybox.git /tmp/pip-req-build-waf5_npl
|
||||||
|
#18 1.617 Resolved https://github.com/foreign-sub/aiofreepybox.git to commit 4ee18ea0f3e76edc839c48eb8df1da59c1baee3d
|
||||||
|
#18 1.620 Installing build dependencies: started
|
||||||
|
#18 3.337 Installing build dependencies: finished with status 'done'
|
||||||
|
#18 3.337 Getting requirements to build wheel: started
|
||||||
|
#18 3.491 Getting requirements to build wheel: finished with status 'done'
|
||||||
|
#18 3.492 Preparing metadata (pyproject.toml): started
|
||||||
|
#18 3.650 Preparing metadata (pyproject.toml): finished with status 'done'
|
||||||
|
#18 3.724 Collecting openwrt-luci-rpc
|
||||||
|
#18 3.753 Downloading openwrt_luci_rpc-1.1.17-py2.py3-none-any.whl.metadata (4.9 kB)
|
||||||
|
#18 3.892 Collecting asusrouter
|
||||||
|
#18 3.900 Downloading asusrouter-1.21.0-py3-none-any.whl.metadata (33 kB)
|
||||||
|
#18 3.999 Collecting asyncio
|
||||||
|
#18 4.007 Downloading asyncio-4.0.0-py3-none-any.whl.metadata (994 bytes)
|
||||||
|
#18 4.576 Collecting aiohttp
|
||||||
|
#18 4.582 Downloading aiohttp-3.12.15-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (7.7 kB)
|
||||||
|
#18 4.729 Collecting graphene
|
||||||
|
#18 4.735 Downloading graphene-3.4.3-py2.py3-none-any.whl.metadata (6.9 kB)
|
||||||
|
#18 4.858 Collecting flask
|
||||||
|
#18 4.866 Downloading flask-3.1.2-py3-none-any.whl.metadata (3.2 kB)
|
||||||
|
#18 4.963 Collecting flask-cors
|
||||||
|
#18 4.972 Downloading flask_cors-6.0.1-py3-none-any.whl.metadata (5.3 kB)
|
||||||
|
#18 5.055 Collecting unifi-sm-api
|
||||||
|
#18 5.065 Downloading unifi_sm_api-0.2.1-py3-none-any.whl.metadata (2.3 kB)
|
||||||
|
#18 5.155 Collecting tplink-omada-client
|
||||||
|
#18 5.166 Downloading tplink_omada_client-1.4.4-py3-none-any.whl.metadata (3.5 kB)
|
||||||
|
#18 5.262 Collecting wakeonlan
|
||||||
|
#18 5.274 Downloading wakeonlan-3.1.0-py3-none-any.whl.metadata (4.3 kB)
|
||||||
|
#18 5.500 Collecting pycryptodome
|
||||||
|
#18 5.505 Downloading pycryptodome-3.23.0-cp37-abi3-musllinux_1_2_x86_64.whl.metadata (3.4 kB)
|
||||||
|
#18 5.653 Collecting requests
|
||||||
|
#18 5.660 Downloading requests-2.32.5-py3-none-any.whl.metadata (4.9 kB)
|
||||||
|
#18 5.764 Collecting paho-mqtt
|
||||||
|
#18 5.775 Downloading paho_mqtt-2.1.0-py3-none-any.whl.metadata (23 kB)
|
||||||
|
#18 5.890 Collecting scapy
|
||||||
|
#18 5.902 Downloading scapy-2.6.1-py3-none-any.whl.metadata (5.6 kB)
|
||||||
|
#18 6.002 Collecting cron-converter
|
||||||
|
#18 6.013 Downloading cron_converter-1.2.2-py3-none-any.whl.metadata (8.1 kB)
|
||||||
|
#18 6.187 Collecting pytz
|
||||||
|
#18 6.193 Downloading pytz-2025.2-py2.py3-none-any.whl.metadata (22 kB)
|
||||||
|
#18 6.285 Collecting json2table
|
||||||
|
#18 6.294 Downloading json2table-1.1.5-py2.py3-none-any.whl.metadata (6.0 kB)
|
||||||
|
#18 6.381 Collecting dhcp-leases
|
||||||
|
#18 6.387 Downloading dhcp_leases-0.1.6-py3-none-any.whl.metadata (5.9 kB)
|
||||||
|
#18 6.461 Collecting pyunifi
|
||||||
|
#18 6.471 Downloading pyunifi-2.21-py3-none-any.whl.metadata (274 bytes)
|
||||||
|
#18 6.582 Collecting speedtest-cli
|
||||||
|
#18 6.596 Downloading speedtest_cli-2.1.3-py2.py3-none-any.whl.metadata (6.8 kB)
|
||||||
|
#18 6.767 Collecting chardet
|
||||||
|
#18 6.780 Downloading chardet-5.2.0-py3-none-any.whl.metadata (3.4 kB)
|
||||||
|
#18 6.878 Collecting python-nmap
|
||||||
|
#18 6.886 Downloading python-nmap-0.7.1.tar.gz (44 kB)
|
||||||
|
#18 6.937 Installing build dependencies: started
|
||||||
|
#18 8.245 Installing build dependencies: finished with status 'done'
|
||||||
|
#18 8.246 Getting requirements to build wheel: started
|
||||||
|
#18 8.411 Getting requirements to build wheel: finished with status 'done'
|
||||||
|
#18 8.412 Preparing metadata (pyproject.toml): started
|
||||||
|
#18 8.575 Preparing metadata (pyproject.toml): finished with status 'done'
|
||||||
|
#18 8.648 Collecting dnspython
|
||||||
|
#18 8.654 Downloading dnspython-2.8.0-py3-none-any.whl.metadata (5.7 kB)
|
||||||
|
#18 8.741 Collecting librouteros
|
||||||
|
#18 8.752 Downloading librouteros-3.4.1-py3-none-any.whl.metadata (1.6 kB)
|
||||||
|
#18 8.869 Collecting yattag
|
||||||
|
#18 8.881 Downloading yattag-1.16.1.tar.gz (29 kB)
|
||||||
|
#18 8.925 Installing build dependencies: started
|
||||||
|
#18 10.23 Installing build dependencies: finished with status 'done'
|
||||||
|
#18 10.23 Getting requirements to build wheel: started
|
||||||
|
#18 10.38 Getting requirements to build wheel: finished with status 'done'
|
||||||
|
#18 10.39 Preparing metadata (pyproject.toml): started
|
||||||
|
#18 10.55 Preparing metadata (pyproject.toml): finished with status 'done'
|
||||||
|
#18 10.60 Collecting Click>=6.0 (from openwrt-luci-rpc)
|
||||||
|
#18 10.60 Downloading click-8.3.0-py3-none-any.whl.metadata (2.6 kB)
|
||||||
|
#18 10.70 Collecting packaging>=19.1 (from openwrt-luci-rpc)
|
||||||
|
#18 10.71 Downloading packaging-25.0-py3-none-any.whl.metadata (3.3 kB)
|
||||||
|
#18 10.87 Collecting urllib3>=1.26.14 (from asusrouter)
|
||||||
|
#18 10.88 Downloading urllib3-2.5.0-py3-none-any.whl.metadata (6.5 kB)
|
||||||
|
#18 10.98 Collecting xmltodict>=0.12.0 (from asusrouter)
|
||||||
|
#18 10.98 Downloading xmltodict-1.0.2-py3-none-any.whl.metadata (15 kB)
|
||||||
|
#18 11.09 Collecting aiohappyeyeballs>=2.5.0 (from aiohttp)
|
||||||
|
#18 11.10 Downloading aiohappyeyeballs-2.6.1-py3-none-any.whl.metadata (5.9 kB)
|
||||||
|
#18 11.19 Collecting aiosignal>=1.4.0 (from aiohttp)
|
||||||
|
#18 11.20 Downloading aiosignal-1.4.0-py3-none-any.whl.metadata (3.7 kB)
|
||||||
|
#18 11.32 Collecting attrs>=17.3.0 (from aiohttp)
|
||||||
|
#18 11.33 Downloading attrs-25.3.0-py3-none-any.whl.metadata (10 kB)
|
||||||
|
#18 11.47 Collecting frozenlist>=1.1.1 (from aiohttp)
|
||||||
|
#18 11.47 Downloading frozenlist-1.7.0-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (18 kB)
|
||||||
|
#18 11.76 Collecting multidict<7.0,>=4.5 (from aiohttp)
|
||||||
|
#18 11.77 Downloading multidict-6.6.4-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (5.3 kB)
|
||||||
|
#18 11.87 Collecting propcache>=0.2.0 (from aiohttp)
|
||||||
|
#18 11.88 Downloading propcache-0.3.2-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (12 kB)
|
||||||
|
#18 12.19 Collecting yarl<2.0,>=1.17.0 (from aiohttp)
|
||||||
|
#18 12.20 Downloading yarl-1.20.1-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (73 kB)
|
||||||
|
#18 12.31 Collecting graphql-core<3.3,>=3.1 (from graphene)
|
||||||
|
#18 12.32 Downloading graphql_core-3.2.6-py3-none-any.whl.metadata (11 kB)
|
||||||
|
#18 12.41 Collecting graphql-relay<3.3,>=3.1 (from graphene)
|
||||||
|
#18 12.42 Downloading graphql_relay-3.2.0-py3-none-any.whl.metadata (12 kB)
|
||||||
|
#18 12.50 Collecting python-dateutil<3,>=2.7.0 (from graphene)
|
||||||
|
#18 12.51 Downloading python_dateutil-2.9.0.post0-py2.py3-none-any.whl.metadata (8.4 kB)
|
||||||
|
#18 12.61 Collecting typing-extensions<5,>=4.7.1 (from graphene)
|
||||||
|
#18 12.61 Downloading typing_extensions-4.15.0-py3-none-any.whl.metadata (3.3 kB)
|
||||||
|
#18 12.71 Collecting blinker>=1.9.0 (from flask)
|
||||||
|
#18 12.72 Downloading blinker-1.9.0-py3-none-any.whl.metadata (1.6 kB)
|
||||||
|
#18 12.84 Collecting itsdangerous>=2.2.0 (from flask)
|
||||||
|
#18 12.85 Downloading itsdangerous-2.2.0-py3-none-any.whl.metadata (1.9 kB)
|
||||||
|
#18 12.97 Collecting jinja2>=3.1.2 (from flask)
|
||||||
|
#18 12.98 Downloading jinja2-3.1.6-py3-none-any.whl.metadata (2.9 kB)
|
||||||
|
#18 13.15 Collecting markupsafe>=2.1.1 (from flask)
|
||||||
|
#18 13.15 Downloading MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (4.0 kB)
|
||||||
|
#18 13.28 Collecting werkzeug>=3.1.0 (from flask)
|
||||||
|
#18 13.29 Downloading werkzeug-3.1.3-py3-none-any.whl.metadata (3.7 kB)
|
||||||
|
#18 13.42 Collecting awesomeversion>=22.9.0 (from tplink-omada-client)
|
||||||
|
#18 13.42 Downloading awesomeversion-25.8.0-py3-none-any.whl.metadata (9.8 kB)
|
||||||
|
#18 13.59 Collecting charset_normalizer<4,>=2 (from requests)
|
||||||
|
#18 13.59 Downloading charset_normalizer-3.4.3-cp312-cp312-musllinux_1_2_x86_64.whl.metadata (36 kB)
|
||||||
|
#18 13.77 Collecting idna<4,>=2.5 (from requests)
|
||||||
|
#18 13.78 Downloading idna-3.10-py3-none-any.whl.metadata (10 kB)
|
||||||
|
#18 13.94 Collecting certifi>=2017.4.17 (from requests)
|
||||||
|
#18 13.94 Downloading certifi-2025.8.3-py3-none-any.whl.metadata (2.4 kB)
|
||||||
|
#18 14.06 Collecting toml<0.11.0,>=0.10.2 (from librouteros)
|
||||||
|
#18 14.07 Downloading toml-0.10.2-py2.py3-none-any.whl.metadata (7.1 kB)
|
||||||
|
#18 14.25 Collecting six>=1.5 (from python-dateutil<3,>=2.7.0->graphene)
|
||||||
|
#18 14.26 Downloading six-1.17.0-py2.py3-none-any.whl.metadata (1.7 kB)
|
||||||
|
#18 14.33 Downloading openwrt_luci_rpc-1.1.17-py2.py3-none-any.whl (9.5 kB)
|
||||||
|
#18 14.37 Downloading asusrouter-1.21.0-py3-none-any.whl (131 kB)
|
||||||
|
#18 14.43 Downloading asyncio-4.0.0-py3-none-any.whl (5.6 kB)
|
||||||
|
#18 14.47 Downloading aiohttp-3.12.15-cp312-cp312-musllinux_1_2_x86_64.whl (1.7 MB)
|
||||||
|
#18 14.67 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 1.7/1.7 MB 8.3 MB/s eta 0:00:00
|
||||||
|
#18 14.68 Downloading graphene-3.4.3-py2.py3-none-any.whl (114 kB)
|
||||||
|
#18 14.73 Downloading flask-3.1.2-py3-none-any.whl (103 kB)
|
||||||
|
#18 14.78 Downloading flask_cors-6.0.1-py3-none-any.whl (13 kB)
|
||||||
|
#18 14.84 Downloading unifi_sm_api-0.2.1-py3-none-any.whl (16 kB)
|
||||||
|
#18 14.88 Downloading tplink_omada_client-1.4.4-py3-none-any.whl (46 kB)
|
||||||
|
#18 14.93 Downloading wakeonlan-3.1.0-py3-none-any.whl (5.0 kB)
|
||||||
|
#18 14.99 Downloading pycryptodome-3.23.0-cp37-abi3-musllinux_1_2_x86_64.whl (2.3 MB)
|
||||||
|
#18 15.23 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 2.3/2.3 MB 8.9 MB/s eta 0:00:00
|
||||||
|
#18 15.24 Downloading requests-2.32.5-py3-none-any.whl (64 kB)
|
||||||
|
#18 15.30 Downloading paho_mqtt-2.1.0-py3-none-any.whl (67 kB)
|
||||||
|
#18 15.34 Downloading scapy-2.6.1-py3-none-any.whl (2.4 MB)
|
||||||
|
#18 15.62 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 2.4/2.4 MB 8.5 MB/s eta 0:00:00
|
||||||
|
#18 15.63 Downloading cron_converter-1.2.2-py3-none-any.whl (13 kB)
|
||||||
|
#18 15.67 Downloading pytz-2025.2-py2.py3-none-any.whl (509 kB)
|
||||||
|
#18 15.76 Downloading json2table-1.1.5-py2.py3-none-any.whl (8.7 kB)
|
||||||
|
#18 15.81 Downloading dhcp_leases-0.1.6-py3-none-any.whl (11 kB)
|
||||||
|
#18 15.86 Downloading pyunifi-2.21-py3-none-any.whl (11 kB)
|
||||||
|
#18 15.90 Downloading speedtest_cli-2.1.3-py2.py3-none-any.whl (23 kB)
|
||||||
|
#18 15.95 Downloading chardet-5.2.0-py3-none-any.whl (199 kB)
|
||||||
|
#18 16.01 Downloading dnspython-2.8.0-py3-none-any.whl (331 kB)
|
||||||
|
#18 16.10 Downloading librouteros-3.4.1-py3-none-any.whl (16 kB)
|
||||||
|
#18 16.14 Downloading aiohappyeyeballs-2.6.1-py3-none-any.whl (15 kB)
|
||||||
|
#18 16.20 Downloading aiosignal-1.4.0-py3-none-any.whl (7.5 kB)
|
||||||
|
#18 16.24 Downloading attrs-25.3.0-py3-none-any.whl (63 kB)
|
||||||
|
#18 16.30 Downloading awesomeversion-25.8.0-py3-none-any.whl (15 kB)
|
||||||
|
#18 16.34 Downloading blinker-1.9.0-py3-none-any.whl (8.5 kB)
|
||||||
|
#18 16.39 Downloading certifi-2025.8.3-py3-none-any.whl (161 kB)
|
||||||
|
#18 16.45 Downloading charset_normalizer-3.4.3-cp312-cp312-musllinux_1_2_x86_64.whl (153 kB)
|
||||||
|
#18 16.50 Downloading click-8.3.0-py3-none-any.whl (107 kB)
|
||||||
|
#18 16.55 Downloading frozenlist-1.7.0-cp312-cp312-musllinux_1_2_x86_64.whl (237 kB)
|
||||||
|
#18 16.62 Downloading graphql_core-3.2.6-py3-none-any.whl (203 kB)
|
||||||
|
#18 16.69 Downloading graphql_relay-3.2.0-py3-none-any.whl (16 kB)
|
||||||
|
#18 16.73 Downloading idna-3.10-py3-none-any.whl (70 kB)
|
||||||
|
#18 16.79 Downloading itsdangerous-2.2.0-py3-none-any.whl (16 kB)
|
||||||
|
#18 16.84 Downloading jinja2-3.1.6-py3-none-any.whl (134 kB)
|
||||||
|
#18 16.96 Downloading MarkupSafe-3.0.2-cp312-cp312-musllinux_1_2_x86_64.whl (23 kB)
|
||||||
|
#18 17.02 Downloading multidict-6.6.4-cp312-cp312-musllinux_1_2_x86_64.whl (251 kB)
|
||||||
|
#18 17.09 Downloading packaging-25.0-py3-none-any.whl (66 kB)
|
||||||
|
#18 17.14 Downloading propcache-0.3.2-cp312-cp312-musllinux_1_2_x86_64.whl (222 kB)
|
||||||
|
#18 17.21 Downloading python_dateutil-2.9.0.post0-py2.py3-none-any.whl (229 kB)
|
||||||
|
#18 17.28 Downloading toml-0.10.2-py2.py3-none-any.whl (16 kB)
|
||||||
|
#18 17.33 Downloading typing_extensions-4.15.0-py3-none-any.whl (44 kB)
|
||||||
|
#18 17.39 Downloading urllib3-2.5.0-py3-none-any.whl (129 kB)
|
||||||
|
#18 17.44 Downloading werkzeug-3.1.3-py3-none-any.whl (224 kB)
|
||||||
|
#18 17.51 Downloading xmltodict-1.0.2-py3-none-any.whl (13 kB)
|
||||||
|
#18 17.56 Downloading yarl-1.20.1-cp312-cp312-musllinux_1_2_x86_64.whl (374 kB)
|
||||||
|
#18 17.65 Downloading six-1.17.0-py2.py3-none-any.whl (11 kB)
|
||||||
|
#18 17.77 Building wheels for collected packages: python-nmap, yattag, aiofreepybox
|
||||||
|
#18 17.77 Building wheel for python-nmap (pyproject.toml): started
|
||||||
|
#18 17.95 Building wheel for python-nmap (pyproject.toml): finished with status 'done'
|
||||||
|
#18 17.96 Created wheel for python-nmap: filename=python_nmap-0.7.1-py2.py3-none-any.whl size=20679 sha256=ecd9b14109651cfaa5bf035f90076b9442985cc254fa5f8a49868fc896e86edb
|
||||||
|
#18 17.96 Stored in directory: /root/.cache/pip/wheels/06/fc/d4/0957e1d9942e696188208772ea0abf909fe6eb3d9dff6e5a9e
|
||||||
|
#18 17.96 Building wheel for yattag (pyproject.toml): started
|
||||||
|
#18 18.14 Building wheel for yattag (pyproject.toml): finished with status 'done'
|
||||||
|
#18 18.14 Created wheel for yattag: filename=yattag-1.16.1-py3-none-any.whl size=15930 sha256=2135fc2034a3847c81eb6a0d7b85608e8272339fa5c1961f87b02dfe6d74d0ad
|
||||||
|
#18 18.14 Stored in directory: /root/.cache/pip/wheels/d2/2f/52/049ff4f7c8c9c932b2ece7ec800d7facf2a141ac5ab0ce7e51
|
||||||
|
#18 18.15 Building wheel for aiofreepybox (pyproject.toml): started
|
||||||
|
#18 18.36 Building wheel for aiofreepybox (pyproject.toml): finished with status 'done'
|
||||||
|
#18 18.36 Created wheel for aiofreepybox: filename=aiofreepybox-6.0.0-py3-none-any.whl size=60051 sha256=dbdee5350b10b6550ede50bc779381b7f39f1e5d5da889f2ee98cb5a869d3425
|
||||||
|
#18 18.36 Stored in directory: /tmp/pip-ephem-wheel-cache-93bgc4e2/wheels/3c/d3/ae/fb97a84a29a5fbe8517de58d67e66586505440af35981e0dd3
|
||||||
|
#18 18.36 Successfully built python-nmap yattag aiofreepybox
|
||||||
|
#18 18.45 Installing collected packages: yattag, speedtest-cli, pytz, python-nmap, json2table, dhcp-leases, xmltodict, wakeonlan, urllib3, typing-extensions, toml, six, scapy, pycryptodome, propcache, paho-mqtt, packaging, multidict, markupsafe, itsdangerous, idna, graphql-core, frozenlist, dnspython, Click, charset_normalizer, chardet, certifi, blinker, awesomeversion, attrs, asyncio, aiohappyeyeballs, yarl, werkzeug, requests, python-dateutil, librouteros, jinja2, graphql-relay, aiosignal, unifi-sm-api, pyunifi, openwrt-luci-rpc, graphene, flask, cron-converter, aiohttp, tplink-omada-client, flask-cors, asusrouter, aiofreepybox
|
||||||
|
#18 24.35 Successfully installed Click-8.3.0 aiofreepybox-6.0.0 aiohappyeyeballs-2.6.1 aiohttp-3.12.15 aiosignal-1.4.0 asusrouter-1.21.0 asyncio-4.0.0 attrs-25.3.0 awesomeversion-25.8.0 blinker-1.9.0 certifi-2025.8.3 chardet-5.2.0 charset_normalizer-3.4.3 cron-converter-1.2.2 dhcp-leases-0.1.6 dnspython-2.8.0 flask-3.1.2 flask-cors-6.0.1 frozenlist-1.7.0 graphene-3.4.3 graphql-core-3.2.6 graphql-relay-3.2.0 idna-3.10 itsdangerous-2.2.0 jinja2-3.1.6 json2table-1.1.5 librouteros-3.4.1 markupsafe-3.0.2 multidict-6.6.4 openwrt-luci-rpc-1.1.17 packaging-25.0 paho-mqtt-2.1.0 propcache-0.3.2 pycryptodome-3.23.0 python-dateutil-2.9.0.post0 python-nmap-0.7.1 pytz-2025.2 pyunifi-2.21 requests-2.32.5 scapy-2.6.1 six-1.17.0 speedtest-cli-2.1.3 toml-0.10.2 tplink-omada-client-1.4.4 typing-extensions-4.15.0 unifi-sm-api-0.2.1 urllib3-2.5.0 wakeonlan-3.1.0 werkzeug-3.1.3 xmltodict-1.0.2 yarl-1.20.1 yattag-1.16.1
|
||||||
|
#18 24.47
|
||||||
|
#18 24.47 [notice] A new release of pip is available: 25.0.1 -> 25.2
|
||||||
|
#18 24.47 [notice] To update, run: pip install --upgrade pip
|
||||||
|
#18 DONE 25.1s
|
||||||
|
|
||||||
|
#19 [builder 14/15] RUN bash -c "find /app -type d -exec chmod 750 {} \;" && bash -c "find /app -type f -exec chmod 640 {} \;" && bash -c "find /app -type f \( -name '*.sh' -o -name '*.py' -o -name 'speedtest-cli' \) -exec chmod 750 {} \;"
|
||||||
|
#19 DONE 11.9s
|
||||||
|
|
||||||
|
#20 [builder 15/15] COPY install/freebox_certificate.pem /opt/venv/lib/python3.12/site-packages/aiofreepybox/freebox_certificates.pem
|
||||||
|
#20 DONE 0.4s
|
||||||
|
|
||||||
|
#21 [runner 2/14] COPY --from=builder /opt/venv /opt/venv
|
||||||
|
#21 DONE 0.8s
|
||||||
|
|
||||||
|
#22 [runner 3/14] COPY --from=builder /usr/sbin/usermod /usr/sbin/groupmod /usr/sbin/
|
||||||
|
#22 DONE 0.4s
|
||||||
|
|
||||||
|
#23 [runner 4/14] RUN apk update --no-cache && apk add --no-cache bash libbsd zip lsblk gettext-envsubst sudo mtr tzdata s6-overlay && apk add --no-cache curl arp-scan iproute2 iproute2-ss nmap nmap-scripts traceroute nbtscan avahi avahi-tools openrc dbus net-tools net-snmp-tools bind-tools awake ca-certificates && apk add --no-cache sqlite php83 php83-fpm php83-cgi php83-curl php83-sqlite3 php83-session && apk add --no-cache python3 nginx && ln -s /usr/bin/awake /usr/bin/wakeonlan && bash -c "install -d -m 750 -o nginx -g www-data /app /app" && rm -f /etc/nginx/http.d/default.conf
|
||||||
|
#23 0.487 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/main/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 0.696 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/community/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 1.156 v3.22.1-472-ga67443520d6 [https://dl-cdn.alpinelinux.org/alpine/v3.22/main]
|
||||||
|
#23 1.156 v3.22.1-473-gcd551a4e006 [https://dl-cdn.alpinelinux.org/alpine/v3.22/community]
|
||||||
|
#23 1.156 OK: 26326 distinct packages available
|
||||||
|
#23 1.195 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/main/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 1.276 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/community/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 1.568 (1/38) Installing ncurses-terminfo-base (6.5_p20250503-r0)
|
||||||
|
#23 1.580 (2/38) Installing libncursesw (6.5_p20250503-r0)
|
||||||
|
#23 1.629 (3/38) Installing readline (8.2.13-r1)
|
||||||
|
#23 1.659 (4/38) Installing bash (5.2.37-r0)
|
||||||
|
#23 1.723 Executing bash-5.2.37-r0.post-install
|
||||||
|
#23 1.740 (5/38) Installing libintl (0.24.1-r0)
|
||||||
|
#23 1.749 (6/38) Installing gettext-envsubst (0.24.1-r0)
|
||||||
|
#23 1.775 (7/38) Installing libmd (1.1.0-r0)
|
||||||
|
#23 1.782 (8/38) Installing libbsd (0.12.2-r0)
|
||||||
|
#23 1.807 (9/38) Installing libeconf (0.6.3-r0)
|
||||||
|
#23 1.812 (10/38) Installing libblkid (2.41-r9)
|
||||||
|
#23 1.831 (11/38) Installing libmount (2.41-r9)
|
||||||
|
#23 1.857 (12/38) Installing libsmartcols (2.41-r9)
|
||||||
|
#23 1.872 (13/38) Installing lsblk (2.41-r9)
|
||||||
|
#23 1.886 (14/38) Installing libcap2 (2.76-r0)
|
||||||
|
#23 1.897 (15/38) Installing jansson (2.14.1-r0)
|
||||||
|
#23 1.910 (16/38) Installing mtr (0.96-r0)
|
||||||
|
#23 1.948 (17/38) Installing skalibs-libs (2.14.4.0-r0)
|
||||||
|
#23 1.966 (18/38) Installing execline-libs (2.9.7.0-r0)
|
||||||
|
#23 1.974 (19/38) Installing execline (2.9.7.0-r0)
|
||||||
|
#23 1.996 Executing execline-2.9.7.0-r0.post-install
|
||||||
|
#23 2.004 (20/38) Installing s6-ipcserver (2.13.2.0-r0)
|
||||||
|
#23 2.010 (21/38) Installing s6-libs (2.13.2.0-r0)
|
||||||
|
#23 2.016 (22/38) Installing s6 (2.13.2.0-r0)
|
||||||
|
#23 2.033 Executing s6-2.13.2.0-r0.pre-install
|
||||||
|
#23 2.159 (23/38) Installing s6-rc-libs (0.5.6.0-r0)
|
||||||
|
#23 2.164 (24/38) Installing s6-rc (0.5.6.0-r0)
|
||||||
|
#23 2.175 (25/38) Installing s6-linux-init (1.1.3.0-r0)
|
||||||
|
#23 2.185 (26/38) Installing s6-portable-utils (2.3.1.0-r0)
|
||||||
|
#23 2.193 (27/38) Installing s6-linux-utils (2.6.3.0-r0)
|
||||||
|
#23 2.200 (28/38) Installing s6-dns-libs (2.4.1.0-r0)
|
||||||
|
#23 2.208 (29/38) Installing s6-dns (2.4.1.0-r0)
|
||||||
|
#23 2.222 (30/38) Installing bearssl-libs (0.6_git20241009-r0)
|
||||||
|
#23 2.254 (31/38) Installing s6-networking-libs (2.7.1.0-r0)
|
||||||
|
#23 2.264 (32/38) Installing s6-networking (2.7.1.0-r0)
|
||||||
|
#23 2.286 (33/38) Installing s6-overlay-helpers (0.1.2.0-r0)
|
||||||
|
#23 2.355 (34/38) Installing s6-overlay (3.2.0.3-r0)
|
||||||
|
#23 2.380 (35/38) Installing sudo (1.9.17_p2-r0)
|
||||||
|
#23 2.511 (36/38) Installing tzdata (2025b-r0)
|
||||||
|
#23 2.641 (37/38) Installing unzip (6.0-r15)
|
||||||
|
#23 2.659 (38/38) Installing zip (3.0-r13)
|
||||||
|
#23 2.694 Executing busybox-1.37.0-r18.trigger
|
||||||
|
#23 2.725 OK: 16 MiB in 54 packages
|
||||||
|
#23 2.778 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/main/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 2.918 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/community/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 3.218 (1/77) Installing libpcap (1.10.5-r1)
|
||||||
|
#23 3.234 (2/77) Installing arp-scan (1.10.0-r2)
|
||||||
|
#23 3.289 (3/77) Installing dbus-libs (1.16.2-r1)
|
||||||
|
#23 3.307 (4/77) Installing avahi-libs (0.8-r21)
|
||||||
|
#23 3.315 (5/77) Installing libdaemon (0.14-r6)
|
||||||
|
#23 3.322 (6/77) Installing libevent (2.1.12-r8)
|
||||||
|
#23 3.355 (7/77) Installing libexpat (2.7.2-r0)
|
||||||
|
#23 3.368 (8/77) Installing avahi (0.8-r21)
|
||||||
|
#23 3.387 Executing avahi-0.8-r21.pre-install
|
||||||
|
#23 3.465 (9/77) Installing gdbm (1.24-r0)
|
||||||
|
#23 3.477 (10/77) Installing avahi-tools (0.8-r21)
|
||||||
|
#23 3.483 (11/77) Installing libbz2 (1.0.8-r6)
|
||||||
|
#23 3.490 (12/77) Installing libffi (3.4.8-r0)
|
||||||
|
#23 3.496 (13/77) Installing xz-libs (5.8.1-r0)
|
||||||
|
#23 3.517 (14/77) Installing libgcc (14.2.0-r6)
|
||||||
|
#23 3.529 (15/77) Installing libstdc++ (14.2.0-r6)
|
||||||
|
#23 3.613 (16/77) Installing mpdecimal (4.0.1-r0)
|
||||||
|
#23 3.628 (17/77) Installing libpanelw (6.5_p20250503-r0)
|
||||||
|
#23 3.634 (18/77) Installing sqlite-libs (3.49.2-r1)
|
||||||
|
#23 3.783 (19/77) Installing python3 (3.12.11-r0)
|
||||||
|
#23 4.494 (20/77) Installing python3-pycache-pyc0 (3.12.11-r0)
|
||||||
|
#23 4.915 (21/77) Installing pyc (3.12.11-r0)
|
||||||
|
#23 4.915 (22/77) Installing py3-awake-pyc (1.0-r12)
|
||||||
|
#23 4.922 (23/77) Installing python3-pyc (3.12.11-r0)
|
||||||
|
#23 4.922 (24/77) Installing py3-awake (1.0-r12)
|
||||||
|
#23 4.928 (25/77) Installing awake (1.0-r12)
|
||||||
|
#23 4.932 (26/77) Installing fstrm (0.6.1-r4)
|
||||||
|
#23 4.940 (27/77) Installing krb5-conf (1.0-r2)
|
||||||
|
#23 5.017 (28/77) Installing libcom_err (1.47.2-r2)
|
||||||
|
#23 5.026 (29/77) Installing keyutils-libs (1.6.3-r4)
|
||||||
|
#23 5.033 (30/77) Installing libverto (0.3.2-r2)
|
||||||
|
#23 5.039 (31/77) Installing krb5-libs (1.21.3-r0)
|
||||||
|
#23 5.115 (32/77) Installing json-c (0.18-r1)
|
||||||
|
#23 5.123 (33/77) Installing nghttp2-libs (1.65.0-r0)
|
||||||
|
#23 5.136 (34/77) Installing protobuf-c (1.5.2-r0)
|
||||||
|
#23 5.142 (35/77) Installing userspace-rcu (0.15.2-r0)
|
||||||
|
#23 5.161 (36/77) Installing libuv (1.51.0-r0)
|
||||||
|
#23 5.178 (37/77) Installing libxml2 (2.13.8-r0)
|
||||||
|
#23 5.232 (38/77) Installing bind-libs (9.20.13-r0)
|
||||||
|
#23 5.355 (39/77) Installing bind-tools (9.20.13-r0)
|
||||||
|
#23 5.395 (40/77) Installing ca-certificates (20250619-r0)
|
||||||
|
#23 5.518 (41/77) Installing brotli-libs (1.1.0-r2)
|
||||||
|
#23 5.559 (42/77) Installing c-ares (1.34.5-r0)
|
||||||
|
#23 5.573 (43/77) Installing libunistring (1.3-r0)
|
||||||
|
#23 5.645 (44/77) Installing libidn2 (2.3.7-r0)
|
||||||
|
#23 5.664 (45/77) Installing libpsl (0.21.5-r3)
|
||||||
|
#23 5.676 (46/77) Installing zstd-libs (1.5.7-r0)
|
||||||
|
#23 5.720 (47/77) Installing libcurl (8.14.1-r1)
|
||||||
|
#23 5.753 (48/77) Installing curl (8.14.1-r1)
|
||||||
|
#23 5.778 (49/77) Installing dbus (1.16.2-r1)
|
||||||
|
#23 5.796 Executing dbus-1.16.2-r1.pre-install
|
||||||
|
#23 5.869 Executing dbus-1.16.2-r1.post-install
|
||||||
|
#23 5.887 (50/77) Installing dbus-daemon-launch-helper (1.16.2-r1)
|
||||||
|
#23 5.896 (51/77) Installing libelf (0.193-r0)
|
||||||
|
#23 5.908 (52/77) Installing libmnl (1.0.5-r2)
|
||||||
|
#23 5.915 (53/77) Installing iproute2-minimal (6.15.0-r0)
|
||||||
|
#23 5.954 (54/77) Installing libxtables (1.8.11-r1)
|
||||||
|
#23 5.963 (55/77) Installing iproute2-tc (6.15.0-r0)
|
||||||
|
#23 6.001 (56/77) Installing iproute2-ss (6.15.0-r0)
|
||||||
|
#23 6.014 (57/77) Installing iproute2 (6.15.0-r0)
|
||||||
|
#23 6.042 Executing iproute2-6.15.0-r0.post-install
|
||||||
|
#23 6.047 (58/77) Installing nbtscan (1.7.2-r0)
|
||||||
|
#23 6.053 (59/77) Installing net-snmp-libs (5.9.4-r1)
|
||||||
|
#23 6.112 (60/77) Installing net-snmp-agent-libs (5.9.4-r1)
|
||||||
|
#23 6.179 (61/77) Installing net-snmp-tools (5.9.4-r1)
|
||||||
|
#23 6.205 (62/77) Installing mii-tool (2.10-r3)
|
||||||
|
#23 6.211 (63/77) Installing net-tools (2.10-r3)
|
||||||
|
#23 6.235 (64/77) Installing lua5.4-libs (5.4.7-r0)
|
||||||
|
#23 6.258 (65/77) Installing libssh2 (1.11.1-r0)
|
||||||
|
#23 6.279 (66/77) Installing nmap (7.97-r0)
|
||||||
|
#23 6.524 (67/77) Installing nmap-nselibs (7.97-r0)
|
||||||
|
#23 6.729 (68/77) Installing nmap-scripts (7.97-r0)
|
||||||
|
#23 6.842 (69/77) Installing bridge (1.5-r5)
|
||||||
|
#23 6.904 (70/77) Installing ifupdown-ng (0.12.1-r7)
|
||||||
|
#23 6.915 (71/77) Installing ifupdown-ng-iproute2 (0.12.1-r7)
|
||||||
|
#23 6.920 (72/77) Installing openrc-user (0.62.6-r0)
|
||||||
|
#23 6.924 (73/77) Installing openrc (0.62.6-r0)
|
||||||
|
#23 7.013 Executing openrc-0.62.6-r0.post-install
|
||||||
|
#23 7.016 (74/77) Installing avahi-openrc (0.8-r21)
|
||||||
|
#23 7.021 (75/77) Installing dbus-openrc (1.16.2-r1)
|
||||||
|
#23 7.026 (76/77) Installing s6-openrc (2.13.2.0-r0)
|
||||||
|
#23 7.032 (77/77) Installing traceroute (2.1.6-r0)
|
||||||
|
#23 7.040 Executing busybox-1.37.0-r18.trigger
|
||||||
|
#23 7.042 Executing ca-certificates-20250619-r0.trigger
|
||||||
|
#23 7.101 Executing dbus-1.16.2-r1.trigger
|
||||||
|
#23 7.104 OK: 102 MiB in 131 packages
|
||||||
|
#23 7.156 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/main/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 7.243 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/community/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 7.543 (1/12) Installing php83-common (8.3.24-r0)
|
||||||
|
#23 7.551 (2/12) Installing argon2-libs (20190702-r5)
|
||||||
|
#23 7.557 (3/12) Installing libedit (20250104.3.1-r1)
|
||||||
|
#23 7.568 (4/12) Installing pcre2 (10.43-r1)
|
||||||
|
#23 7.600 (5/12) Installing php83 (8.3.24-r0)
|
||||||
|
#23 7.777 (6/12) Installing php83-cgi (8.3.24-r0)
|
||||||
|
#23 7.953 (7/12) Installing php83-curl (8.3.24-r0)
|
||||||
|
#23 7.968 (8/12) Installing acl-libs (2.3.2-r1)
|
||||||
|
#23 7.975 (9/12) Installing php83-fpm (8.3.24-r0)
|
||||||
|
#23 8.193 (10/12) Installing php83-session (8.3.24-r0)
|
||||||
|
#23 8.204 (11/12) Installing php83-sqlite3 (8.3.24-r0)
|
||||||
|
#23 8.213 (12/12) Installing sqlite (3.49.2-r1)
|
||||||
|
#23 8.309 Executing busybox-1.37.0-r18.trigger
|
||||||
|
#23 8.317 OK: 129 MiB in 143 packages
|
||||||
|
#23 8.369 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/main/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 8.449 fetch https://dl-cdn.alpinelinux.org/alpine/v3.22/community/x86_64/APKINDEX.tar.gz
|
||||||
|
#23 8.747 (1/2) Installing nginx (1.28.0-r3)
|
||||||
|
#23 8.766 Executing nginx-1.28.0-r3.pre-install
|
||||||
|
#23 8.863 Executing nginx-1.28.0-r3.post-install
|
||||||
|
#23 8.865 (2/2) Installing nginx-openrc (1.28.0-r3)
|
||||||
|
#23 8.870 Executing busybox-1.37.0-r18.trigger
|
||||||
|
#23 8.873 OK: 130 MiB in 145 packages
|
||||||
|
#23 DONE 9.5s
|
||||||
|
|
||||||
|
#24 [runner 5/14] COPY --from=builder --chown=nginx:www-data /app/ /app/
|
||||||
|
#24 DONE 0.5s
|
||||||
|
|
||||||
|
#25 [runner 6/14] RUN mkdir -p /app/config /app/db /app/log/plugins
|
||||||
|
#25 DONE 0.5s
|
||||||
|
|
||||||
|
#26 [runner 7/14] COPY --chmod=600 --chown=root:root install/crontab /etc/crontabs/root
|
||||||
|
#26 DONE 0.3s
|
||||||
|
|
||||||
|
#27 [runner 8/14] COPY --chmod=755 dockerfiles/healthcheck.sh /usr/local/bin/healthcheck.sh
|
||||||
|
#27 DONE 0.3s
|
||||||
|
|
||||||
|
#28 [runner 9/14] RUN touch /app/log/app.log && touch /app/log/execution_queue.log && touch /app/log/app_front.log && touch /app/log/app.php_errors.log && touch /app/log/stderr.log && touch /app/log/stdout.log && touch /app/log/db_is_locked.log && touch /app/log/IP_changes.log && touch /app/log/report_output.txt && touch /app/log/report_output.html && touch /app/log/report_output.json && touch /app/api/user_notifications.json
|
||||||
|
#28 DONE 0.6s
|
||||||
|
|
||||||
|
#29 [runner 10/14] COPY dockerfiles /app/dockerfiles
|
||||||
|
#29 DONE 0.3s
|
||||||
|
|
||||||
|
#30 [runner 11/14] RUN chmod +x /app/dockerfiles/*.sh
|
||||||
|
#30 DONE 0.8s
|
||||||
|
|
||||||
|
#31 [runner 12/14] RUN /app/dockerfiles/init-nginx.sh && /app/dockerfiles/init-php-fpm.sh && /app/dockerfiles/init-crond.sh && /app/dockerfiles/init-backend.sh
|
||||||
|
#31 0.417 Initializing nginx...
|
||||||
|
#31 0.417 Setting webserver to address (0.0.0.0) and port (20211)
|
||||||
|
#31 0.418 /app/dockerfiles/init-nginx.sh: line 5: /app/install/netalertx.template.conf: No such file or directory
|
||||||
|
#31 0.611 nginx initialized.
|
||||||
|
#31 0.612 Initializing php-fpm...
|
||||||
|
#31 0.654 php-fpm initialized.
|
||||||
|
#31 0.655 Initializing crond...
|
||||||
|
#31 0.689 crond initialized.
|
||||||
|
#31 0.690 Initializing backend...
|
||||||
|
#31 12.19 Backend initialized.
|
||||||
|
#31 DONE 12.3s
|
||||||
|
|
||||||
|
#32 [runner 13/14] RUN rm -rf /app/dockerfiles
|
||||||
|
#32 DONE 0.6s
|
||||||
|
|
||||||
|
#33 [runner 14/14] RUN date +%s > /app/front/buildtimestamp.txt
|
||||||
|
#33 DONE 0.6s
|
||||||
|
|
||||||
|
#34 exporting to image
|
||||||
|
#34 exporting layers
|
||||||
|
#34 exporting layers 2.4s done
|
||||||
|
#34 writing image sha256:0afcbc41473de559eff0dd93250595494fe4d8ea620861e9e90d50a248fcefda 0.0s done
|
||||||
|
#34 naming to docker.io/library/netalertx 0.0s done
|
||||||
|
#34 DONE 2.5s
|
||||||
@@ -1,674 +0,0 @@
|
|||||||
GNU GENERAL PUBLIC LICENSE
|
|
||||||
Version 3, 29 June 2007
|
|
||||||
|
|
||||||
Copyright (C) 2007 Free Software Foundation, Inc. <https://fsf.org/>
|
|
||||||
Everyone is permitted to copy and distribute verbatim copies
|
|
||||||
of this license document, but changing it is not allowed.
|
|
||||||
|
|
||||||
Preamble
|
|
||||||
|
|
||||||
The GNU General Public License is a free, copyleft license for
|
|
||||||
software and other kinds of works.
|
|
||||||
|
|
||||||
The licenses for most software and other practical works are designed
|
|
||||||
to take away your freedom to share and change the works. By contrast,
|
|
||||||
the GNU General Public License is intended to guarantee your freedom to
|
|
||||||
share and change all versions of a program--to make sure it remains free
|
|
||||||
software for all its users. We, the Free Software Foundation, use the
|
|
||||||
GNU General Public License for most of our software; it applies also to
|
|
||||||
any other work released this way by its authors. You can apply it to
|
|
||||||
your programs, too.
|
|
||||||
|
|
||||||
When we speak of free software, we are referring to freedom, not
|
|
||||||
price. Our General Public Licenses are designed to make sure that you
|
|
||||||
have the freedom to distribute copies of free software (and charge for
|
|
||||||
them if you wish), that you receive source code or can get it if you
|
|
||||||
want it, that you can change the software or use pieces of it in new
|
|
||||||
free programs, and that you know you can do these things.
|
|
||||||
|
|
||||||
To protect your rights, we need to prevent others from denying you
|
|
||||||
these rights or asking you to surrender the rights. Therefore, you have
|
|
||||||
certain responsibilities if you distribute copies of the software, or if
|
|
||||||
you modify it: responsibilities to respect the freedom of others.
|
|
||||||
|
|
||||||
For example, if you distribute copies of such a program, whether
|
|
||||||
gratis or for a fee, you must pass on to the recipients the same
|
|
||||||
freedoms that you received. You must make sure that they, too, receive
|
|
||||||
or can get the source code. And you must show them these terms so they
|
|
||||||
know their rights.
|
|
||||||
|
|
||||||
Developers that use the GNU GPL protect your rights with two steps:
|
|
||||||
(1) assert copyright on the software, and (2) offer you this License
|
|
||||||
giving you legal permission to copy, distribute and/or modify it.
|
|
||||||
|
|
||||||
For the developers' and authors' protection, the GPL clearly explains
|
|
||||||
that there is no warranty for this free software. For both users' and
|
|
||||||
authors' sake, the GPL requires that modified versions be marked as
|
|
||||||
changed, so that their problems will not be attributed erroneously to
|
|
||||||
authors of previous versions.
|
|
||||||
|
|
||||||
Some devices are designed to deny users access to install or run
|
|
||||||
modified versions of the software inside them, although the manufacturer
|
|
||||||
can do so. This is fundamentally incompatible with the aim of
|
|
||||||
protecting users' freedom to change the software. The systematic
|
|
||||||
pattern of such abuse occurs in the area of products for individuals to
|
|
||||||
use, which is precisely where it is most unacceptable. Therefore, we
|
|
||||||
have designed this version of the GPL to prohibit the practice for those
|
|
||||||
products. If such problems arise substantially in other domains, we
|
|
||||||
stand ready to extend this provision to those domains in future versions
|
|
||||||
of the GPL, as needed to protect the freedom of users.
|
|
||||||
|
|
||||||
Finally, every program is threatened constantly by software patents.
|
|
||||||
States should not allow patents to restrict development and use of
|
|
||||||
software on general-purpose computers, but in those that do, we wish to
|
|
||||||
avoid the special danger that patents applied to a free program could
|
|
||||||
make it effectively proprietary. To prevent this, the GPL assures that
|
|
||||||
patents cannot be used to render the program non-free.
|
|
||||||
|
|
||||||
The precise terms and conditions for copying, distribution and
|
|
||||||
modification follow.
|
|
||||||
|
|
||||||
TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
0. Definitions.
|
|
||||||
|
|
||||||
"This License" refers to version 3 of the GNU General Public License.
|
|
||||||
|
|
||||||
"Copyright" also means copyright-like laws that apply to other kinds of
|
|
||||||
works, such as semiconductor masks.
|
|
||||||
|
|
||||||
"The Program" refers to any copyrightable work licensed under this
|
|
||||||
License. Each licensee is addressed as "you". "Licensees" and
|
|
||||||
"recipients" may be individuals or organizations.
|
|
||||||
|
|
||||||
To "modify" a work means to copy from or adapt all or part of the work
|
|
||||||
in a fashion requiring copyright permission, other than the making of an
|
|
||||||
exact copy. The resulting work is called a "modified version" of the
|
|
||||||
earlier work or a work "based on" the earlier work.
|
|
||||||
|
|
||||||
A "covered work" means either the unmodified Program or a work based
|
|
||||||
on the Program.
|
|
||||||
|
|
||||||
To "propagate" a work means to do anything with it that, without
|
|
||||||
permission, would make you directly or secondarily liable for
|
|
||||||
infringement under applicable copyright law, except executing it on a
|
|
||||||
computer or modifying a private copy. Propagation includes copying,
|
|
||||||
distribution (with or without modification), making available to the
|
|
||||||
public, and in some countries other activities as well.
|
|
||||||
|
|
||||||
To "convey" a work means any kind of propagation that enables other
|
|
||||||
parties to make or receive copies. Mere interaction with a user through
|
|
||||||
a computer network, with no transfer of a copy, is not conveying.
|
|
||||||
|
|
||||||
An interactive user interface displays "Appropriate Legal Notices"
|
|
||||||
to the extent that it includes a convenient and prominently visible
|
|
||||||
feature that (1) displays an appropriate copyright notice, and (2)
|
|
||||||
tells the user that there is no warranty for the work (except to the
|
|
||||||
extent that warranties are provided), that licensees may convey the
|
|
||||||
work under this License, and how to view a copy of this License. If
|
|
||||||
the interface presents a list of user commands or options, such as a
|
|
||||||
menu, a prominent item in the list meets this criterion.
|
|
||||||
|
|
||||||
1. Source Code.
|
|
||||||
|
|
||||||
The "source code" for a work means the preferred form of the work
|
|
||||||
for making modifications to it. "Object code" means any non-source
|
|
||||||
form of a work.
|
|
||||||
|
|
||||||
A "Standard Interface" means an interface that either is an official
|
|
||||||
standard defined by a recognized standards body, or, in the case of
|
|
||||||
interfaces specified for a particular programming language, one that
|
|
||||||
is widely used among developers working in that language.
|
|
||||||
|
|
||||||
The "System Libraries" of an executable work include anything, other
|
|
||||||
than the work as a whole, that (a) is included in the normal form of
|
|
||||||
packaging a Major Component, but which is not part of that Major
|
|
||||||
Component, and (b) serves only to enable use of the work with that
|
|
||||||
Major Component, or to implement a Standard Interface for which an
|
|
||||||
implementation is available to the public in source code form. A
|
|
||||||
"Major Component", in this context, means a major essential component
|
|
||||||
(kernel, window system, and so on) of the specific operating system
|
|
||||||
(if any) on which the executable work runs, or a compiler used to
|
|
||||||
produce the work, or an object code interpreter used to run it.
|
|
||||||
|
|
||||||
The "Corresponding Source" for a work in object code form means all
|
|
||||||
the source code needed to generate, install, and (for an executable
|
|
||||||
work) run the object code and to modify the work, including scripts to
|
|
||||||
control those activities. However, it does not include the work's
|
|
||||||
System Libraries, or general-purpose tools or generally available free
|
|
||||||
programs which are used unmodified in performing those activities but
|
|
||||||
which are not part of the work. For example, Corresponding Source
|
|
||||||
includes interface definition files associated with source files for
|
|
||||||
the work, and the source code for shared libraries and dynamically
|
|
||||||
linked subprograms that the work is specifically designed to require,
|
|
||||||
such as by intimate data communication or control flow between those
|
|
||||||
subprograms and other parts of the work.
|
|
||||||
|
|
||||||
The Corresponding Source need not include anything that users
|
|
||||||
can regenerate automatically from other parts of the Corresponding
|
|
||||||
Source.
|
|
||||||
|
|
||||||
The Corresponding Source for a work in source code form is that
|
|
||||||
same work.
|
|
||||||
|
|
||||||
2. Basic Permissions.
|
|
||||||
|
|
||||||
All rights granted under this License are granted for the term of
|
|
||||||
copyright on the Program, and are irrevocable provided the stated
|
|
||||||
conditions are met. This License explicitly affirms your unlimited
|
|
||||||
permission to run the unmodified Program. The output from running a
|
|
||||||
covered work is covered by this License only if the output, given its
|
|
||||||
content, constitutes a covered work. This License acknowledges your
|
|
||||||
rights of fair use or other equivalent, as provided by copyright law.
|
|
||||||
|
|
||||||
You may make, run and propagate covered works that you do not
|
|
||||||
convey, without conditions so long as your license otherwise remains
|
|
||||||
in force. You may convey covered works to others for the sole purpose
|
|
||||||
of having them make modifications exclusively for you, or provide you
|
|
||||||
with facilities for running those works, provided that you comply with
|
|
||||||
the terms of this License in conveying all material for which you do
|
|
||||||
not control copyright. Those thus making or running the covered works
|
|
||||||
for you must do so exclusively on your behalf, under your direction
|
|
||||||
and control, on terms that prohibit them from making any copies of
|
|
||||||
your copyrighted material outside their relationship with you.
|
|
||||||
|
|
||||||
Conveying under any other circumstances is permitted solely under
|
|
||||||
the conditions stated below. Sublicensing is not allowed; section 10
|
|
||||||
makes it unnecessary.
|
|
||||||
|
|
||||||
3. Protecting Users' Legal Rights From Anti-Circumvention Law.
|
|
||||||
|
|
||||||
No covered work shall be deemed part of an effective technological
|
|
||||||
measure under any applicable law fulfilling obligations under article
|
|
||||||
11 of the WIPO copyright treaty adopted on 20 December 1996, or
|
|
||||||
similar laws prohibiting or restricting circumvention of such
|
|
||||||
measures.
|
|
||||||
|
|
||||||
When you convey a covered work, you waive any legal power to forbid
|
|
||||||
circumvention of technological measures to the extent such circumvention
|
|
||||||
is effected by exercising rights under this License with respect to
|
|
||||||
the covered work, and you disclaim any intention to limit operation or
|
|
||||||
modification of the work as a means of enforcing, against the work's
|
|
||||||
users, your or third parties' legal rights to forbid circumvention of
|
|
||||||
technological measures.
|
|
||||||
|
|
||||||
4. Conveying Verbatim Copies.
|
|
||||||
|
|
||||||
You may convey verbatim copies of the Program's source code as you
|
|
||||||
receive it, in any medium, provided that you conspicuously and
|
|
||||||
appropriately publish on each copy an appropriate copyright notice;
|
|
||||||
keep intact all notices stating that this License and any
|
|
||||||
non-permissive terms added in accord with section 7 apply to the code;
|
|
||||||
keep intact all notices of the absence of any warranty; and give all
|
|
||||||
recipients a copy of this License along with the Program.
|
|
||||||
|
|
||||||
You may charge any price or no price for each copy that you convey,
|
|
||||||
and you may offer support or warranty protection for a fee.
|
|
||||||
|
|
||||||
5. Conveying Modified Source Versions.
|
|
||||||
|
|
||||||
You may convey a work based on the Program, or the modifications to
|
|
||||||
produce it from the Program, in the form of source code under the
|
|
||||||
terms of section 4, provided that you also meet all of these conditions:
|
|
||||||
|
|
||||||
a) The work must carry prominent notices stating that you modified
|
|
||||||
it, and giving a relevant date.
|
|
||||||
|
|
||||||
b) The work must carry prominent notices stating that it is
|
|
||||||
released under this License and any conditions added under section
|
|
||||||
7. This requirement modifies the requirement in section 4 to
|
|
||||||
"keep intact all notices".
|
|
||||||
|
|
||||||
c) You must license the entire work, as a whole, under this
|
|
||||||
License to anyone who comes into possession of a copy. This
|
|
||||||
License will therefore apply, along with any applicable section 7
|
|
||||||
additional terms, to the whole of the work, and all its parts,
|
|
||||||
regardless of how they are packaged. This License gives no
|
|
||||||
permission to license the work in any other way, but it does not
|
|
||||||
invalidate such permission if you have separately received it.
|
|
||||||
|
|
||||||
d) If the work has interactive user interfaces, each must display
|
|
||||||
Appropriate Legal Notices; however, if the Program has interactive
|
|
||||||
interfaces that do not display Appropriate Legal Notices, your
|
|
||||||
work need not make them do so.
|
|
||||||
|
|
||||||
A compilation of a covered work with other separate and independent
|
|
||||||
works, which are not by their nature extensions of the covered work,
|
|
||||||
and which are not combined with it such as to form a larger program,
|
|
||||||
in or on a volume of a storage or distribution medium, is called an
|
|
||||||
"aggregate" if the compilation and its resulting copyright are not
|
|
||||||
used to limit the access or legal rights of the compilation's users
|
|
||||||
beyond what the individual works permit. Inclusion of a covered work
|
|
||||||
in an aggregate does not cause this License to apply to the other
|
|
||||||
parts of the aggregate.
|
|
||||||
|
|
||||||
6. Conveying Non-Source Forms.
|
|
||||||
|
|
||||||
You may convey a covered work in object code form under the terms
|
|
||||||
of sections 4 and 5, provided that you also convey the
|
|
||||||
machine-readable Corresponding Source under the terms of this License,
|
|
||||||
in one of these ways:
|
|
||||||
|
|
||||||
a) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by the
|
|
||||||
Corresponding Source fixed on a durable physical medium
|
|
||||||
customarily used for software interchange.
|
|
||||||
|
|
||||||
b) Convey the object code in, or embodied in, a physical product
|
|
||||||
(including a physical distribution medium), accompanied by a
|
|
||||||
written offer, valid for at least three years and valid for as
|
|
||||||
long as you offer spare parts or customer support for that product
|
|
||||||
model, to give anyone who possesses the object code either (1) a
|
|
||||||
copy of the Corresponding Source for all the software in the
|
|
||||||
product that is covered by this License, on a durable physical
|
|
||||||
medium customarily used for software interchange, for a price no
|
|
||||||
more than your reasonable cost of physically performing this
|
|
||||||
conveying of source, or (2) access to copy the
|
|
||||||
Corresponding Source from a network server at no charge.
|
|
||||||
|
|
||||||
c) Convey individual copies of the object code with a copy of the
|
|
||||||
written offer to provide the Corresponding Source. This
|
|
||||||
alternative is allowed only occasionally and noncommercially, and
|
|
||||||
only if you received the object code with such an offer, in accord
|
|
||||||
with subsection 6b.
|
|
||||||
|
|
||||||
d) Convey the object code by offering access from a designated
|
|
||||||
place (gratis or for a charge), and offer equivalent access to the
|
|
||||||
Corresponding Source in the same way through the same place at no
|
|
||||||
further charge. You need not require recipients to copy the
|
|
||||||
Corresponding Source along with the object code. If the place to
|
|
||||||
copy the object code is a network server, the Corresponding Source
|
|
||||||
may be on a different server (operated by you or a third party)
|
|
||||||
that supports equivalent copying facilities, provided you maintain
|
|
||||||
clear directions next to the object code saying where to find the
|
|
||||||
Corresponding Source. Regardless of what server hosts the
|
|
||||||
Corresponding Source, you remain obligated to ensure that it is
|
|
||||||
available for as long as needed to satisfy these requirements.
|
|
||||||
|
|
||||||
e) Convey the object code using peer-to-peer transmission, provided
|
|
||||||
you inform other peers where the object code and Corresponding
|
|
||||||
Source of the work are being offered to the general public at no
|
|
||||||
charge under subsection 6d.
|
|
||||||
|
|
||||||
A separable portion of the object code, whose source code is excluded
|
|
||||||
from the Corresponding Source as a System Library, need not be
|
|
||||||
included in conveying the object code work.
|
|
||||||
|
|
||||||
A "User Product" is either (1) a "consumer product", which means any
|
|
||||||
tangible personal property which is normally used for personal, family,
|
|
||||||
or household purposes, or (2) anything designed or sold for incorporation
|
|
||||||
into a dwelling. In determining whether a product is a consumer product,
|
|
||||||
doubtful cases shall be resolved in favor of coverage. For a particular
|
|
||||||
product received by a particular user, "normally used" refers to a
|
|
||||||
typical or common use of that class of product, regardless of the status
|
|
||||||
of the particular user or of the way in which the particular user
|
|
||||||
actually uses, or expects or is expected to use, the product. A product
|
|
||||||
is a consumer product regardless of whether the product has substantial
|
|
||||||
commercial, industrial or non-consumer uses, unless such uses represent
|
|
||||||
the only significant mode of use of the product.
|
|
||||||
|
|
||||||
"Installation Information" for a User Product means any methods,
|
|
||||||
procedures, authorization keys, or other information required to install
|
|
||||||
and execute modified versions of a covered work in that User Product from
|
|
||||||
a modified version of its Corresponding Source. The information must
|
|
||||||
suffice to ensure that the continued functioning of the modified object
|
|
||||||
code is in no case prevented or interfered with solely because
|
|
||||||
modification has been made.
|
|
||||||
|
|
||||||
If you convey an object code work under this section in, or with, or
|
|
||||||
specifically for use in, a User Product, and the conveying occurs as
|
|
||||||
part of a transaction in which the right of possession and use of the
|
|
||||||
User Product is transferred to the recipient in perpetuity or for a
|
|
||||||
fixed term (regardless of how the transaction is characterized), the
|
|
||||||
Corresponding Source conveyed under this section must be accompanied
|
|
||||||
by the Installation Information. But this requirement does not apply
|
|
||||||
if neither you nor any third party retains the ability to install
|
|
||||||
modified object code on the User Product (for example, the work has
|
|
||||||
been installed in ROM).
|
|
||||||
|
|
||||||
The requirement to provide Installation Information does not include a
|
|
||||||
requirement to continue to provide support service, warranty, or updates
|
|
||||||
for a work that has been modified or installed by the recipient, or for
|
|
||||||
the User Product in which it has been modified or installed. Access to a
|
|
||||||
network may be denied when the modification itself materially and
|
|
||||||
adversely affects the operation of the network or violates the rules and
|
|
||||||
protocols for communication across the network.
|
|
||||||
|
|
||||||
Corresponding Source conveyed, and Installation Information provided,
|
|
||||||
in accord with this section must be in a format that is publicly
|
|
||||||
documented (and with an implementation available to the public in
|
|
||||||
source code form), and must require no special password or key for
|
|
||||||
unpacking, reading or copying.
|
|
||||||
|
|
||||||
7. Additional Terms.
|
|
||||||
|
|
||||||
"Additional permissions" are terms that supplement the terms of this
|
|
||||||
License by making exceptions from one or more of its conditions.
|
|
||||||
Additional permissions that are applicable to the entire Program shall
|
|
||||||
be treated as though they were included in this License, to the extent
|
|
||||||
that they are valid under applicable law. If additional permissions
|
|
||||||
apply only to part of the Program, that part may be used separately
|
|
||||||
under those permissions, but the entire Program remains governed by
|
|
||||||
this License without regard to the additional permissions.
|
|
||||||
|
|
||||||
When you convey a copy of a covered work, you may at your option
|
|
||||||
remove any additional permissions from that copy, or from any part of
|
|
||||||
it. (Additional permissions may be written to require their own
|
|
||||||
removal in certain cases when you modify the work.) You may place
|
|
||||||
additional permissions on material, added by you to a covered work,
|
|
||||||
for which you have or can give appropriate copyright permission.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, for material you
|
|
||||||
add to a covered work, you may (if authorized by the copyright holders of
|
|
||||||
that material) supplement the terms of this License with terms:
|
|
||||||
|
|
||||||
a) Disclaiming warranty or limiting liability differently from the
|
|
||||||
terms of sections 15 and 16 of this License; or
|
|
||||||
|
|
||||||
b) Requiring preservation of specified reasonable legal notices or
|
|
||||||
author attributions in that material or in the Appropriate Legal
|
|
||||||
Notices displayed by works containing it; or
|
|
||||||
|
|
||||||
c) Prohibiting misrepresentation of the origin of that material, or
|
|
||||||
requiring that modified versions of such material be marked in
|
|
||||||
reasonable ways as different from the original version; or
|
|
||||||
|
|
||||||
d) Limiting the use for publicity purposes of names of licensors or
|
|
||||||
authors of the material; or
|
|
||||||
|
|
||||||
e) Declining to grant rights under trademark law for use of some
|
|
||||||
trade names, trademarks, or service marks; or
|
|
||||||
|
|
||||||
f) Requiring indemnification of licensors and authors of that
|
|
||||||
material by anyone who conveys the material (or modified versions of
|
|
||||||
it) with contractual assumptions of liability to the recipient, for
|
|
||||||
any liability that these contractual assumptions directly impose on
|
|
||||||
those licensors and authors.
|
|
||||||
|
|
||||||
All other non-permissive additional terms are considered "further
|
|
||||||
restrictions" within the meaning of section 10. If the Program as you
|
|
||||||
received it, or any part of it, contains a notice stating that it is
|
|
||||||
governed by this License along with a term that is a further
|
|
||||||
restriction, you may remove that term. If a license document contains
|
|
||||||
a further restriction but permits relicensing or conveying under this
|
|
||||||
License, you may add to a covered work material governed by the terms
|
|
||||||
of that license document, provided that the further restriction does
|
|
||||||
not survive such relicensing or conveying.
|
|
||||||
|
|
||||||
If you add terms to a covered work in accord with this section, you
|
|
||||||
must place, in the relevant source files, a statement of the
|
|
||||||
additional terms that apply to those files, or a notice indicating
|
|
||||||
where to find the applicable terms.
|
|
||||||
|
|
||||||
Additional terms, permissive or non-permissive, may be stated in the
|
|
||||||
form of a separately written license, or stated as exceptions;
|
|
||||||
the above requirements apply either way.
|
|
||||||
|
|
||||||
8. Termination.
|
|
||||||
|
|
||||||
You may not propagate or modify a covered work except as expressly
|
|
||||||
provided under this License. Any attempt otherwise to propagate or
|
|
||||||
modify it is void, and will automatically terminate your rights under
|
|
||||||
this License (including any patent licenses granted under the third
|
|
||||||
paragraph of section 11).
|
|
||||||
|
|
||||||
However, if you cease all violation of this License, then your
|
|
||||||
license from a particular copyright holder is reinstated (a)
|
|
||||||
provisionally, unless and until the copyright holder explicitly and
|
|
||||||
finally terminates your license, and (b) permanently, if the copyright
|
|
||||||
holder fails to notify you of the violation by some reasonable means
|
|
||||||
prior to 60 days after the cessation.
|
|
||||||
|
|
||||||
Moreover, your license from a particular copyright holder is
|
|
||||||
reinstated permanently if the copyright holder notifies you of the
|
|
||||||
violation by some reasonable means, this is the first time you have
|
|
||||||
received notice of violation of this License (for any work) from that
|
|
||||||
copyright holder, and you cure the violation prior to 30 days after
|
|
||||||
your receipt of the notice.
|
|
||||||
|
|
||||||
Termination of your rights under this section does not terminate the
|
|
||||||
licenses of parties who have received copies or rights from you under
|
|
||||||
this License. If your rights have been terminated and not permanently
|
|
||||||
reinstated, you do not qualify to receive new licenses for the same
|
|
||||||
material under section 10.
|
|
||||||
|
|
||||||
9. Acceptance Not Required for Having Copies.
|
|
||||||
|
|
||||||
You are not required to accept this License in order to receive or
|
|
||||||
run a copy of the Program. Ancillary propagation of a covered work
|
|
||||||
occurring solely as a consequence of using peer-to-peer transmission
|
|
||||||
to receive a copy likewise does not require acceptance. However,
|
|
||||||
nothing other than this License grants you permission to propagate or
|
|
||||||
modify any covered work. These actions infringe copyright if you do
|
|
||||||
not accept this License. Therefore, by modifying or propagating a
|
|
||||||
covered work, you indicate your acceptance of this License to do so.
|
|
||||||
|
|
||||||
10. Automatic Licensing of Downstream Recipients.
|
|
||||||
|
|
||||||
Each time you convey a covered work, the recipient automatically
|
|
||||||
receives a license from the original licensors, to run, modify and
|
|
||||||
propagate that work, subject to this License. You are not responsible
|
|
||||||
for enforcing compliance by third parties with this License.
|
|
||||||
|
|
||||||
An "entity transaction" is a transaction transferring control of an
|
|
||||||
organization, or substantially all assets of one, or subdividing an
|
|
||||||
organization, or merging organizations. If propagation of a covered
|
|
||||||
work results from an entity transaction, each party to that
|
|
||||||
transaction who receives a copy of the work also receives whatever
|
|
||||||
licenses to the work the party's predecessor in interest had or could
|
|
||||||
give under the previous paragraph, plus a right to possession of the
|
|
||||||
Corresponding Source of the work from the predecessor in interest, if
|
|
||||||
the predecessor has it or can get it with reasonable efforts.
|
|
||||||
|
|
||||||
You may not impose any further restrictions on the exercise of the
|
|
||||||
rights granted or affirmed under this License. For example, you may
|
|
||||||
not impose a license fee, royalty, or other charge for exercise of
|
|
||||||
rights granted under this License, and you may not initiate litigation
|
|
||||||
(including a cross-claim or counterclaim in a lawsuit) alleging that
|
|
||||||
any patent claim is infringed by making, using, selling, offering for
|
|
||||||
sale, or importing the Program or any portion of it.
|
|
||||||
|
|
||||||
11. Patents.
|
|
||||||
|
|
||||||
A "contributor" is a copyright holder who authorizes use under this
|
|
||||||
License of the Program or a work on which the Program is based. The
|
|
||||||
work thus licensed is called the contributor's "contributor version".
|
|
||||||
|
|
||||||
A contributor's "essential patent claims" are all patent claims
|
|
||||||
owned or controlled by the contributor, whether already acquired or
|
|
||||||
hereafter acquired, that would be infringed by some manner, permitted
|
|
||||||
by this License, of making, using, or selling its contributor version,
|
|
||||||
but do not include claims that would be infringed only as a
|
|
||||||
consequence of further modification of the contributor version. For
|
|
||||||
purposes of this definition, "control" includes the right to grant
|
|
||||||
patent sublicenses in a manner consistent with the requirements of
|
|
||||||
this License.
|
|
||||||
|
|
||||||
Each contributor grants you a non-exclusive, worldwide, royalty-free
|
|
||||||
patent license under the contributor's essential patent claims, to
|
|
||||||
make, use, sell, offer for sale, import and otherwise run, modify and
|
|
||||||
propagate the contents of its contributor version.
|
|
||||||
|
|
||||||
In the following three paragraphs, a "patent license" is any express
|
|
||||||
agreement or commitment, however denominated, not to enforce a patent
|
|
||||||
(such as an express permission to practice a patent or covenant not to
|
|
||||||
sue for patent infringement). To "grant" such a patent license to a
|
|
||||||
party means to make such an agreement or commitment not to enforce a
|
|
||||||
patent against the party.
|
|
||||||
|
|
||||||
If you convey a covered work, knowingly relying on a patent license,
|
|
||||||
and the Corresponding Source of the work is not available for anyone
|
|
||||||
to copy, free of charge and under the terms of this License, through a
|
|
||||||
publicly available network server or other readily accessible means,
|
|
||||||
then you must either (1) cause the Corresponding Source to be so
|
|
||||||
available, or (2) arrange to deprive yourself of the benefit of the
|
|
||||||
patent license for this particular work, or (3) arrange, in a manner
|
|
||||||
consistent with the requirements of this License, to extend the patent
|
|
||||||
license to downstream recipients. "Knowingly relying" means you have
|
|
||||||
actual knowledge that, but for the patent license, your conveying the
|
|
||||||
covered work in a country, or your recipient's use of the covered work
|
|
||||||
in a country, would infringe one or more identifiable patents in that
|
|
||||||
country that you have reason to believe are valid.
|
|
||||||
|
|
||||||
If, pursuant to or in connection with a single transaction or
|
|
||||||
arrangement, you convey, or propagate by procuring conveyance of, a
|
|
||||||
covered work, and grant a patent license to some of the parties
|
|
||||||
receiving the covered work authorizing them to use, propagate, modify
|
|
||||||
or convey a specific copy of the covered work, then the patent license
|
|
||||||
you grant is automatically extended to all recipients of the covered
|
|
||||||
work and works based on it.
|
|
||||||
|
|
||||||
A patent license is "discriminatory" if it does not include within
|
|
||||||
the scope of its coverage, prohibits the exercise of, or is
|
|
||||||
conditioned on the non-exercise of one or more of the rights that are
|
|
||||||
specifically granted under this License. You may not convey a covered
|
|
||||||
work if you are a party to an arrangement with a third party that is
|
|
||||||
in the business of distributing software, under which you make payment
|
|
||||||
to the third party based on the extent of your activity of conveying
|
|
||||||
the work, and under which the third party grants, to any of the
|
|
||||||
parties who would receive the covered work from you, a discriminatory
|
|
||||||
patent license (a) in connection with copies of the covered work
|
|
||||||
conveyed by you (or copies made from those copies), or (b) primarily
|
|
||||||
for and in connection with specific products or compilations that
|
|
||||||
contain the covered work, unless you entered into that arrangement,
|
|
||||||
or that patent license was granted, prior to 28 March 2007.
|
|
||||||
|
|
||||||
Nothing in this License shall be construed as excluding or limiting
|
|
||||||
any implied license or other defenses to infringement that may
|
|
||||||
otherwise be available to you under applicable patent law.
|
|
||||||
|
|
||||||
12. No Surrender of Others' Freedom.
|
|
||||||
|
|
||||||
If conditions are imposed on you (whether by court order, agreement or
|
|
||||||
otherwise) that contradict the conditions of this License, they do not
|
|
||||||
excuse you from the conditions of this License. If you cannot convey a
|
|
||||||
covered work so as to satisfy simultaneously your obligations under this
|
|
||||||
License and any other pertinent obligations, then as a consequence you may
|
|
||||||
not convey it at all. For example, if you agree to terms that obligate you
|
|
||||||
to collect a royalty for further conveying from those to whom you convey
|
|
||||||
the Program, the only way you could satisfy both those terms and this
|
|
||||||
License would be to refrain entirely from conveying the Program.
|
|
||||||
|
|
||||||
13. Use with the GNU Affero General Public License.
|
|
||||||
|
|
||||||
Notwithstanding any other provision of this License, you have
|
|
||||||
permission to link or combine any covered work with a work licensed
|
|
||||||
under version 3 of the GNU Affero General Public License into a single
|
|
||||||
combined work, and to convey the resulting work. The terms of this
|
|
||||||
License will continue to apply to the part which is the covered work,
|
|
||||||
but the special requirements of the GNU Affero General Public License,
|
|
||||||
section 13, concerning interaction through a network will apply to the
|
|
||||||
combination as such.
|
|
||||||
|
|
||||||
14. Revised Versions of this License.
|
|
||||||
|
|
||||||
The Free Software Foundation may publish revised and/or new versions of
|
|
||||||
the GNU General Public License from time to time. Such new versions will
|
|
||||||
be similar in spirit to the present version, but may differ in detail to
|
|
||||||
address new problems or concerns.
|
|
||||||
|
|
||||||
Each version is given a distinguishing version number. If the
|
|
||||||
Program specifies that a certain numbered version of the GNU General
|
|
||||||
Public License "or any later version" applies to it, you have the
|
|
||||||
option of following the terms and conditions either of that numbered
|
|
||||||
version or of any later version published by the Free Software
|
|
||||||
Foundation. If the Program does not specify a version number of the
|
|
||||||
GNU General Public License, you may choose any version ever published
|
|
||||||
by the Free Software Foundation.
|
|
||||||
|
|
||||||
If the Program specifies that a proxy can decide which future
|
|
||||||
versions of the GNU General Public License can be used, that proxy's
|
|
||||||
public statement of acceptance of a version permanently authorizes you
|
|
||||||
to choose that version for the Program.
|
|
||||||
|
|
||||||
Later license versions may give you additional or different
|
|
||||||
permissions. However, no additional obligations are imposed on any
|
|
||||||
author or copyright holder as a result of your choosing to follow a
|
|
||||||
later version.
|
|
||||||
|
|
||||||
15. Disclaimer of Warranty.
|
|
||||||
|
|
||||||
THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
|
|
||||||
APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
|
|
||||||
HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
|
|
||||||
OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
|
|
||||||
THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
||||||
PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
|
|
||||||
IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
|
|
||||||
ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
|
|
||||||
|
|
||||||
16. Limitation of Liability.
|
|
||||||
|
|
||||||
IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
|
|
||||||
WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
|
|
||||||
THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
|
|
||||||
GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
|
|
||||||
USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
|
|
||||||
DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
|
|
||||||
PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
|
|
||||||
EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
|
|
||||||
SUCH DAMAGES.
|
|
||||||
|
|
||||||
17. Interpretation of Sections 15 and 16.
|
|
||||||
|
|
||||||
If the disclaimer of warranty and limitation of liability provided
|
|
||||||
above cannot be given local legal effect according to their terms,
|
|
||||||
reviewing courts shall apply local law that most closely approximates
|
|
||||||
an absolute waiver of all civil liability in connection with the
|
|
||||||
Program, unless a warranty or assumption of liability accompanies a
|
|
||||||
copy of the Program in return for a fee.
|
|
||||||
|
|
||||||
END OF TERMS AND CONDITIONS
|
|
||||||
|
|
||||||
How to Apply These Terms to Your New Programs
|
|
||||||
|
|
||||||
If you develop a new program, and you want it to be of the greatest
|
|
||||||
possible use to the public, the best way to achieve this is to make it
|
|
||||||
free software which everyone can redistribute and change under these terms.
|
|
||||||
|
|
||||||
To do so, attach the following notices to the program. It is safest
|
|
||||||
to attach them to the start of each source file to most effectively
|
|
||||||
state the exclusion of warranty; and each file should have at least
|
|
||||||
the "copyright" line and a pointer to where the full notice is found.
|
|
||||||
|
|
||||||
<one line to give the program's name and a brief idea of what it does.>
|
|
||||||
Copyright (C) <year> <name of author>
|
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
|
||||||
it under the terms of the GNU General Public License as published by
|
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
This program 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
|
|
||||||
GNU General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
|
||||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
Also add information on how to contact you by electronic and paper mail.
|
|
||||||
|
|
||||||
If the program does terminal interaction, make it output a short
|
|
||||||
notice like this when it starts in an interactive mode:
|
|
||||||
|
|
||||||
<program> Copyright (C) <year> <name of author>
|
|
||||||
This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
|
|
||||||
This is free software, and you are welcome to redistribute it
|
|
||||||
under certain conditions; type `show c' for details.
|
|
||||||
|
|
||||||
The hypothetical commands `show w' and `show c' should show the appropriate
|
|
||||||
parts of the General Public License. Of course, your program's commands
|
|
||||||
might be different; for a GUI interface, you would use an "about box".
|
|
||||||
|
|
||||||
You should also get your employer (if you work as a programmer) or school,
|
|
||||||
if any, to sign a "copyright disclaimer" for the program, if necessary.
|
|
||||||
For more information on this, and how to apply and follow the GNU GPL, see
|
|
||||||
<https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
The GNU General Public License does not permit incorporating your program
|
|
||||||
into proprietary programs. If your program is a subroutine library, you
|
|
||||||
may consider it more useful to permit linking proprietary applications with
|
|
||||||
the library. If this is what you want to do, use the GNU Lesser General
|
|
||||||
Public License instead of this License. But first, please read
|
|
||||||
<https://www.gnu.org/licenses/why-not-lgpl.html>.
|
|
||||||
@@ -1,103 +0,0 @@
|
|||||||
[](https://hub.docker.com/r/jokobsk/netalertx)
|
|
||||||
[](https://hub.docker.com/r/jokobsk/netalertx)
|
|
||||||
[](https://github.com/jokob-sk/NetAlertX/releases)
|
|
||||||
[](https://discord.gg/NczTUTWyRr)
|
|
||||||
[](https://my.home-assistant.io/redirect/supervisor_add_addon_repository/?repository_url=https%3A%2F%2Fgithub.com%2Falexbelgium%2Fhassio-addons)
|
|
||||||
|
|
||||||
# NetAlertX - Network scanner & notification framework
|
|
||||||
|
|
||||||
| [📑 Docker guide](https://github.com/jokob-sk/NetAlertX/blob/main/dockerfiles/README.md) | [🚀 Releases](https://github.com/jokob-sk/NetAlertX/releases) | [📚 Docs](https://jokob-sk.github.io/NetAlertX/) | [🔌 Plugins](https://github.com/jokob-sk/NetAlertX/blob/main/docs/PLUGINS.md) | [🤖 Ask AI](https://gurubase.io/g/netalertx)
|
|
||||||
|----------------------| ----------------------| ----------------------| ----------------------| ----------------------|
|
|
||||||
|
|
||||||
<a href="https://raw.githubusercontent.com/jokob-sk/NetAlertX/main/docs/img/GENERAL/github_social_image.jpg" target="_blank">
|
|
||||||
<img src="https://raw.githubusercontent.com/jokob-sk/NetAlertX/main/docs/img/GENERAL/github_social_image.jpg" width="1000px" />
|
|
||||||
</a>
|
|
||||||
|
|
||||||
Head to [https://netalertx.com/](https://netalertx.com/) for more gifs and screenshots 📷.
|
|
||||||
|
|
||||||
> [!NOTE]
|
|
||||||
> There is also an experimental 🧪 [bare-metal install](https://github.com/jokob-sk/NetAlertX/blob/main/docs/HW_INSTALL.md) method available.
|
|
||||||
|
|
||||||
## 📕 Basic Usage
|
|
||||||
|
|
||||||
> [!WARNING]
|
|
||||||
> You will have to run the container on the `host` network and specify `SCAN_SUBNETS` unless you use other [plugin scanners](https://github.com/jokob-sk/NetAlertX/blob/main/docs/PLUGINS.md). The initial scan can take a few minutes, so please wait 5-10 minutes for the initial discovery to finish.
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
docker run -d --rm --network=host \
|
|
||||||
-v local_path/config:/app/config \
|
|
||||||
-v local_path/db:/app/db \
|
|
||||||
--mount type=tmpfs,target=/app/api \
|
|
||||||
-e PUID=200 -e PGID=300 \
|
|
||||||
-e TZ=Europe/Berlin \
|
|
||||||
-e PORT=20211 \
|
|
||||||
ghcr.io/jokob-sk/netalertx:latest
|
|
||||||
```
|
|
||||||
|
|
||||||
See alternative [docked-compose examples](https://github.com/jokob-sk/NetAlertX/blob/main/docs/DOCKER_COMPOSE.md).
|
|
||||||
|
|
||||||
### Docker environment variables
|
|
||||||
|
|
||||||
| Variable | Description | Example Value |
|
|
||||||
| :------------- |:------------------------| -----:|
|
|
||||||
| `PORT` |Port of the web interface | `20211` |
|
|
||||||
| `PUID` |Application User UID | `102` |
|
|
||||||
| `PGID` |Application User GID | `82` |
|
|
||||||
| `LISTEN_ADDR` |Set the specific IP Address for the listener address for the nginx webserver (web interface). This could be useful when using multiple subnets to hide the web interface from all untrusted networks. | `0.0.0.0` |
|
|
||||||
|`TZ` |Time zone to display stats correctly. Find your time zone [here](https://en.wikipedia.org/wiki/List_of_tz_database_time_zones) | `Europe/Berlin` |
|
|
||||||
|`LOADED_PLUGINS` | Default [plugins](https://github.com/jokob-sk/NetAlertX/blob/main/docs/PLUGINS.md) to load. Plugins cannot be loaded with `APP_CONF_OVERRIDE`, you need to use this variable instead and then specify the plugins settings with `APP_CONF_OVERRIDE`. | `["PIHOLE","ASUSWRT"]` |
|
|
||||||
|`APP_CONF_OVERRIDE` | JSON override for settings (except `LOADED_PLUGINS`). | `{"SCAN_SUBNETS":"['192.168.1.0/24 --interface=eth1']","GRAPHQL_PORT":"20212"}` |
|
|
||||||
|`ALWAYS_FRESH_INSTALL` | ⚠ If `true` will delete the content of the `/db` & `/config` folders. For testing purposes. Can be coupled with [watchtower](https://github.com/containrrr/watchtower) to have an always freshly installed `netalertx`/`netalertx-dev` image. | `true` |
|
|
||||||
|
|
||||||
> You can override the default GraphQL port setting `GRAPHQL_PORT` (set to `20212`) by using the `APP_CONF_OVERRIDE` env variable. `LOADED_PLUGINS` and settings in `APP_CONF_OVERRIDE` can be specified via the UI as well.
|
|
||||||
|
|
||||||
### Docker paths
|
|
||||||
|
|
||||||
> [!NOTE]
|
|
||||||
> See also [Backup strategies](https://github.com/jokob-sk/NetAlertX/blob/main/docs/BACKUPS.md).
|
|
||||||
|
|
||||||
| Required | Path | Description |
|
|
||||||
| :------------- | :------------- | :-------------|
|
|
||||||
| ✅ | `:/app/config` | Folder which will contain the `app.conf` & `devices.csv` ([read about devices.csv](https://github.com/jokob-sk/NetAlertX/blob/main/docs/DEVICES_BULK_EDITING.md)) files |
|
|
||||||
| ✅ | `:/app/db` | Folder which will contain the `app.db` database file |
|
|
||||||
| | `:/app/log` | Logs folder useful for debugging if you have issues setting up the container |
|
|
||||||
| | `:/app/api` | A simple [API endpoint](https://github.com/jokob-sk/NetAlertX/blob/main/docs/API.md) containing static (but regularly updated) json and other files. |
|
|
||||||
| | `:/app/front/plugins/<plugin>/ignore_plugin` | Map a file `ignore_plugin` to ignore a plugin. Plugins can be soft-disabled via settings. More in the [Plugin docs](https://github.com/jokob-sk/NetAlertX/blob/main/docs/PLUGINS.md). |
|
|
||||||
| | `:/etc/resolv.conf` | Use a custom `resolv.conf` file for [better name resolution](https://github.com/jokob-sk/NetAlertX/blob/main/docs/REVERSE_DNS.md). |
|
|
||||||
|
|
||||||
> Use separate `db` and `config` directories, do not nest them.
|
|
||||||
|
|
||||||
### Initial setup
|
|
||||||
|
|
||||||
- If unavailable, the app generates a default `app.conf` and `app.db` file on the first run.
|
|
||||||
- The preferred way is to manage the configuration via the Settings section in the UI, if UI is inaccessible you can modify [app.conf](https://github.com/jokob-sk/NetAlertX/tree/main/back) in the `/app/config/` folder directly
|
|
||||||
|
|
||||||
#### Setting up scanners
|
|
||||||
|
|
||||||
You have to specify which network(s) should be scanned. This is done by entering subnets that are accessible from the host. If you use the default `ARPSCAN` plugin, you have to specify at least one valid subnet and interface in the `SCAN_SUBNETS` setting. See the documentation on [How to set up multiple SUBNETS, VLANs and what are limitations](https://github.com/jokob-sk/NetAlertX/blob/main/docs/SUBNETS.md) for troubleshooting and more advanced scenarios.
|
|
||||||
|
|
||||||
If you are running PiHole you can synchronize devices directly. Check the [PiHole configuration guide](https://github.com/jokob-sk/NetAlertX/blob/main/docs/PIHOLE_GUIDE.md) for details.
|
|
||||||
|
|
||||||
> [!NOTE]
|
|
||||||
> You can bulk-import devices via the [CSV import method](https://github.com/jokob-sk/NetAlertX/blob/main/docs/DEVICES_BULK_EDITING.md).
|
|
||||||
|
|
||||||
#### Community guides
|
|
||||||
|
|
||||||
You can read or watch several [community configuration guides](https://github.com/jokob-sk/NetAlertX/blob/main/docs/COMMUNITY_GUIDES.md) in Chinese, Korean, German, or French.
|
|
||||||
|
|
||||||
> Please note these might be outdated. Rely on official documentation first.
|
|
||||||
|
|
||||||
#### Common issues
|
|
||||||
|
|
||||||
- Before creating a new issue, please check if a similar issue was [already resolved](https://github.com/jokob-sk/NetAlertX/issues?q=is%3Aissue+is%3Aclosed).
|
|
||||||
- Check also common issues and [debugging tips](https://github.com/jokob-sk/NetAlertX/blob/main/docs/DEBUG_TIPS.md).
|
|
||||||
|
|
||||||
## 💙 Support me
|
|
||||||
|
|
||||||
| [](https://github.com/sponsors/jokob-sk) | [](https://www.buymeacoffee.com/jokobsk) | [](https://www.patreon.com/user?u=84385063) |
|
|
||||||
| --- | --- | --- |
|
|
||||||
|
|
||||||
- Bitcoin: `1N8tupjeCK12qRVU2XrV17WvKK7LCawyZM`
|
|
||||||
- Ethereum: `0x6e2749Cb42F4411bc98501406BdcD82244e3f9C7`
|
|
||||||
|
|
||||||
> 📧 Email me at [netalertx@gmail.com](mailto:netalertx@gmail.com?subject=NetAlertX Donations) if you want to get in touch or if I should add other sponsorship platforms.
|
|
||||||
@@ -1,169 +0,0 @@
|
|||||||
#!/usr/bin/with-contenv bash
|
|
||||||
|
|
||||||
echo "---------------------------------------------------------
|
|
||||||
[INSTALL] Run init.sh
|
|
||||||
---------------------------------------------------------"
|
|
||||||
|
|
||||||
DEFAULT_PUID=102
|
|
||||||
DEFAULT_GID=82
|
|
||||||
|
|
||||||
PUID=${PUID:-${DEFAULT_PUID}}
|
|
||||||
PGID=${PGID:-${DEFAULT_GID}}
|
|
||||||
|
|
||||||
echo "[INSTALL] Setting up user UID and GID"
|
|
||||||
|
|
||||||
if ! groupmod -o -g "$PGID" www-data && [ "$PGID" != "$DEFAULT_GID" ] ; then
|
|
||||||
echo "Failed to set user GID to ${PGID}, trying with default GID ${DEFAULT_GID}"
|
|
||||||
groupmod -o -g "$DEFAULT_GID" www-data
|
|
||||||
fi
|
|
||||||
if ! usermod -o -u "$PUID" nginx && [ "$PUID" != "$DEFAULT_PUID" ] ; then
|
|
||||||
echo "Failed to set user UID to ${PUID}, trying with default PUID ${DEFAULT_PUID}"
|
|
||||||
usermod -o -u "$DEFAULT_PUID" nginx
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "
|
|
||||||
---------------------------------------------------------
|
|
||||||
GID/UID
|
|
||||||
---------------------------------------------------------
|
|
||||||
User UID: $(id -u nginx)
|
|
||||||
User GID: $(getent group www-data | cut -d: -f3)
|
|
||||||
---------------------------------------------------------"
|
|
||||||
|
|
||||||
chown nginx:nginx /run/nginx/ /var/log/nginx/ /var/lib/nginx/ /var/lib/nginx/tmp/
|
|
||||||
chgrp www-data /var/www/localhost/htdocs/
|
|
||||||
|
|
||||||
export INSTALL_DIR=/app # Specify the installation directory here
|
|
||||||
|
|
||||||
# DO NOT CHANGE ANYTHING BELOW THIS LINE!
|
|
||||||
|
|
||||||
CONF_FILE="app.conf"
|
|
||||||
NGINX_CONF_FILE=netalertx.conf
|
|
||||||
DB_FILE="app.db"
|
|
||||||
FULL_FILEDB_PATH="${INSTALL_DIR}/db/${DB_FILE}"
|
|
||||||
NGINX_CONFIG_FILE="/etc/nginx/http.d/${NGINX_CONF_FILE}"
|
|
||||||
OUI_FILE="/usr/share/arp-scan/ieee-oui.txt" # Define the path to ieee-oui.txt and ieee-iab.txt
|
|
||||||
|
|
||||||
INSTALL_DIR_OLD=/home/pi/pialert
|
|
||||||
OLD_APP_NAME=pialert
|
|
||||||
|
|
||||||
# DO NOT CHANGE ANYTHING ABOVE THIS LINE!
|
|
||||||
|
|
||||||
# Check if script is run as root
|
|
||||||
if [[ $EUID -ne 0 ]]; then
|
|
||||||
echo "This script must be run as root. Please use 'sudo'."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# DANGER ZONE: ALWAYS_FRESH_INSTALL
|
|
||||||
if [ "$ALWAYS_FRESH_INSTALL" = true ]; then
|
|
||||||
echo "[INSTALL] ❗ ALERT /db and /config folders are cleared because the ALWAYS_FRESH_INSTALL is set to: $ALWAYS_FRESH_INSTALL❗"
|
|
||||||
|
|
||||||
# Delete content of "$INSTALL_DIR/config/"
|
|
||||||
rm -rf "$INSTALL_DIR/config/"*
|
|
||||||
rm -rf "$INSTALL_DIR_OLD/config/"*
|
|
||||||
|
|
||||||
# Delete content of "$INSTALL_DIR/db/"
|
|
||||||
rm -rf "$INSTALL_DIR/db/"*
|
|
||||||
rm -rf "$INSTALL_DIR_OLD/db/"*
|
|
||||||
fi
|
|
||||||
|
|
||||||
# OVERRIDE settings: Handling APP_CONF_OVERRIDE
|
|
||||||
# Check if APP_CONF_OVERRIDE is set
|
|
||||||
|
|
||||||
# remove old
|
|
||||||
rm "${INSTALL_DIR}/config/app_conf_override.json"
|
|
||||||
|
|
||||||
if [ -z "$APP_CONF_OVERRIDE" ]; then
|
|
||||||
echo "APP_CONF_OVERRIDE is not set. Skipping config file creation."
|
|
||||||
else
|
|
||||||
# Save the APP_CONF_OVERRIDE env variable as a JSON file
|
|
||||||
echo "$APP_CONF_OVERRIDE" > "${INSTALL_DIR}/config/app_conf_override.json"
|
|
||||||
echo "Config file saved to ${INSTALL_DIR}/config/app_conf_override.json"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# 🔻 FOR BACKWARD COMPATIBILITY - REMOVE AFTER 12/12/2025
|
|
||||||
|
|
||||||
# Check if pialert.db exists, then create a symbolic link to app.db
|
|
||||||
if [ -f "${INSTALL_DIR_OLD}/db/${OLD_APP_NAME}.db" ]; then
|
|
||||||
ln -s "${INSTALL_DIR_OLD}/db/${OLD_APP_NAME}.db" "${FULL_FILEDB_PATH}"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Check if ${OLD_APP_NAME}.conf exists, then create a symbolic link to app.conf
|
|
||||||
if [ -f "${INSTALL_DIR_OLD}/config/${OLD_APP_NAME}.conf" ]; then
|
|
||||||
ln -s "${INSTALL_DIR_OLD}/config/${OLD_APP_NAME}.conf" "${INSTALL_DIR}/config/${CONF_FILE}"
|
|
||||||
fi
|
|
||||||
# 🔺 FOR BACKWARD COMPATIBILITY - REMOVE AFTER 12/12/2025
|
|
||||||
|
|
||||||
echo "[INSTALL] Copy starter ${DB_FILE} and ${CONF_FILE} if they don't exist"
|
|
||||||
|
|
||||||
# Copy starter app.db, app.conf if they don't exist
|
|
||||||
cp -na "${INSTALL_DIR}/back/${CONF_FILE}" "${INSTALL_DIR}/config/${CONF_FILE}"
|
|
||||||
cp -na "${INSTALL_DIR}/back/${DB_FILE}" "${FULL_FILEDB_PATH}"
|
|
||||||
|
|
||||||
# if custom variables not set we do not need to do anything
|
|
||||||
if [ -n "${TZ}" ]; then
|
|
||||||
FILECONF="${INSTALL_DIR}/config/${CONF_FILE}"
|
|
||||||
echo "[INSTALL] Setup timezone"
|
|
||||||
sed -i "\#^TIMEZONE=#c\TIMEZONE='${TZ}'" "${FILECONF}"
|
|
||||||
|
|
||||||
# set TimeZone in container
|
|
||||||
cp /usr/share/zoneinfo/$TZ /etc/localtime
|
|
||||||
echo $TZ > /etc/timezone
|
|
||||||
fi
|
|
||||||
|
|
||||||
# if custom variables not set we do not need to do anything
|
|
||||||
if [ -n "${LOADED_PLUGINS}" ]; then
|
|
||||||
FILECONF="${INSTALL_DIR}/config/${CONF_FILE}"
|
|
||||||
echo "[INSTALL] Setup custom LOADED_PLUGINS variable"
|
|
||||||
sed -i "\#^LOADED_PLUGINS=#c\LOADED_PLUGINS=${LOADED_PLUGINS}" "${FILECONF}"
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo "[INSTALL] Setup NGINX"
|
|
||||||
echo "Setting webserver to address ($LISTEN_ADDR) and port ($PORT)"
|
|
||||||
envsubst '$INSTALL_DIR $LISTEN_ADDR $PORT' < "${INSTALL_DIR}/install/netalertx.template.conf" > "${NGINX_CONFIG_FILE}"
|
|
||||||
|
|
||||||
# Run the hardware vendors update at least once
|
|
||||||
echo "[INSTALL] Run the hardware vendors update"
|
|
||||||
|
|
||||||
# Check if ieee-oui.txt or ieee-iab.txt exist
|
|
||||||
if [ -f "${OUI_FILE}" ]; then
|
|
||||||
echo "The file ieee-oui.txt exists. Skipping update_vendors.sh..."
|
|
||||||
else
|
|
||||||
echo "The file ieee-oui.txt does not exist. Running update_vendors..."
|
|
||||||
|
|
||||||
# Run the update_vendors.sh script
|
|
||||||
if [ -f "${INSTALL_DIR}/back/update_vendors.sh" ]; then
|
|
||||||
"${INSTALL_DIR}/back/update_vendors.sh"
|
|
||||||
else
|
|
||||||
echo "update_vendors.sh script not found in ${INSTALL_DIR}."
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Create an empty log files
|
|
||||||
# Create the execution_queue.log and app_front.log files if they don't exist
|
|
||||||
touch "${INSTALL_DIR}"/log/{app.log,execution_queue.log,app_front.log,app.php_errors.log,stderr.log,stdout.log,db_is_locked.log}
|
|
||||||
touch "${INSTALL_DIR}"/api/user_notifications.json
|
|
||||||
|
|
||||||
# Create plugins sub-directory if it doesn't exist in case a custom log folder is used
|
|
||||||
mkdir -p "${INSTALL_DIR}"/log/plugins
|
|
||||||
|
|
||||||
echo "[INSTALL] Fixing permissions after copied starter config & DB"
|
|
||||||
chown -R nginx:www-data "${INSTALL_DIR}"
|
|
||||||
|
|
||||||
chmod 750 "${INSTALL_DIR}"/{config,log,db}
|
|
||||||
find "${INSTALL_DIR}"/{config,log,db} -type f -exec chmod 640 {} \;
|
|
||||||
|
|
||||||
# Check if buildtimestamp.txt doesn't exist
|
|
||||||
if [ ! -f "${INSTALL_DIR}/front/buildtimestamp.txt" ]; then
|
|
||||||
# Create buildtimestamp.txt
|
|
||||||
date +%s > "${INSTALL_DIR}/front/buildtimestamp.txt"
|
|
||||||
chown nginx:www-data "${INSTALL_DIR}/front/buildtimestamp.txt"
|
|
||||||
fi
|
|
||||||
|
|
||||||
echo -e "
|
|
||||||
[ENV] PATH is ${PATH}
|
|
||||||
[ENV] PORT is ${PORT}
|
|
||||||
[ENV] TZ is ${TZ}
|
|
||||||
[ENV] LISTEN_ADDR is ${LISTEN_ADDR}
|
|
||||||
[ENV] ALWAYS_FRESH_INSTALL is ${ALWAYS_FRESH_INSTALL}
|
|
||||||
"
|
|
||||||
@@ -1,49 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
export INSTALL_DIR=/app
|
|
||||||
export APP_NAME=netalertx
|
|
||||||
|
|
||||||
# php-fpm setup
|
|
||||||
install -d -o nginx -g www-data /run/php/
|
|
||||||
sed -i "/^;pid/c\pid = /run/php/php8.3-fpm.pid" /etc/php83/php-fpm.conf
|
|
||||||
sed -i "/^listen/c\listen = /run/php/php8.3-fpm.sock" /etc/php83/php-fpm.d/www.conf
|
|
||||||
sed -i "/^;listen.owner/c\listen.owner = nginx" /etc/php83/php-fpm.d/www.conf
|
|
||||||
sed -i "/^;listen.group/c\listen.group = www-data" /etc/php83/php-fpm.d/www.conf
|
|
||||||
sed -i "/^user/c\user = nginx" /etc/php83/php-fpm.d/www.conf
|
|
||||||
sed -i "/^group/c\group = www-data" /etc/php83/php-fpm.d/www.conf
|
|
||||||
|
|
||||||
# s6 overlay setup
|
|
||||||
mkdir -p /etc/s6-overlay/s6-rc.d/{SetupOneshot,crond/dependencies.d,php-fpm/dependencies.d,nginx/dependencies.d,$APP_NAME/dependencies.d}
|
|
||||||
echo "oneshot" > /etc/s6-overlay/s6-rc.d/SetupOneshot/type
|
|
||||||
echo "longrun" > /etc/s6-overlay/s6-rc.d/crond/type
|
|
||||||
echo "longrun" > /etc/s6-overlay/s6-rc.d/php-fpm/type
|
|
||||||
echo "longrun" > /etc/s6-overlay/s6-rc.d/nginx/type
|
|
||||||
echo "longrun" > /etc/s6-overlay/s6-rc.d/$APP_NAME/type
|
|
||||||
echo -e "${INSTALL_DIR}/dockerfiles/init.sh" > /etc/s6-overlay/s6-rc.d/SetupOneshot/up
|
|
||||||
echo -e '#!/bin/execlineb -P
|
|
||||||
|
|
||||||
if { echo
|
|
||||||
"
|
|
||||||
[INSTALL] Starting crond service...
|
|
||||||
|
|
||||||
" }' > /etc/s6-overlay/s6-rc.d/crond/run
|
|
||||||
echo -e "/usr/sbin/crond -f" >> /etc/s6-overlay/s6-rc.d/crond/run
|
|
||||||
echo -e "#!/bin/execlineb -P\n/usr/sbin/php-fpm83 -F" > /etc/s6-overlay/s6-rc.d/php-fpm/run
|
|
||||||
echo -e '#!/bin/execlineb -P\nnginx -g "daemon off;"' > /etc/s6-overlay/s6-rc.d/nginx/run
|
|
||||||
echo -e '#!/bin/execlineb -P
|
|
||||||
with-contenv
|
|
||||||
|
|
||||||
importas -u PORT PORT
|
|
||||||
|
|
||||||
if { echo
|
|
||||||
"
|
|
||||||
[INSTALL] 🚀 Starting app (:${PORT})
|
|
||||||
|
|
||||||
" }' > /etc/s6-overlay/s6-rc.d/$APP_NAME/run
|
|
||||||
echo -e "python ${INSTALL_DIR}/server" >> /etc/s6-overlay/s6-rc.d/$APP_NAME/run
|
|
||||||
touch /etc/s6-overlay/s6-rc.d/user/contents.d/{SetupOneshot,crond,php-fpm,nginx,$APP_NAME} /etc/s6-overlay/s6-rc.d/{crond,php-fpm,nginx,$APP_NAME}/dependencies.d/SetupOneshot
|
|
||||||
touch /etc/s6-overlay/s6-rc.d/nginx/dependencies.d/php-fpm
|
|
||||||
touch /etc/s6-overlay/s6-rc.d/$APP_NAME/dependencies.d/nginx
|
|
||||||
|
|
||||||
# this removes the current file
|
|
||||||
rm -f $0
|
|
||||||
44
docs/DEV_PORTS_HOST_MODE.md
Normal file
44
docs/DEV_PORTS_HOST_MODE.md
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
# Dev Ports in Host Network Mode
|
||||||
|
|
||||||
|
When using `"--network=host"` in the devcontainer, VS Code's normal port forwarding model doesn't apply. All container ports are already on the host network namespace, so:
|
||||||
|
|
||||||
|
- Listing ports in `forwardPorts` can cause VS Code to pre-bind or reserve them (conflicts with startup scripts waiting for a free port).
|
||||||
|
- The PORTS panel will not auto-detect services reliably, because forwarding isn't occurring.
|
||||||
|
- Debugger ports (e.g. Xdebug `9003`, Python debugpy `5678`) can still be listed safely.
|
||||||
|
|
||||||
|
## Recommended Pattern
|
||||||
|
|
||||||
|
1. Only include debugger ports in `forwardPorts`:
|
||||||
|
```jsonc
|
||||||
|
"forwardPorts": [5678, 9003]
|
||||||
|
```
|
||||||
|
2. Do NOT list application service ports (e.g. 20211, 20212) there when in host mode.
|
||||||
|
3. Use the helper task to enumerate current bindings:
|
||||||
|
- Run task: `> Tasks: Run Task` → `[Dev Container] List NetAlertX Ports`
|
||||||
|
|
||||||
|
## Port Enumeration Script
|
||||||
|
Script: `scripts/list-ports.sh`
|
||||||
|
Outputs binding address, PID (if resolvable) and process name for key ports.
|
||||||
|
|
||||||
|
You can edit the PORTS variable inside that script to add/remove watched ports.
|
||||||
|
|
||||||
|
## Xdebug Notes
|
||||||
|
Set in `99-xdebug.ini`:
|
||||||
|
```ini
|
||||||
|
xdebug.client_host=127.0.0.1
|
||||||
|
xdebug.client_port=9003
|
||||||
|
xdebug.discover_client_host=1
|
||||||
|
```
|
||||||
|
Ensure your IDE is listening on 9003.
|
||||||
|
|
||||||
|
## Troubleshooting
|
||||||
|
| Symptom | Cause | Fix |
|
||||||
|
|---------|-------|-----|
|
||||||
|
| `Waiting for port 20211 to free...` repeats | VS Code pre-bound the port via `forwardPorts` | Remove the port from `forwardPorts`, rebuild, retry |
|
||||||
|
| PHP request hangs at start | Xdebug trying to connect to unresolved host (`host.docker.internal`) | Use `127.0.0.1` or rely on discovery |
|
||||||
|
| PORTS panel empty | Expected in host mode | Use the port enumeration task |
|
||||||
|
|
||||||
|
## Future Improvements
|
||||||
|
- Optional: add a small web status endpoint summarizing runtime ports.
|
||||||
|
- Optional: detect host mode in `setup.sh` and skip the wait loop if the PID using port is the intended process.
|
||||||
|
|
||||||
@@ -10,12 +10,47 @@ require dirname(__FILE__).'/../server/init.php';
|
|||||||
// check if authenticated
|
// check if authenticated
|
||||||
require_once $_SERVER['DOCUMENT_ROOT'] . '/php/templates/security.php';
|
require_once $_SERVER['DOCUMENT_ROOT'] . '/php/templates/security.php';
|
||||||
|
|
||||||
exec('../../../back/speedtest-cli --secure --simple', $output);
|
// Prefer explicit binary paths, fall back to a sanitised PATH
|
||||||
|
$speedtestCandidates = [
|
||||||
|
'/opt/venv/bin/speedtest-cli',
|
||||||
|
'/usr/local/bin/speedtest-cli',
|
||||||
|
'/usr/bin/speedtest-cli',
|
||||||
|
];
|
||||||
|
|
||||||
echo '<h4>'. lang('Speedtest_Results') .'</h4>';
|
$candidateDirs = array_unique(array_map('dirname', $speedtestCandidates));
|
||||||
echo '<pre style="border: none;">';
|
$safePath = implode(':', $candidateDirs);
|
||||||
foreach($output as $line){
|
|
||||||
echo $line . "\n";
|
$resolvedCommand = null;
|
||||||
|
foreach ($speedtestCandidates as $candidate) {
|
||||||
|
if (is_executable($candidate)) {
|
||||||
|
$resolvedCommand = escapeshellcmd($candidate) . ' --secure --simple';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$output = [];
|
||||||
|
$returnCode = 0;
|
||||||
|
|
||||||
|
if ($resolvedCommand === null) {
|
||||||
|
$resolvedCommand = 'env PATH=' . escapeshellarg($safePath) . ' speedtest-cli --secure --simple';
|
||||||
|
}
|
||||||
|
|
||||||
|
exec($resolvedCommand, $output, $returnCode);
|
||||||
|
|
||||||
|
echo '<h4>' . lang('Speedtest_Results') . '</h4>';
|
||||||
|
|
||||||
|
if ($returnCode !== 0 || empty($output)) {
|
||||||
|
$errorMessage = $returnCode === 127
|
||||||
|
? 'speedtest-cli command not found. Checked paths: ' . $safePath
|
||||||
|
: (empty($output) ? 'speedtest-cli returned no output.' : implode("\n", $output));
|
||||||
|
|
||||||
|
echo '<div class="alert alert-danger">' . htmlspecialchars('Speedtest failed: ' . $errorMessage, ENT_QUOTES, 'UTF-8') . '</div>';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
echo '<pre style="border: none;">';
|
||||||
|
foreach ($output as $line) {
|
||||||
|
echo htmlspecialchars($line, ENT_QUOTES, 'UTF-8') . "\n";
|
||||||
}
|
}
|
||||||
echo '</pre>';
|
echo '</pre>';
|
||||||
?>
|
?>
|
||||||
|
|||||||
@@ -139,4 +139,4 @@ def main():
|
|||||||
# Entrypoint
|
# Entrypoint
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
main()
|
main()
|
||||||
@@ -41,7 +41,7 @@ def main():
|
|||||||
plugin_objects = Plugin_Objects(RESULT_FILE)
|
plugin_objects = Plugin_Objects(RESULT_FILE)
|
||||||
timeoutSec = get_setting_value('DHCPSRVS_RUN_TIMEOUT')
|
timeoutSec = get_setting_value('DHCPSRVS_RUN_TIMEOUT')
|
||||||
|
|
||||||
nmapArgs = ['sudo', 'nmap', '--script', 'broadcast-dhcp-discover']
|
nmapArgs = ['sudo', 'nmap', '--privileged', '--script', 'broadcast-dhcp-discover']
|
||||||
|
|
||||||
try:
|
try:
|
||||||
dhcp_probes = 1
|
dhcp_probes = 1
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ def update_vendor_database():
|
|||||||
|
|
||||||
# Update vendors DB (iab oui)
|
# Update vendors DB (iab oui)
|
||||||
mylog('verbose', [' Updating vendors DB (iab & oui)'])
|
mylog('verbose', [' Updating vendors DB (iab & oui)'])
|
||||||
update_args = ['sh', applicationPath + '/back/update_vendors.sh']
|
update_args = ['sh', applicationPath + '/services/update_vendors.sh']
|
||||||
|
|
||||||
# Execute command
|
# Execute command
|
||||||
try:
|
try:
|
||||||
|
|||||||
@@ -6,6 +6,15 @@ echo "---------------------------------------------------------"
|
|||||||
|
|
||||||
# ❗ IMPORTANT - if you modify this file modify the root Dockerfile as well ❗
|
# ❗ IMPORTANT - if you modify this file modify the root Dockerfile as well ❗
|
||||||
|
|
||||||
|
SCRIPT_DIR="$(cd -- "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
||||||
|
REPO_ROOT="$(cd "${SCRIPT_DIR}/../.." && pwd)"
|
||||||
|
REQUIREMENTS_FILE="${REPO_ROOT}/requirements.txt"
|
||||||
|
|
||||||
|
if [[ ! -f "${REQUIREMENTS_FILE}" ]]; then
|
||||||
|
echo "requirements.txt not found at ${REQUIREMENTS_FILE}. Please ensure the repository root is available." >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
# Check if script is run as root
|
# Check if script is run as root
|
||||||
if [[ $EUID -ne 0 ]]; then
|
if [[ $EUID -ne 0 ]]; then
|
||||||
echo "This script must be run as root. Please use 'sudo'."
|
echo "This script must be run as root. Please use 'sudo'."
|
||||||
@@ -14,7 +23,7 @@ fi
|
|||||||
|
|
||||||
# Install dependencies
|
# Install dependencies
|
||||||
apt-get install -y \
|
apt-get install -y \
|
||||||
tini snmp ca-certificates curl libwww-perl arp-scan perl apt-utils cron sudo \
|
tini snmp ca-certificates curl libwww-perl arp-scan perl apt-utils cron sudo gettext-base \
|
||||||
nginx-light php php-cgi php-fpm php-sqlite3 php-curl sqlite3 dnsutils net-tools \
|
nginx-light php php-cgi php-fpm php-sqlite3 php-curl sqlite3 dnsutils net-tools \
|
||||||
python3 python3-dev iproute2 nmap python3-pip zip usbutils traceroute nbtscan avahi-daemon avahi-utils openrc build-essential git
|
python3 python3-dev iproute2 nmap python3-pip zip usbutils traceroute nbtscan avahi-daemon avahi-utils openrc build-essential git
|
||||||
|
|
||||||
@@ -24,10 +33,10 @@ sudo phpenmod -v 8.2 sqlite3
|
|||||||
|
|
||||||
# setup virtual python environment so we can use pip3 to install packages
|
# setup virtual python environment so we can use pip3 to install packages
|
||||||
apt-get install python3-venv -y
|
apt-get install python3-venv -y
|
||||||
python3 -m venv myenv
|
python3 -m venv /opt/venv
|
||||||
source myenv/bin/activate
|
source /opt/venv/bin/activate
|
||||||
|
|
||||||
update-alternatives --install /usr/bin/python python /usr/bin/python3 10
|
update-alternatives --install /usr/bin/python python /usr/bin/python3 10
|
||||||
|
|
||||||
# install packages thru pip3
|
# install packages thru pip3
|
||||||
pip3 install openwrt-luci-rpc asusrouter asyncio aiohttp graphene flask flask-cors unifi-sm-api tplink-omada-client wakeonlan pycryptodome requests paho-mqtt scapy cron-converter pytz json2table dhcp-leases pyunifi speedtest-cli chardet python-nmap dnspython librouteros yattag zeroconf git+https://github.com/foreign-sub/aiofreepybox.git
|
pip3 install -r "${REQUIREMENTS_FILE}"
|
||||||
|
|||||||
@@ -102,10 +102,10 @@ else
|
|||||||
echo "The file ieee-oui.txt does not exist. Running update_vendors..."
|
echo "The file ieee-oui.txt does not exist. Running update_vendors..."
|
||||||
|
|
||||||
# Run the update_vendors.sh script
|
# Run the update_vendors.sh script
|
||||||
if [ -f "${INSTALL_PATH}/back/update_vendors.sh" ]; then
|
if [ -f "${SYSTEM_SERVICES}/update_vendors.sh" ]; then
|
||||||
"${INSTALL_PATH}/back/update_vendors.sh"
|
"${SYSTEM_SERVICES}/update_vendors.sh"
|
||||||
else
|
else
|
||||||
echo "update_vendors.sh script not found in $INSTALL_DIR."
|
echo "update_vendors.sh script not found in $SYSTEM_SERVICES."
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -175,7 +175,7 @@ nginx -t || { echo "[INSTALL] nginx config test failed"; exit 1; }
|
|||||||
# sudo systemctl restart nginx
|
# sudo systemctl restart nginx
|
||||||
|
|
||||||
# Activate the virtual python environment
|
# Activate the virtual python environment
|
||||||
source myenv/bin/activate
|
source /opt/venv/bin/activate
|
||||||
|
|
||||||
echo "[INSTALL] 🚀 Starting app - navigate to your <server IP>:${PORT}"
|
echo "[INSTALL] 🚀 Starting app - navigate to your <server IP>:${PORT}"
|
||||||
|
|
||||||
|
|||||||
94
install/production-filesystem/README.md
Normal file
94
install/production-filesystem/README.md
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
|
||||||
|
# NetAlertX Production Container Filesystem
|
||||||
|
|
||||||
|
This document describes the filesystem structure of the NetAlertX production Docker container. This setup focuses on security by separating application code, configuration, and runtime data.
|
||||||
|
|
||||||
|
## Directory Structure
|
||||||
|
|
||||||
|
### `/app` - Main Application Directory
|
||||||
|
The core application location where NetAlertX runs. This directory contains the main application code and working data, with source code directories mounted in read-only mode for security. It provides the runtime environment for all NetAlertX operations including device scanning, web interface, and data processing.
|
||||||
|
|
||||||
|
The core application location. Contains:
|
||||||
|
- Source code directories (`back`, `front`, `server`) copied in read-only mode
|
||||||
|
- Working directories for runtime data (`config`, `db`, `log`)
|
||||||
|
- Other directories are not needed in production and are excluded
|
||||||
|
|
||||||
|
### `/build` - Build-Time Scripts
|
||||||
|
Temporary directory used during Docker image building to prepare the container environment. Scripts in this directory run during the build process to set up the system before it's locked down for production use. This ensures the container is properly configured before runtime.
|
||||||
|
|
||||||
|
Temporary directory used during Docker image building:
|
||||||
|
- Scripts run at the end of the build process
|
||||||
|
- Deleted after build to reduce image size
|
||||||
|
- Only exists during container creation
|
||||||
|
|
||||||
|
### `/opt/venv/lib/python3.12/site-packages/aiofreebox` - Certificate Storage
|
||||||
|
Contains SSL certificates required for secure communication with Freebox OS devices. The aiofreebox Python package uses these certificates to authenticate and establish encrypted connections when integrating with Freebox routers for network device discovery.
|
||||||
|
|
||||||
|
Contains certificates for the aiofreebox package, which communicates with Freebox OS devices.
|
||||||
|
|
||||||
|
### `/services` - Service Management
|
||||||
|
Contains all scripts and configurations for running NetAlertX services. This directory holds the complete service orchestration layer that manages the container's runtime behavior, including startup scripts, configuration files, and utility tools for system maintenance and monitoring.
|
||||||
|
|
||||||
|
Contains all scripts and configurations for running NetAlertX services:
|
||||||
|
|
||||||
|
#### `/services/config` - Service Configurations
|
||||||
|
Configuration files for each service that runs in the container. These files define how services like the web server, task scheduler, and Python backend operate, including security settings, resource limits, and integration parameters.
|
||||||
|
|
||||||
|
Configuration files for each service:
|
||||||
|
- `crond/` - Task scheduler settings
|
||||||
|
- `nginx/` - Web server configuration
|
||||||
|
- `php/` - PHP interpreter settings
|
||||||
|
- `php-fpm.d/` - Additional PHP configurations
|
||||||
|
- `python/` - Python backend launch parameters
|
||||||
|
|
||||||
|
#### `/services/scripts` - System Scripts and Utilities
|
||||||
|
Pre-startup checks and specialized maintenance tools. Files named `check-*` are intended to verify system functions at startup and correct issues or warn users as needed. Additional scripts perform various update tasks and provide integration capabilities with external systems.
|
||||||
|
|
||||||
|
Pre-startup checks and specialized maintenance tools:
|
||||||
|
- `check-cap.sh` - Verifies container permissions for network tools
|
||||||
|
- `check-first-run-config.sh` - Sets up initial configuration
|
||||||
|
- `check-first-run-db.sh` - Prepares database on first run
|
||||||
|
- `check-permissions.sh` - Validates file and directory permissions
|
||||||
|
- `check-ramdisk.sh` - Checks temporary storage setup
|
||||||
|
- `check-root.sh` - Confirms proper user privileges
|
||||||
|
- `check-storage.sh` - Ensures storage directories exist
|
||||||
|
- `update_vendors.sh` - Updates MAC address vendor database
|
||||||
|
- `checkmk/` - Checkmk monitoring integration scripts
|
||||||
|
- `db_cleanup/` - Database maintenance and cleanup tools
|
||||||
|
- `db_empty/` - Database reset utilities
|
||||||
|
- `list-ports.sh` - Network port enumeration script
|
||||||
|
- `opnsense_leases/` - OPNsense DHCP lease integration tools
|
||||||
|
|
||||||
|
#### `/services/run` - Runtime Data
|
||||||
|
Directory for storing runtime data and logs generated by services during container operation. This provides a centralized location for monitoring service activity and troubleshooting issues that occur during normal operation.
|
||||||
|
|
||||||
|
- `logs/` - Service runtime log files
|
||||||
|
|
||||||
|
#### Service Control Scripts
|
||||||
|
Scripts that start and manage the core services required for NetAlertX operation. These scripts handle the initialization of the web server, application server, task scheduler, and backend processing components that work together to provide network monitoring functionality.
|
||||||
|
|
||||||
|
- `start-backend.sh` - Launches Python backend service
|
||||||
|
- `start-crond.sh` - Starts task scheduler
|
||||||
|
- `start-nginx.sh` - Starts web server
|
||||||
|
- `start-php-fpm.sh` - Starts PHP processor
|
||||||
|
- `healthcheck.sh` - Container health verification
|
||||||
|
- `cron_script.sh` - Scheduled task definitions
|
||||||
|
|
||||||
|
### `/entrypoint.sh` - Container Startup Script
|
||||||
|
The main orchestration script that runs when the container starts. It coordinates the entire container initialization process, from pre-startup validation through service startup and ongoing monitoring, ensuring NetAlertX operates reliably in production environments.
|
||||||
|
|
||||||
|
The main script that runs when the container starts:
|
||||||
|
- Runs all pre-startup checks from `/services/scripts`
|
||||||
|
- Creates necessary directories and files
|
||||||
|
- Starts all required services (crond, PHP-FPM, nginx, Python backend)
|
||||||
|
- Monitors services and handles failures
|
||||||
|
- Ensures clean shutdown on container stop
|
||||||
|
|
||||||
|
## Security Considerations
|
||||||
|
|
||||||
|
- Application code is read-only to prevent modifications
|
||||||
|
- Services run with minimal required permissions
|
||||||
|
- Configurations are separated from code
|
||||||
|
- Pre-startup checks verify system integrity
|
||||||
|
- Runtime data is isolated in dedicated directories
|
||||||
|
- Container exits immediately if any service fails (enables restart policies)
|
||||||
108
install/production-filesystem/app/config/app.conf
Executable file
108
install/production-filesystem/app/config/app.conf
Executable file
@@ -0,0 +1,108 @@
|
|||||||
|
#-----------------AUTOGENERATED FILE-----------------#
|
||||||
|
# #
|
||||||
|
# Generated: 2022-12-30_22-19-40 #
|
||||||
|
# #
|
||||||
|
# Config file for the LAN intruder detection app: #
|
||||||
|
# https://github.com/jokob-sk/NetAlertX #
|
||||||
|
# #
|
||||||
|
#-----------------AUTOGENERATED FILE-----------------#
|
||||||
|
|
||||||
|
# 🔺 Use the Settings UI - only edit when necessary 🔺
|
||||||
|
|
||||||
|
# General
|
||||||
|
#---------------------------
|
||||||
|
# Scan using interface eth0
|
||||||
|
# SCAN_SUBNETS = ['192.168.1.0/24 --interface=eth0']
|
||||||
|
#
|
||||||
|
# Scan multiple interfaces (eth1 and eth0):
|
||||||
|
# SCAN_SUBNETS = [ '192.168.1.0/24 --interface=eth1', '192.168.1.0/24 --interface=eth0' ]
|
||||||
|
|
||||||
|
DISCOVER_PLUGINS=True
|
||||||
|
SCAN_SUBNETS=['--localnet']
|
||||||
|
TIMEZONE='Europe/Berlin'
|
||||||
|
LOADED_PLUGINS=['ARPSCAN', 'AVAHISCAN', 'CSVBCKP','DBCLNP', 'DIGSCAN', 'INTRNT', 'MAINT', 'NEWDEV', 'NBTSCAN', 'NSLOOKUP','NTFPRCS', 'SETPWD', 'SMTP', 'SYNC', 'VNDRPDT', 'WORKFLOWS', 'UI']
|
||||||
|
|
||||||
|
DAYS_TO_KEEP_EVENTS=90
|
||||||
|
# Used for generating links in emails. Make sure not to add a trailing slash!
|
||||||
|
REPORT_DASHBOARD_URL='update_REPORT_DASHBOARD_URL_setting'
|
||||||
|
|
||||||
|
# Make sure at least these scanners are enabled for new installs, other defaults are taken from the config.json
|
||||||
|
INTRNT_RUN='schedule'
|
||||||
|
ARPSCAN_RUN='schedule'
|
||||||
|
NSLOOKUP_RUN='before_name_updates'
|
||||||
|
AVAHISCAN_RUN='before_name_updates'
|
||||||
|
NBTSCAN_RUN='before_name_updates'
|
||||||
|
|
||||||
|
# Email
|
||||||
|
#-------------------------------------
|
||||||
|
# (add SMTP to LOADED_PLUGINS to load)
|
||||||
|
#-------------------------------------
|
||||||
|
SMTP_RUN='disabled' # use 'on_notification' to enable
|
||||||
|
SMTP_SERVER='smtp.gmail.com'
|
||||||
|
SMTP_PORT=587
|
||||||
|
SMTP_REPORT_TO='user@gmail.com'
|
||||||
|
SMTP_REPORT_FROM='NetAlertX <user@gmail.com>'
|
||||||
|
SMTP_SKIP_LOGIN=False
|
||||||
|
SMTP_USER='user@gmail.com'
|
||||||
|
SMTP_PASS='password'
|
||||||
|
SMTP_SKIP_TLS=False
|
||||||
|
|
||||||
|
|
||||||
|
# Webhook
|
||||||
|
#-------------------------------------
|
||||||
|
# (add WEBHOOK to LOADED_PLUGINS to load)
|
||||||
|
#-------------------------------------
|
||||||
|
WEBHOOK_RUN='disabled' # use 'on_notification' to enable
|
||||||
|
WEBHOOK_URL='http://n8n.local:5555/webhook-test/aaaaaaaa-aaaa-aaaa-aaaaa-aaaaaaaaaaaa'
|
||||||
|
WEBHOOK_PAYLOAD='json' # webhook payload data format for the "body > attachements > text" attribute
|
||||||
|
# in https://github.com/jokob-sk/NetAlertX/blob/main/docs/webhook_json_sample.json
|
||||||
|
# supported values: 'json', 'html' or 'text'
|
||||||
|
# e.g.: for discord use 'html'
|
||||||
|
WEBHOOK_REQUEST_METHOD='GET'
|
||||||
|
|
||||||
|
|
||||||
|
# Apprise
|
||||||
|
#-------------------------------------
|
||||||
|
# (add APPRISE to LOADED_PLUGINS to load)
|
||||||
|
#-------------------------------------
|
||||||
|
APPRISE_RUN='disabled' # use 'on_notification' to enable
|
||||||
|
APPRISE_HOST='http://localhost:8000/notify'
|
||||||
|
APPRISE_URL='mailto://smtp-relay.sendinblue.com:587?from=user@gmail.com&name=apprise&user=user@gmail.com&pass=password&to=user@gmail.com'
|
||||||
|
|
||||||
|
|
||||||
|
# NTFY
|
||||||
|
#-------------------------------------
|
||||||
|
# (add NTFY to LOADED_PLUGINS to load)
|
||||||
|
#-------------------------------------
|
||||||
|
NTFY_RUN='disabled' # use 'on_notification' to enable
|
||||||
|
NTFY_HOST='https://ntfy.sh'
|
||||||
|
NTFY_TOPIC='replace_my_secure_topicname_91h889f28'
|
||||||
|
NTFY_USER='user'
|
||||||
|
NTFY_PASSWORD='passw0rd'
|
||||||
|
|
||||||
|
|
||||||
|
# PUSHSAFER
|
||||||
|
#-------------------------------------
|
||||||
|
# (add PUSHSAFER to LOADED_PLUGINS to load)
|
||||||
|
#-------------------------------------
|
||||||
|
PUSHSAFER_RUN='disabled' # use 'on_notification' to enable
|
||||||
|
PUSHSAFER_TOKEN='ApiKey'
|
||||||
|
|
||||||
|
|
||||||
|
# MQTT
|
||||||
|
#-------------------------------------
|
||||||
|
# (add MQTT to LOADED_PLUGINS to load)
|
||||||
|
#-------------------------------------
|
||||||
|
MQTT_RUN='disabled' # use 'on_notification' to enable
|
||||||
|
MQTT_BROKER='192.168.1.2'
|
||||||
|
MQTT_PORT=1883
|
||||||
|
MQTT_USER='mqtt'
|
||||||
|
MQTT_PASSWORD='passw0rd'
|
||||||
|
MQTT_QOS=0
|
||||||
|
MQTT_DELAY_SEC=2
|
||||||
|
|
||||||
|
|
||||||
|
#-------------------IMPORTANT INFO-------------------#
|
||||||
|
# This file is ingested by a python script, so if #
|
||||||
|
# modified it needs to use python syntax #
|
||||||
|
#-------------------IMPORTANT INFO-------------------#
|
||||||
BIN
install/production-filesystem/app/db/app.db
Executable file
BIN
install/production-filesystem/app/db/app.db
Executable file
Binary file not shown.
0
api/.git-placeholder → install/production-filesystem/app/log/IP_changes.log
Executable file → Normal file
0
api/.git-placeholder → install/production-filesystem/app/log/IP_changes.log
Executable file → Normal file
0
log/plugins/.git-placeholder → install/production-filesystem/app/log/app.log
Executable file → Normal file
0
log/plugins/.git-placeholder → install/production-filesystem/app/log/app.log
Executable file → Normal file
0
install/production-filesystem/app/log/app_front.log
Normal file
0
install/production-filesystem/app/log/app_front.log
Normal file
0
install/production-filesystem/app/log/crond.log
Normal file
0
install/production-filesystem/app/log/crond.log
Normal file
0
install/production-filesystem/app/log/stderr.log
Normal file
0
install/production-filesystem/app/log/stderr.log
Normal file
0
install/production-filesystem/app/log/stdout.log
Normal file
0
install/production-filesystem/app/log/stdout.log
Normal file
4
install/production-filesystem/build/init-backend.sh
Normal file
4
install/production-filesystem/build/init-backend.sh
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
echo "Initializing backend..."
|
||||||
|
# Future backend initialization steps can go here.
|
||||||
|
echo "Backend initialized."
|
||||||
4
install/production-filesystem/build/init-crond.sh
Normal file
4
install/production-filesystem/build/init-crond.sh
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
echo "Initializing crond..."
|
||||||
|
#Future crond initializations can go here.
|
||||||
|
echo "crond initialized."
|
||||||
4
install/production-filesystem/build/init-nginx.sh
Normal file
4
install/production-filesystem/build/init-nginx.sh
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
echo "Initializing nginx..."
|
||||||
|
install -d -o netalertx -g netalertx -m 700 ${SYSTEM_SERVICES_RUN_TMP}/client_body;
|
||||||
|
echo "nginx initialized."
|
||||||
7
install/production-filesystem/build/init-php-fpm.sh
Normal file
7
install/production-filesystem/build/init-php-fpm.sh
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
echo "Initializing php-fpm..."
|
||||||
|
# Set up PHP-FPM directories and socket configuration
|
||||||
|
install -d -o netalertx -g netalertx /services/config/run
|
||||||
|
|
||||||
|
|
||||||
|
echo "php-fpm initialized."
|
||||||
260
install/production-filesystem/entrypoint.sh
Normal file
260
install/production-filesystem/entrypoint.sh
Normal file
@@ -0,0 +1,260 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# NetAlertX Container Entrypoint
|
||||||
|
################################################################################
|
||||||
|
#
|
||||||
|
# Purpose: Main entrypoint script for NetAlertX Docker containers
|
||||||
|
#
|
||||||
|
# Responsibilities:
|
||||||
|
# 1. Display NetAlertX banner and container startup info
|
||||||
|
# 2. Run pre-startup health checks
|
||||||
|
# 3. Initialize required directories and log files
|
||||||
|
# 4. Start and monitor core services (crond, php-fpm, nginx, Python backend)
|
||||||
|
# 5. Handle service failures and graceful shutdown
|
||||||
|
# 6. Manage process signals (INT, TERM) for clean container termination
|
||||||
|
#
|
||||||
|
# Environment Variables:
|
||||||
|
# - ENVIRONMENT: Container environment type (debian or alpine). If not "debian",
|
||||||
|
# crond scheduler service will be started.
|
||||||
|
# - NETALERTX_DEBUG: If set to 1, services won't auto-shutdown on failure;
|
||||||
|
# container will wait for all to exit naturally (development mode).
|
||||||
|
# - NETALERTX_PLUGINS_LOG: Directory path for plugin logs
|
||||||
|
# - SYSTEM_SERVICES_RUN_LOG: Directory path for service runtime logs
|
||||||
|
# - SYSTEM_SERVICES_RUN_TMP: Directory path for service temporary files
|
||||||
|
# - LOG_DB_IS_LOCKED: File path for database lock status
|
||||||
|
# - LOG_EXECUTION_QUEUE: File path for execution queue log
|
||||||
|
#
|
||||||
|
# Exit Codes:
|
||||||
|
# - 0: Graceful shutdown (unlikely in production)
|
||||||
|
# - 143: Caught signal (INT/TERM)
|
||||||
|
# - Non-zero: Service failure status code
|
||||||
|
#
|
||||||
|
# Service Monitoring:
|
||||||
|
# In production mode (NETALERTX_DEBUG != 1), if any service exits, all services
|
||||||
|
# are terminated and the container exits with the failed service's status code.
|
||||||
|
# This ensures container restart policies can properly reinitialize the stack.
|
||||||
|
#
|
||||||
|
################################################################################
|
||||||
|
|
||||||
|
# Banner display
|
||||||
|
printf '
|
||||||
|
\033[1;31m
|
||||||
|
_ _ _ ___ _ _ __ __
|
||||||
|
| \ | | | | / _ \| | | | \ \ / /
|
||||||
|
| \| | ___| |_/ /_\ \ | ___ _ __| |_ \ V /
|
||||||
|
| . |/ _ \ __| _ | |/ _ \ __| __|/ \
|
||||||
|
| |\ | __/ |_| | | | | __/ | | |_/ /^\ \
|
||||||
|
\_| \_/\___|\__\_| |_/_|\___|_| \__\/ \/
|
||||||
|
\033[0m
|
||||||
|
Network intruder and presence detector.
|
||||||
|
https://netalertx.com
|
||||||
|
'
|
||||||
|
|
||||||
|
set -u
|
||||||
|
|
||||||
|
# Run all pre-startup checks to validate container environment and dependencies
|
||||||
|
for script in ${SYSTEM_SERVICES_SCRIPTS}/check-*.sh; do
|
||||||
|
sh "$script"
|
||||||
|
done
|
||||||
|
|
||||||
|
# Update vendor data (MAC address OUI database) in the background
|
||||||
|
# This happens concurrently with service startup to avoid blocking container readiness
|
||||||
|
${SYSTEM_SERVICES_SCRIPTS}/update_vendors.sh &
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Service management state variables
|
||||||
|
SERVICES="" # Space-separated list of active services in format "pid:name"
|
||||||
|
FAILED_NAME="" # Name of service that failed (used for error reporting)
|
||||||
|
FAILED_STATUS=0 # Exit status code from failed service or signal
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# is_pid_active() - Check if a process is alive and not in zombie/dead state
|
||||||
|
################################################################################
|
||||||
|
# Arguments:
|
||||||
|
# $1: Process ID to check
|
||||||
|
# Returns:
|
||||||
|
# 0 (success): Process is alive and healthy
|
||||||
|
# 1 (failure): Process is dead, zombie, or PID is empty
|
||||||
|
################################################################################
|
||||||
|
is_pid_active() {
|
||||||
|
pid="$1"
|
||||||
|
[ -z "${pid}" ] && return 1
|
||||||
|
|
||||||
|
if ! kill -0 "${pid}" 2>/dev/null; then
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -r "/proc/${pid}/status" ]; then
|
||||||
|
state_line=$(grep '^State:' "/proc/${pid}/status" 2>/dev/null || true)
|
||||||
|
case "${state_line}" in
|
||||||
|
*"(zombie)"*|*"(dead)"*)
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
add_service() {
|
||||||
|
# Start a new service script and track it for monitoring
|
||||||
|
# Arguments:
|
||||||
|
# $1: Path to service startup script (e.g., /services/start-backend.sh)
|
||||||
|
# $2: Human-readable service name (for logging and error reporting)
|
||||||
|
script="$1"
|
||||||
|
name="$2"
|
||||||
|
"$script" &
|
||||||
|
pid=$!
|
||||||
|
SERVICES="${SERVICES} ${pid}:${name}"
|
||||||
|
}
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# remove_service() - Remove a service from the active services list
|
||||||
|
################################################################################
|
||||||
|
# Arguments:
|
||||||
|
# $1: Process ID to remove
|
||||||
|
# Updates: SERVICES variable to exclude the specified PID
|
||||||
|
################################################################################
|
||||||
|
remove_service() {
|
||||||
|
target_pid="$1"
|
||||||
|
updated=""
|
||||||
|
for entry in ${SERVICES}; do
|
||||||
|
pid="${entry%%:*}"
|
||||||
|
[ -z "${pid}" ] && continue
|
||||||
|
[ "${pid}" = "${target_pid}" ] && continue
|
||||||
|
updated="${updated} ${entry}"
|
||||||
|
done
|
||||||
|
SERVICES="${updated}"
|
||||||
|
}
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# shutdown_services() - Gracefully stop all active services
|
||||||
|
################################################################################
|
||||||
|
# Process:
|
||||||
|
# 1. Send SIGTERM to all active services (time to clean up)
|
||||||
|
# 2. Wait for all services to fully terminate
|
||||||
|
# Notes:
|
||||||
|
# - Tolerates services that are already dead
|
||||||
|
# - Uses 'wait' to reap zombie processes
|
||||||
|
################################################################################
|
||||||
|
shutdown_services() {
|
||||||
|
for entry in ${SERVICES}; do
|
||||||
|
pid="${entry%%:*}"
|
||||||
|
[ -z "${pid}" ] && continue
|
||||||
|
if is_pid_active "${pid}"; then
|
||||||
|
kill "${pid}" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
for entry in ${SERVICES}; do
|
||||||
|
pid="${entry%%:*}"
|
||||||
|
[ -z "${pid}" ] && continue
|
||||||
|
wait "${pid}" 2>/dev/null || true
|
||||||
|
done
|
||||||
|
echo "All services stopped."
|
||||||
|
}
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# handle_exit() - Terminate all services and exit container
|
||||||
|
################################################################################
|
||||||
|
# Process:
|
||||||
|
# 1. Log failure information if a service exited abnormally
|
||||||
|
# 2. Shut down all remaining services gracefully
|
||||||
|
# 3. Exit container with recorded status code
|
||||||
|
# Note: Used when a monitored service fails or signal is caught
|
||||||
|
################################################################################
|
||||||
|
handle_exit() {
|
||||||
|
if [ -n "${FAILED_NAME}" ]; then
|
||||||
|
echo "Service ${FAILED_NAME} exited with status ${FAILED_STATUS}."
|
||||||
|
fi
|
||||||
|
shutdown_services
|
||||||
|
exit "${FAILED_STATUS}"
|
||||||
|
}
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# on_signal() - Handle container signals (INT, TERM) for graceful shutdown
|
||||||
|
################################################################################
|
||||||
|
# Signals handled: SIGINT (Ctrl+C), SIGTERM (docker stop)
|
||||||
|
# Process:
|
||||||
|
# 1. Set exit status to 143 (128 + 15, standard SIGTERM code)
|
||||||
|
# 2. Trigger full shutdown sequence
|
||||||
|
# Note: Registered via 'trap' command below
|
||||||
|
################################################################################
|
||||||
|
on_signal() {
|
||||||
|
echo "Caught signal, shutting down services..."
|
||||||
|
FAILED_NAME="signal"
|
||||||
|
FAILED_STATUS=143
|
||||||
|
handle_exit
|
||||||
|
}
|
||||||
|
|
||||||
|
# Register signal handlers for graceful shutdown
|
||||||
|
trap on_signal INT TERM
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# Service Startup Section
|
||||||
|
################################################################################
|
||||||
|
# Start services based on environment configuration
|
||||||
|
|
||||||
|
# Only start crond scheduler on Alpine (non-Debian) environments
|
||||||
|
# Debian typically uses systemd or other schedulers
|
||||||
|
if [ "${ENVIRONMENT:-}" ] && [ "${ENVIRONMENT:-}" != "debian" ]; then
|
||||||
|
add_service "/services/start-crond.sh" "crond"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Start core frontend and backend services
|
||||||
|
# Order: web server, application server, then Python backend
|
||||||
|
add_service "${SYSTEM_SERVICES}/start-php-fpm.sh" "php-fpm83"
|
||||||
|
add_service "${SYSTEM_SERVICES}/start-nginx.sh" "nginx"
|
||||||
|
add_service "${SYSTEM_SERVICES}/start-backend.sh" "python3"
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# Development Mode Debug Switch
|
||||||
|
################################################################################
|
||||||
|
# If NETALERTX_DEBUG=1, skip automatic service restart on failure
|
||||||
|
# Useful for devcontainer debugging where individual services need to be debugged
|
||||||
|
if [ "${NETALERTX_DEBUG:-0}" -eq 1 ]; then
|
||||||
|
echo "NETALERTX_DEBUG is set to 1, will not shut down other services if one fails."
|
||||||
|
wait
|
||||||
|
exit $?
|
||||||
|
fi
|
||||||
|
|
||||||
|
################################################################################
|
||||||
|
# Service Monitoring Loop (Production Mode)
|
||||||
|
################################################################################
|
||||||
|
# Behavior depends on NETALERTX_DEBUG setting:
|
||||||
|
# - DEBUG OFF (production): Any service failure triggers full container restart
|
||||||
|
# - DEBUG ON: Services can fail individually; container waits for natural exit
|
||||||
|
#
|
||||||
|
# Loop Process:
|
||||||
|
# 1. Check each active service every 10 seconds
|
||||||
|
# 2. If service is not active, wait for it and capture exit status
|
||||||
|
# 3. Log failure and terminate all other services
|
||||||
|
# 4. Exit container with failed service's status code
|
||||||
|
# 5. This enables Docker restart policies to reinitialize the stack
|
||||||
|
################################################################################
|
||||||
|
while [ -n "${SERVICES}" ]; do
|
||||||
|
for entry in ${SERVICES}; do
|
||||||
|
pid="${entry%%:*}"
|
||||||
|
name="${entry#*:}"
|
||||||
|
[ -z "${pid}" ] && continue
|
||||||
|
if ! is_pid_active "${pid}"; then
|
||||||
|
wait "${pid}" 2>/dev/null
|
||||||
|
status=$?
|
||||||
|
FAILED_STATUS=$status
|
||||||
|
FAILED_NAME="${name}"
|
||||||
|
remove_service "${pid}"
|
||||||
|
handle_exit
|
||||||
|
fi
|
||||||
|
|
||||||
|
done
|
||||||
|
sleep 10
|
||||||
|
done
|
||||||
|
|
||||||
|
# If we exit the loop with no service failures, set status to 1 (error)
|
||||||
|
# This should not happen in normal operation
|
||||||
|
if [ "${FAILED_STATUS}" -eq 0 ] && [ "${FAILED_NAME}" != "signal" ]; then
|
||||||
|
FAILED_STATUS=1
|
||||||
|
fi
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
|
||||||
|
-----BEGIN CERTIFICATE-----
|
||||||
|
MIICOjCCAcCgAwIBAgIUI0Tu7zsrBJACQIZgLMJobtbdNn4wCgYIKoZIzj0EAwIw
|
||||||
|
TDELMAkGA1UEBhMCSVQxDjAMBgNVBAgMBUl0YWx5MQ4wDAYDVQQKDAVJbGlhZDEd
|
||||||
|
MBsGA1UEAwwUSWxpYWRib3ggRUNDIFJvb3QgQ0EwHhcNMjAxMTI3MDkzODEzWhcN
|
||||||
|
NDAxMTIyMDkzODEzWjBMMQswCQYDVQQGEwJJVDEOMAwGA1UECAwFSXRhbHkxDjAM
|
||||||
|
BgNVBAoMBUlsaWFkMR0wGwYDVQQDDBRJbGlhZGJveCBFQ0MgUm9vdCBDQTB2MBAG
|
||||||
|
ByqGSM49AgEGBSuBBAAiA2IABMryJyb2loHNAioY8IztN5MI3UgbVHVP/vZwcnre
|
||||||
|
ZvJOyDvE4HJgIti5qmfswlnMzpNbwf/MkT+7HAU8jJoTorRm1wtAnQ9cWD3Ebv79
|
||||||
|
RPwtjjy3Bza3SgdVxmd6fWPUKaNjMGEwHQYDVR0OBBYEFDUij/4lpoJ+kOXRyrcM
|
||||||
|
jf2RPzOqMB8GA1UdIwQYMBaAFDUij/4lpoJ+kOXRyrcMjf2RPzOqMA8GA1UdEwEB
|
||||||
|
/wQFMAMBAf8wDgYDVR0PAQH/BAQDAgGGMAoGCCqGSM49BAMCA2gAMGUCMQC6eUV1
|
||||||
|
pFh4UpJOTc1JToztN4ttnQR6rIzxMZ6mNCe+nhjkohWp24pr7BpUYSbEizYCMAQ6
|
||||||
|
LCiBKV2j7QQGy7N1aBmdur17ZepYzR1YV0eI+Kd978aZggsmhjXENQYVTmm/XA==
|
||||||
|
-----END CERTIFICATE-----
|
||||||
4
install/production-filesystem/services/config/crond/netalertx
Executable file
4
install/production-filesystem/services/config/crond/netalertx
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
# Every minute check for cron jobs
|
||||||
|
* * * * * /services/scripts/cron_script.sh
|
||||||
|
# Update vendors 4x/d
|
||||||
|
0 */6 * * * /services/scripts/update_vendors.sh
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
Nginx's conf is in /services/config/nginx/conf.active. This is the default configuration when run as a read-only container without a mount.
|
||||||
|
|
||||||
|
With a tmpfs mount on /services/config/nginx/conf.active, the nginx template will be rewritten to allow ENV customization of listen address and port.
|
||||||
|
|
||||||
|
The act of running /services/start-nginx.sh writes a new nginx.conf file, using envsubst, then starts nginx based on the parameters in that file.
|
||||||
|
|
||||||
|
Defaults:
|
||||||
|
LISTEN_ADDR=0.0.0.0
|
||||||
|
PORT=20211
|
||||||
@@ -0,0 +1,121 @@
|
|||||||
|
# Set number of worker processes automatically based on number of CPU cores.
|
||||||
|
worker_processes auto;
|
||||||
|
|
||||||
|
# Enables the use of JIT for regular expressions to speed-up their processing.
|
||||||
|
pcre_jit on;
|
||||||
|
|
||||||
|
# Configures default error logger.
|
||||||
|
error_log /app/log/nginx-error.log warn;
|
||||||
|
|
||||||
|
events {
|
||||||
|
# The maximum number of simultaneous connections that can be opened by
|
||||||
|
# a worker process.
|
||||||
|
worker_connections 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
http {
|
||||||
|
|
||||||
|
# Mapping of temp paths for various nginx modules.
|
||||||
|
client_body_temp_path /services/run/tmp/client_body;
|
||||||
|
proxy_temp_path /services/run/tmp/proxy;
|
||||||
|
fastcgi_temp_path /services/run/tmp/fastcgi;
|
||||||
|
uwsgi_temp_path /services/run/tmp/uwsgi;
|
||||||
|
scgi_temp_path /services/run/tmp/scgi;
|
||||||
|
|
||||||
|
# Includes mapping of file name extensions to MIME types of responses
|
||||||
|
# and defines the default type.
|
||||||
|
include /services/config/nginx/mime.types;
|
||||||
|
default_type application/octet-stream;
|
||||||
|
|
||||||
|
# Name servers used to resolve names of upstream servers into addresses.
|
||||||
|
# It's also needed when using tcpsocket and udpsocket in Lua modules.
|
||||||
|
#resolver 1.1.1.1 1.0.0.1 [2606:4700:4700::1111] [2606:4700:4700::1001];
|
||||||
|
|
||||||
|
# Don't tell nginx version to the clients. Default is 'on'.
|
||||||
|
server_tokens off;
|
||||||
|
|
||||||
|
# Specifies the maximum accepted body size of a client request, as
|
||||||
|
# indicated by the request header Content-Length. If the stated content
|
||||||
|
# length is greater than this size, then the client receives the HTTP
|
||||||
|
# error code 413. Set to 0 to disable. Default is '1m'.
|
||||||
|
client_max_body_size 1m;
|
||||||
|
|
||||||
|
# Sendfile copies data between one FD and other from within the kernel,
|
||||||
|
# which is more efficient than read() + write(). Default is off.
|
||||||
|
sendfile on;
|
||||||
|
|
||||||
|
# Causes nginx to attempt to send its HTTP response head in one packet,
|
||||||
|
# instead of using partial frames. Default is 'off'.
|
||||||
|
tcp_nopush on;
|
||||||
|
|
||||||
|
|
||||||
|
# Enables the specified protocols. Default is TLSv1 TLSv1.1 TLSv1.2.
|
||||||
|
# TIP: If you're not obligated to support ancient clients, remove TLSv1.1.
|
||||||
|
ssl_protocols TLSv1.2 TLSv1.3;
|
||||||
|
|
||||||
|
# Path of the file with Diffie-Hellman parameters for EDH ciphers.
|
||||||
|
# TIP: Generate with: `openssl dhparam -out /etc/ssl/nginx/dh2048.pem 2048`
|
||||||
|
#ssl_dhparam /etc/ssl/nginx/dh2048.pem;
|
||||||
|
|
||||||
|
# Specifies that our cipher suits should be preferred over client ciphers.
|
||||||
|
# Default is 'off'.
|
||||||
|
ssl_prefer_server_ciphers on;
|
||||||
|
|
||||||
|
# Enables a shared SSL cache with size that can hold around 8000 sessions.
|
||||||
|
# Default is 'none'.
|
||||||
|
ssl_session_cache shared:SSL:2m;
|
||||||
|
|
||||||
|
# Specifies a time during which a client may reuse the session parameters.
|
||||||
|
# Default is '5m'.
|
||||||
|
ssl_session_timeout 1h;
|
||||||
|
|
||||||
|
# Disable TLS session tickets (they are insecure). Default is 'on'.
|
||||||
|
ssl_session_tickets off;
|
||||||
|
|
||||||
|
|
||||||
|
# Enable gzipping of responses.
|
||||||
|
gzip on;
|
||||||
|
|
||||||
|
# Set the Vary HTTP header as defined in the RFC 2616. Default is 'off'.
|
||||||
|
gzip_vary on;
|
||||||
|
|
||||||
|
|
||||||
|
# Helper variable for proxying websockets.
|
||||||
|
map $http_upgrade $connection_upgrade {
|
||||||
|
default upgrade;
|
||||||
|
'' close;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# Specifies the main log format.
|
||||||
|
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
|
||||||
|
'$status $body_bytes_sent "$http_referer" '
|
||||||
|
'"$http_user_agent" "$http_x_forwarded_for"';
|
||||||
|
|
||||||
|
# Sets the path, format, and configuration for a buffered log write.
|
||||||
|
access_log /app/log/nginx-access.log main;
|
||||||
|
|
||||||
|
|
||||||
|
# Virtual host config
|
||||||
|
server {
|
||||||
|
listen 0.0.0.0:20211 default_server;
|
||||||
|
root /app/front;
|
||||||
|
index index.php;
|
||||||
|
add_header X-Forwarded-Prefix "/app" always;
|
||||||
|
add_header X-Frame-Options "SAMEORIGIN" always;
|
||||||
|
add_header X-Content-Type-Options "nosniff" always;
|
||||||
|
add_header X-XSS-Protection "1; mode=block" always;
|
||||||
|
location ~ \.php$ {
|
||||||
|
try_files $uri =404;
|
||||||
|
# Set Cache-Control header to prevent caching on the first load
|
||||||
|
add_header Cache-Control "no-store";
|
||||||
|
fastcgi_pass unix:/services/run/php.sock;
|
||||||
|
include /services/config/nginx/fastcgi_params;
|
||||||
|
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
||||||
|
fastcgi_param SCRIPT_NAME $fastcgi_script_name;
|
||||||
|
fastcgi_connect_timeout 75;
|
||||||
|
fastcgi_send_timeout 600;
|
||||||
|
fastcgi_read_timeout 600;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
|
||||||
|
fastcgi_param QUERY_STRING $query_string;
|
||||||
|
fastcgi_param REQUEST_METHOD $request_method;
|
||||||
|
fastcgi_param CONTENT_TYPE $content_type;
|
||||||
|
fastcgi_param CONTENT_LENGTH $content_length;
|
||||||
|
|
||||||
|
fastcgi_param SCRIPT_NAME $fastcgi_script_name;
|
||||||
|
fastcgi_param REQUEST_URI $request_uri;
|
||||||
|
fastcgi_param DOCUMENT_URI $document_uri;
|
||||||
|
fastcgi_param DOCUMENT_ROOT $document_root;
|
||||||
|
fastcgi_param SERVER_PROTOCOL $server_protocol;
|
||||||
|
fastcgi_param REQUEST_SCHEME $scheme;
|
||||||
|
fastcgi_param HTTPS $https if_not_empty;
|
||||||
|
|
||||||
|
fastcgi_param GATEWAY_INTERFACE CGI/1.1;
|
||||||
|
fastcgi_param SERVER_SOFTWARE nginx/$nginx_version;
|
||||||
|
|
||||||
|
fastcgi_param REMOTE_ADDR $remote_addr;
|
||||||
|
fastcgi_param REMOTE_PORT $remote_port;
|
||||||
|
fastcgi_param SERVER_ADDR $server_addr;
|
||||||
|
fastcgi_param SERVER_PORT $server_port;
|
||||||
|
fastcgi_param SERVER_NAME $server_name;
|
||||||
@@ -0,0 +1,98 @@
|
|||||||
|
types {
|
||||||
|
text/html html htm shtml;
|
||||||
|
text/css css;
|
||||||
|
text/xml xml;
|
||||||
|
image/gif gif;
|
||||||
|
image/jpeg jpeg jpg;
|
||||||
|
application/javascript js;
|
||||||
|
application/atom+xml atom;
|
||||||
|
application/rss+xml rss;
|
||||||
|
|
||||||
|
text/mathml mml;
|
||||||
|
text/plain txt;
|
||||||
|
text/vnd.sun.j2me.app-descriptor jad;
|
||||||
|
text/vnd.wap.wml wml;
|
||||||
|
text/x-component htc;
|
||||||
|
|
||||||
|
image/avif avif;
|
||||||
|
image/png png;
|
||||||
|
image/svg+xml svg svgz;
|
||||||
|
image/tiff tif tiff;
|
||||||
|
image/vnd.wap.wbmp wbmp;
|
||||||
|
image/webp webp;
|
||||||
|
image/x-icon ico;
|
||||||
|
image/x-jng jng;
|
||||||
|
image/x-ms-bmp bmp;
|
||||||
|
|
||||||
|
font/woff woff;
|
||||||
|
font/woff2 woff2;
|
||||||
|
|
||||||
|
application/java-archive jar war ear;
|
||||||
|
application/json json;
|
||||||
|
application/mac-binhex40 hqx;
|
||||||
|
application/msword doc;
|
||||||
|
application/pdf pdf;
|
||||||
|
application/postscript ps eps ai;
|
||||||
|
application/rtf rtf;
|
||||||
|
application/vnd.apple.mpegurl m3u8;
|
||||||
|
application/vnd.google-earth.kml+xml kml;
|
||||||
|
application/vnd.google-earth.kmz kmz;
|
||||||
|
application/vnd.ms-excel xls;
|
||||||
|
application/vnd.ms-fontobject eot;
|
||||||
|
application/vnd.ms-powerpoint ppt;
|
||||||
|
application/vnd.oasis.opendocument.graphics odg;
|
||||||
|
application/vnd.oasis.opendocument.presentation odp;
|
||||||
|
application/vnd.oasis.opendocument.spreadsheet ods;
|
||||||
|
application/vnd.oasis.opendocument.text odt;
|
||||||
|
application/vnd.openxmlformats-officedocument.presentationml.presentation
|
||||||
|
pptx;
|
||||||
|
application/vnd.openxmlformats-officedocument.spreadsheetml.sheet
|
||||||
|
xlsx;
|
||||||
|
application/vnd.openxmlformats-officedocument.wordprocessingml.document
|
||||||
|
docx;
|
||||||
|
application/vnd.wap.wmlc wmlc;
|
||||||
|
application/wasm wasm;
|
||||||
|
application/x-7z-compressed 7z;
|
||||||
|
application/x-cocoa cco;
|
||||||
|
application/x-java-archive-diff jardiff;
|
||||||
|
application/x-java-jnlp-file jnlp;
|
||||||
|
application/x-makeself run;
|
||||||
|
application/x-perl pl pm;
|
||||||
|
application/x-pilot prc pdb;
|
||||||
|
application/x-rar-compressed rar;
|
||||||
|
application/x-redhat-package-manager rpm;
|
||||||
|
application/x-sea sea;
|
||||||
|
application/x-shockwave-flash swf;
|
||||||
|
application/x-stuffit sit;
|
||||||
|
application/x-tcl tcl tk;
|
||||||
|
application/x-x509-ca-cert der pem crt;
|
||||||
|
application/x-xpinstall xpi;
|
||||||
|
application/xhtml+xml xhtml;
|
||||||
|
application/xspf+xml xspf;
|
||||||
|
application/zip zip;
|
||||||
|
|
||||||
|
application/octet-stream bin exe dll;
|
||||||
|
application/octet-stream deb;
|
||||||
|
application/octet-stream dmg;
|
||||||
|
application/octet-stream iso img;
|
||||||
|
application/octet-stream msi msp msm;
|
||||||
|
|
||||||
|
audio/midi mid midi kar;
|
||||||
|
audio/mpeg mp3;
|
||||||
|
audio/ogg ogg;
|
||||||
|
audio/x-m4a m4a;
|
||||||
|
audio/x-realaudio ra;
|
||||||
|
|
||||||
|
video/3gpp 3gpp 3gp;
|
||||||
|
video/mp2t ts;
|
||||||
|
video/mp4 mp4;
|
||||||
|
video/mpeg mpeg mpg;
|
||||||
|
video/quicktime mov;
|
||||||
|
video/webm webm;
|
||||||
|
video/x-flv flv;
|
||||||
|
video/x-m4v m4v;
|
||||||
|
video/x-mng mng;
|
||||||
|
video/x-ms-asf asx asf;
|
||||||
|
video/x-ms-wmv wmv;
|
||||||
|
video/x-msvideo avi;
|
||||||
|
}
|
||||||
@@ -0,0 +1,113 @@
|
|||||||
|
# Set number of worker processes automatically based on number of CPU cores.
|
||||||
|
worker_processes auto;
|
||||||
|
|
||||||
|
# Enables the use of JIT for regular expressions to speed-up their processing.
|
||||||
|
pcre_jit on;
|
||||||
|
|
||||||
|
# Configures default error logger.
|
||||||
|
error_log /app/log/nginx-error.log warn;
|
||||||
|
|
||||||
|
events {
|
||||||
|
# The maximum number of simultaneous connections that can be opened by
|
||||||
|
# a worker process.
|
||||||
|
worker_connections 1024;
|
||||||
|
}
|
||||||
|
|
||||||
|
http {
|
||||||
|
|
||||||
|
# Mapping of temp paths for various nginx modules.
|
||||||
|
client_body_temp_path /services/run/tmp/client_body;
|
||||||
|
proxy_temp_path /services/run/tmp/proxy;
|
||||||
|
fastcgi_temp_path /services/run/tmp/fastcgi;
|
||||||
|
uwsgi_temp_path /services/run/tmp/uwsgi;
|
||||||
|
scgi_temp_path /services/run/tmp/scgi;
|
||||||
|
|
||||||
|
# Includes mapping of file name extensions to MIME types of responses
|
||||||
|
# and defines the default type.
|
||||||
|
include /services/config/nginx/mime.types;
|
||||||
|
default_type application/octet-stream;
|
||||||
|
|
||||||
|
# Name servers used to resolve names of upstream servers into addresses.
|
||||||
|
# It's also needed when using tcpsocket and udpsocket in Lua modules.
|
||||||
|
#resolver 1.1.1.1 1.0.0.1 [2606:4700:4700::1111] [2606:4700:4700::1001];
|
||||||
|
|
||||||
|
# Don't tell nginx version to the clients. Default is 'on'.
|
||||||
|
server_tokens off;
|
||||||
|
|
||||||
|
# Specifies the maximum accepted body size of a client request, as
|
||||||
|
# indicated by the request header Content-Length. If the stated content
|
||||||
|
# length is greater than this size, then the client receives the HTTP
|
||||||
|
# error code 413. Set to 0 to disable. Default is '1m'.
|
||||||
|
client_max_body_size 1m;
|
||||||
|
|
||||||
|
# Sendfile copies data between one FD and other from within the kernel,
|
||||||
|
# which is more efficient than read() + write(). Default is off.
|
||||||
|
sendfile on;
|
||||||
|
|
||||||
|
# Causes nginx to attempt to send its HTTP response head in one packet,
|
||||||
|
# instead of using partial frames. Default is 'off'.
|
||||||
|
tcp_nopush on;
|
||||||
|
|
||||||
|
|
||||||
|
# Enables the specified protocols. Default is TLSv1 TLSv1.1 TLSv1.2.
|
||||||
|
# TIP: If you're not obligated to support ancient clients, remove TLSv1.1.
|
||||||
|
ssl_protocols TLSv1.2 TLSv1.3;
|
||||||
|
|
||||||
|
# Path of the file with Diffie-Hellman parameters for EDH ciphers.
|
||||||
|
# TIP: Generate with: `openssl dhparam -out /etc/ssl/nginx/dh2048.pem 2048`
|
||||||
|
#ssl_dhparam /etc/ssl/nginx/dh2048.pem;
|
||||||
|
|
||||||
|
# Specifies that our cipher suits should be preferred over client ciphers.
|
||||||
|
# Default is 'off'.
|
||||||
|
ssl_prefer_server_ciphers on;
|
||||||
|
|
||||||
|
# Enables a shared SSL cache with size that can hold around 8000 sessions.
|
||||||
|
# Default is 'none'.
|
||||||
|
ssl_session_cache shared:SSL:2m;
|
||||||
|
|
||||||
|
# Specifies a time during which a client may reuse the session parameters.
|
||||||
|
# Default is '5m'.
|
||||||
|
ssl_session_timeout 1h;
|
||||||
|
|
||||||
|
# Disable TLS session tickets (they are insecure). Default is 'on'.
|
||||||
|
ssl_session_tickets off;
|
||||||
|
|
||||||
|
|
||||||
|
# Enable gzipping of responses.
|
||||||
|
gzip on;
|
||||||
|
|
||||||
|
# Set the Vary HTTP header as defined in the RFC 2616. Default is 'off'.
|
||||||
|
gzip_vary on;
|
||||||
|
|
||||||
|
|
||||||
|
# Specifies the main log format.
|
||||||
|
log_format main '$remote_addr - $remote_user [$time_local] "$request" '
|
||||||
|
'$status $body_bytes_sent "$http_referer" '
|
||||||
|
'"$http_user_agent" "$http_x_forwarded_for"';
|
||||||
|
|
||||||
|
# Sets the path, format, and configuration for a buffered log write.
|
||||||
|
access_log /app/log/nginx-access.log main;
|
||||||
|
|
||||||
|
|
||||||
|
# Virtual host config
|
||||||
|
server {
|
||||||
|
listen ${LISTEN_ADDR}:${PORT} default_server;
|
||||||
|
large_client_header_buffers 4 16k;
|
||||||
|
root /app/front;
|
||||||
|
index index.php;
|
||||||
|
add_header X-Forwarded-Prefix "/app" always;
|
||||||
|
|
||||||
|
|
||||||
|
location ~* \.php$ {
|
||||||
|
# Set Cache-Control header to prevent caching on the first load
|
||||||
|
add_header Cache-Control "no-store";
|
||||||
|
fastcgi_pass unix:/services/run/php.sock;
|
||||||
|
include /services/config/nginx/fastcgi_params;
|
||||||
|
fastcgi_param SCRIPT_FILENAME $document_root$fastcgi_script_name;
|
||||||
|
fastcgi_param SCRIPT_NAME $fastcgi_script_name;
|
||||||
|
fastcgi_connect_timeout 75;
|
||||||
|
fastcgi_send_timeout 600;
|
||||||
|
fastcgi_read_timeout 600;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
; NETALERTX PHP FPM config
|
||||||
|
;
|
||||||
|
; This was created with the following command on a fresh install:
|
||||||
|
; grep -v -e ';' -e '^$' /etc/php83/php-fpm.conf| sed 's/\\n\\n//g'
|
||||||
|
;
|
||||||
|
; error_log and include are modified
|
||||||
|
;
|
||||||
|
|
||||||
|
[global]
|
||||||
|
pid = /services/run/php8.3-fpm.pid
|
||||||
|
error_log = /app/log/app.php_errors.log
|
||||||
|
include=/services/config/php/php-fpm.d/*.conf
|
||||||
@@ -0,0 +1,499 @@
|
|||||||
|
; NetAlertX php-fpm www.conf
|
||||||
|
;
|
||||||
|
; Commented out user/group
|
||||||
|
; No further changes
|
||||||
|
|
||||||
|
; Start a new pool named 'www'.
|
||||||
|
; the variable $pool can be used in any directive and will be replaced by the
|
||||||
|
; pool name ('www' here)
|
||||||
|
[www]
|
||||||
|
|
||||||
|
; Per pool prefix
|
||||||
|
; It only applies on the following directives:
|
||||||
|
; - 'access.log'
|
||||||
|
; - 'slowlog'
|
||||||
|
; - 'listen' (unixsocket)
|
||||||
|
; - 'chroot'
|
||||||
|
; - 'chdir'
|
||||||
|
; - 'php_values'
|
||||||
|
; - 'php_admin_values'
|
||||||
|
; When not set, the global prefix (or /usr) applies instead.
|
||||||
|
; Note: This directive can also be relative to the global prefix.
|
||||||
|
; Default Value: none
|
||||||
|
;prefix = /path/to/pools/$pool
|
||||||
|
|
||||||
|
; Unix user/group of the child processes. This can be used only if the master
|
||||||
|
; process running user is root. It is set after the child process is created.
|
||||||
|
; The user and group can be specified either by their name or by their numeric
|
||||||
|
; IDs.
|
||||||
|
; Note: If the user is root, the executable needs to be started with
|
||||||
|
; --allow-to-run-as-root option to work.
|
||||||
|
; Default Values: The user is set to master process running user by default.
|
||||||
|
; If the group is not set, the user's group is used.
|
||||||
|
; user = nobody
|
||||||
|
; group = nobody
|
||||||
|
|
||||||
|
; The address on which to accept FastCGI requests.
|
||||||
|
; Valid syntaxes are:
|
||||||
|
; 'ip.add.re.ss:port' - to listen on a TCP socket to a specific IPv4 address on
|
||||||
|
; a specific port;
|
||||||
|
; '[ip:6:addr:ess]:port' - to listen on a TCP socket to a specific IPv6 address on
|
||||||
|
; a specific port;
|
||||||
|
; 'port' - to listen on a TCP socket to all addresses
|
||||||
|
; (IPv6 and IPv4-mapped) on a specific port;
|
||||||
|
; '/path/to/unix/socket' - to listen on a unix socket.
|
||||||
|
; Note: This value is mandatory.
|
||||||
|
listen = /services/run/php.sock
|
||||||
|
|
||||||
|
; Set listen(2) backlog.
|
||||||
|
; Default Value: 511 (-1 on Linux, FreeBSD and OpenBSD)
|
||||||
|
;listen.backlog = 511
|
||||||
|
|
||||||
|
; Set permissions for unix socket, if one is used. In Linux, read/write
|
||||||
|
; permissions must be set in order to allow connections from a web server. Many
|
||||||
|
; BSD-derived systems allow connections regardless of permissions. The owner
|
||||||
|
; and group can be specified either by name or by their numeric IDs.
|
||||||
|
; Default Values: Owner is set to the master process running user. If the group
|
||||||
|
; is not set, the owner's group is used. Mode is set to 0660.
|
||||||
|
;listen.owner = nobody
|
||||||
|
;listen.group = nobody
|
||||||
|
;listen.mode = 0660
|
||||||
|
|
||||||
|
; When POSIX Access Control Lists are supported you can set them using
|
||||||
|
; these options, value is a comma separated list of user/group names.
|
||||||
|
; When set, listen.owner and listen.group are ignored
|
||||||
|
;listen.acl_users =
|
||||||
|
;listen.acl_groups =
|
||||||
|
|
||||||
|
; List of addresses (IPv4/IPv6) of FastCGI clients which are allowed to connect.
|
||||||
|
; Equivalent to the FCGI_WEB_SERVER_ADDRS environment variable in the original
|
||||||
|
; PHP FCGI (5.2.2+). Makes sense only with a tcp listening socket. Each address
|
||||||
|
; must be separated by a comma. If this value is left blank, connections will be
|
||||||
|
; accepted from any ip address.
|
||||||
|
; Default Value: any
|
||||||
|
;listen.allowed_clients = 127.0.0.1
|
||||||
|
|
||||||
|
; Set the associated the route table (FIB). FreeBSD only
|
||||||
|
; Default Value: -1
|
||||||
|
;listen.setfib = 1
|
||||||
|
|
||||||
|
; Specify the nice(2) priority to apply to the pool processes (only if set)
|
||||||
|
; The value can vary from -19 (highest priority) to 20 (lower priority)
|
||||||
|
; Note: - It will only work if the FPM master process is launched as root
|
||||||
|
; - The pool processes will inherit the master process priority
|
||||||
|
; unless it specified otherwise
|
||||||
|
; Default Value: no set
|
||||||
|
; process.priority = -19
|
||||||
|
|
||||||
|
; Set the process dumpable flag (PR_SET_DUMPABLE prctl for Linux or
|
||||||
|
; PROC_TRACE_CTL procctl for FreeBSD) even if the process user
|
||||||
|
; or group is different than the master process user. It allows to create process
|
||||||
|
; core dump and ptrace the process for the pool user.
|
||||||
|
; Default Value: no
|
||||||
|
; process.dumpable = yes
|
||||||
|
|
||||||
|
; Choose how the process manager will control the number of child processes.
|
||||||
|
; Possible Values:
|
||||||
|
; static - a fixed number (pm.max_children) of child processes;
|
||||||
|
; dynamic - the number of child processes are set dynamically based on the
|
||||||
|
; following directives. With this process management, there will be
|
||||||
|
; always at least 1 children.
|
||||||
|
; pm.max_children - the maximum number of children that can
|
||||||
|
; be alive at the same time.
|
||||||
|
; pm.start_servers - the number of children created on startup.
|
||||||
|
; pm.min_spare_servers - the minimum number of children in 'idle'
|
||||||
|
; state (waiting to process). If the number
|
||||||
|
; of 'idle' processes is less than this
|
||||||
|
; number then some children will be created.
|
||||||
|
; pm.max_spare_servers - the maximum number of children in 'idle'
|
||||||
|
; state (waiting to process). If the number
|
||||||
|
; of 'idle' processes is greater than this
|
||||||
|
; number then some children will be killed.
|
||||||
|
; pm.max_spawn_rate - the maximum number of rate to spawn child
|
||||||
|
; processes at once.
|
||||||
|
; ondemand - no children are created at startup. Children will be forked when
|
||||||
|
; new requests will connect. The following parameter are used:
|
||||||
|
; pm.max_children - the maximum number of children that
|
||||||
|
; can be alive at the same time.
|
||||||
|
; pm.process_idle_timeout - The number of seconds after which
|
||||||
|
; an idle process will be killed.
|
||||||
|
; Note: This value is mandatory.
|
||||||
|
pm = dynamic
|
||||||
|
|
||||||
|
; The number of child processes to be created when pm is set to 'static' and the
|
||||||
|
; maximum number of child processes when pm is set to 'dynamic' or 'ondemand'.
|
||||||
|
; This value sets the limit on the number of simultaneous requests that will be
|
||||||
|
; served. Equivalent to the ApacheMaxClients directive with mpm_prefork.
|
||||||
|
; Equivalent to the PHP_FCGI_CHILDREN environment variable in the original PHP
|
||||||
|
; CGI. The below defaults are based on a server without much resources. Don't
|
||||||
|
; forget to tweak pm.* to fit your needs.
|
||||||
|
; Note: Used when pm is set to 'static', 'dynamic' or 'ondemand'
|
||||||
|
; Note: This value is mandatory.
|
||||||
|
pm.max_children = 10
|
||||||
|
|
||||||
|
; The number of child processes created on startup.
|
||||||
|
; Note: Used only when pm is set to 'dynamic'
|
||||||
|
; Default Value: (min_spare_servers + max_spare_servers) / 2
|
||||||
|
pm.start_servers = 2
|
||||||
|
|
||||||
|
; The desired minimum number of idle server processes.
|
||||||
|
; Note: Used only when pm is set to 'dynamic'
|
||||||
|
; Note: Mandatory when pm is set to 'dynamic'
|
||||||
|
pm.min_spare_servers = 1
|
||||||
|
|
||||||
|
; The desired maximum number of idle server processes.
|
||||||
|
; Note: Used only when pm is set to 'dynamic'
|
||||||
|
; Note: Mandatory when pm is set to 'dynamic'
|
||||||
|
pm.max_spare_servers = 3
|
||||||
|
|
||||||
|
; The number of rate to spawn child processes at once.
|
||||||
|
; Note: Used only when pm is set to 'dynamic'
|
||||||
|
; Note: Mandatory when pm is set to 'dynamic'
|
||||||
|
; Default Value: 32
|
||||||
|
;pm.max_spawn_rate = 32
|
||||||
|
|
||||||
|
; The number of seconds after which an idle process will be killed.
|
||||||
|
; Note: Used only when pm is set to 'ondemand'
|
||||||
|
; Default Value: 10s
|
||||||
|
;pm.process_idle_timeout = 10s;
|
||||||
|
|
||||||
|
; The number of requests each child process should execute before respawning.
|
||||||
|
; This can be useful to work around memory leaks in 3rd party libraries. For
|
||||||
|
; endless request processing specify '0'. Equivalent to PHP_FCGI_MAX_REQUESTS.
|
||||||
|
; Default Value: 0
|
||||||
|
;pm.max_requests = 500
|
||||||
|
|
||||||
|
; The URI to view the FPM status page. If this value is not set, no URI will be
|
||||||
|
; recognized as a status page. It shows the following information:
|
||||||
|
; pool - the name of the pool;
|
||||||
|
; process manager - static, dynamic or ondemand;
|
||||||
|
; start time - the date and time FPM has started;
|
||||||
|
; start since - number of seconds since FPM has started;
|
||||||
|
; accepted conn - the number of request accepted by the pool;
|
||||||
|
; listen queue - the number of request in the queue of pending
|
||||||
|
; connections (see backlog in listen(2));
|
||||||
|
; max listen queue - the maximum number of requests in the queue
|
||||||
|
; of pending connections since FPM has started;
|
||||||
|
; listen queue len - the size of the socket queue of pending connections;
|
||||||
|
; idle processes - the number of idle processes;
|
||||||
|
; active processes - the number of active processes;
|
||||||
|
; total processes - the number of idle + active processes;
|
||||||
|
; max active processes - the maximum number of active processes since FPM
|
||||||
|
; has started;
|
||||||
|
; max children reached - number of times, the process limit has been reached,
|
||||||
|
; when pm tries to start more children (works only for
|
||||||
|
; pm 'dynamic' and 'ondemand');
|
||||||
|
; Value are updated in real time.
|
||||||
|
; Example output:
|
||||||
|
; pool: www
|
||||||
|
; process manager: static
|
||||||
|
; start time: 01/Jul/2011:17:53:49 +0200
|
||||||
|
; start since: 62636
|
||||||
|
; accepted conn: 190460
|
||||||
|
; listen queue: 0
|
||||||
|
; max listen queue: 1
|
||||||
|
; listen queue len: 42
|
||||||
|
; idle processes: 4
|
||||||
|
; active processes: 11
|
||||||
|
; total processes: 15
|
||||||
|
; max active processes: 12
|
||||||
|
; max children reached: 0
|
||||||
|
;
|
||||||
|
; By default the status page output is formatted as text/plain. Passing either
|
||||||
|
; 'html', 'xml' or 'json' in the query string will return the corresponding
|
||||||
|
; output syntax. Example:
|
||||||
|
; http://www.foo.bar/status
|
||||||
|
; http://www.foo.bar/status?json
|
||||||
|
; http://www.foo.bar/status?html
|
||||||
|
; http://www.foo.bar/status?xml
|
||||||
|
;
|
||||||
|
; By default the status page only outputs short status. Passing 'full' in the
|
||||||
|
; query string will also return status for each pool process.
|
||||||
|
; Example:
|
||||||
|
; http://www.foo.bar/status?full
|
||||||
|
; http://www.foo.bar/status?json&full
|
||||||
|
; http://www.foo.bar/status?html&full
|
||||||
|
; http://www.foo.bar/status?xml&full
|
||||||
|
; The Full status returns for each process:
|
||||||
|
; pid - the PID of the process;
|
||||||
|
; state - the state of the process (Idle, Running, ...);
|
||||||
|
; start time - the date and time the process has started;
|
||||||
|
; start since - the number of seconds since the process has started;
|
||||||
|
; requests - the number of requests the process has served;
|
||||||
|
; request duration - the duration in µs of the requests;
|
||||||
|
; request method - the request method (GET, POST, ...);
|
||||||
|
; request URI - the request URI with the query string;
|
||||||
|
; content length - the content length of the request (only with POST);
|
||||||
|
; user - the user (PHP_AUTH_USER) (or '-' if not set);
|
||||||
|
; script - the main script called (or '-' if not set);
|
||||||
|
; last request cpu - the %cpu the last request consumed
|
||||||
|
; it's always 0 if the process is not in Idle state
|
||||||
|
; because CPU calculation is done when the request
|
||||||
|
; processing has terminated;
|
||||||
|
; last request memory - the max amount of memory the last request consumed
|
||||||
|
; it's always 0 if the process is not in Idle state
|
||||||
|
; because memory calculation is done when the request
|
||||||
|
; processing has terminated;
|
||||||
|
; If the process is in Idle state, then informations are related to the
|
||||||
|
; last request the process has served. Otherwise informations are related to
|
||||||
|
; the current request being served.
|
||||||
|
; Example output:
|
||||||
|
; ************************
|
||||||
|
; pid: 31330
|
||||||
|
; state: Running
|
||||||
|
; start time: 01/Jul/2011:17:53:49 +0200
|
||||||
|
; start since: 63087
|
||||||
|
; requests: 12808
|
||||||
|
; request duration: 1250261
|
||||||
|
; request method: GET
|
||||||
|
; request URI: /test_mem.php?N=10000
|
||||||
|
; content length: 0
|
||||||
|
; user: -
|
||||||
|
; script: /home/fat/web/docs/php/test_mem.php
|
||||||
|
; last request cpu: 0.00
|
||||||
|
; last request memory: 0
|
||||||
|
;
|
||||||
|
; Note: There is a real-time FPM status monitoring sample web page available
|
||||||
|
; It's available in: /usr/share/php83/fpm/status.html
|
||||||
|
;
|
||||||
|
; Note: The value must start with a leading slash (/). The value can be
|
||||||
|
; anything, but it may not be a good idea to use the .php extension or it
|
||||||
|
; may conflict with a real PHP file.
|
||||||
|
; Default Value: not set
|
||||||
|
;pm.status_path = /status
|
||||||
|
|
||||||
|
; The address on which to accept FastCGI status request. This creates a new
|
||||||
|
; invisible pool that can handle requests independently. This is useful
|
||||||
|
; if the main pool is busy with long running requests because it is still possible
|
||||||
|
; to get the status before finishing the long running requests.
|
||||||
|
;
|
||||||
|
; Valid syntaxes are:
|
||||||
|
; 'ip.add.re.ss:port' - to listen on a TCP socket to a specific IPv4 address on
|
||||||
|
; a specific port;
|
||||||
|
; '[ip:6:addr:ess]:port' - to listen on a TCP socket to a specific IPv6 address on
|
||||||
|
; a specific port;
|
||||||
|
; 'port' - to listen on a TCP socket to all addresses
|
||||||
|
; (IPv6 and IPv4-mapped) on a specific port;
|
||||||
|
; '/path/to/unix/socket' - to listen on a unix socket.
|
||||||
|
; Default Value: value of the listen option
|
||||||
|
;pm.status_listen = 127.0.0.1:9001
|
||||||
|
|
||||||
|
; The ping URI to call the monitoring page of FPM. If this value is not set, no
|
||||||
|
; URI will be recognized as a ping page. This could be used to test from outside
|
||||||
|
; that FPM is alive and responding, or to
|
||||||
|
; - create a graph of FPM availability (rrd or such);
|
||||||
|
; - remove a server from a group if it is not responding (load balancing);
|
||||||
|
; - trigger alerts for the operating team (24/7).
|
||||||
|
; Note: The value must start with a leading slash (/). The value can be
|
||||||
|
; anything, but it may not be a good idea to use the .php extension or it
|
||||||
|
; may conflict with a real PHP file.
|
||||||
|
; Default Value: not set
|
||||||
|
;ping.path = /ping
|
||||||
|
|
||||||
|
; This directive may be used to customize the response of a ping request. The
|
||||||
|
; response is formatted as text/plain with a 200 response code.
|
||||||
|
; Default Value: pong
|
||||||
|
;ping.response = pong
|
||||||
|
|
||||||
|
; The access log file
|
||||||
|
; Default: not set
|
||||||
|
;access.log = log/php83/$pool.access.log
|
||||||
|
|
||||||
|
; The access log format.
|
||||||
|
; The following syntax is allowed
|
||||||
|
; %%: the '%' character
|
||||||
|
; %C: %CPU used by the request
|
||||||
|
; it can accept the following format:
|
||||||
|
; - %{user}C for user CPU only
|
||||||
|
; - %{system}C for system CPU only
|
||||||
|
; - %{total}C for user + system CPU (default)
|
||||||
|
; %d: time taken to serve the request
|
||||||
|
; it can accept the following format:
|
||||||
|
; - %{seconds}d (default)
|
||||||
|
; - %{milliseconds}d
|
||||||
|
; - %{milli}d
|
||||||
|
; - %{microseconds}d
|
||||||
|
; - %{micro}d
|
||||||
|
; %e: an environment variable (same as $_ENV or $_SERVER)
|
||||||
|
; it must be associated with embraces to specify the name of the env
|
||||||
|
; variable. Some examples:
|
||||||
|
; - server specifics like: %{REQUEST_METHOD}e or %{SERVER_PROTOCOL}e
|
||||||
|
; - HTTP headers like: %{HTTP_HOST}e or %{HTTP_USER_AGENT}e
|
||||||
|
; %f: script filename
|
||||||
|
; %l: content-length of the request (for POST request only)
|
||||||
|
; %m: request method
|
||||||
|
; %M: peak of memory allocated by PHP
|
||||||
|
; it can accept the following format:
|
||||||
|
; - %{bytes}M (default)
|
||||||
|
; - %{kilobytes}M
|
||||||
|
; - %{kilo}M
|
||||||
|
; - %{megabytes}M
|
||||||
|
; - %{mega}M
|
||||||
|
; %n: pool name
|
||||||
|
; %o: output header
|
||||||
|
; it must be associated with embraces to specify the name of the header:
|
||||||
|
; - %{Content-Type}o
|
||||||
|
; - %{X-Powered-By}o
|
||||||
|
; - %{Transfert-Encoding}o
|
||||||
|
; - ....
|
||||||
|
; %p: PID of the child that serviced the request
|
||||||
|
; %P: PID of the parent of the child that serviced the request
|
||||||
|
; %q: the query string
|
||||||
|
; %Q: the '?' character if query string exists
|
||||||
|
; %r: the request URI (without the query string, see %q and %Q)
|
||||||
|
; %R: remote IP address
|
||||||
|
; %s: status (response code)
|
||||||
|
; %t: server time the request was received
|
||||||
|
; it can accept a strftime(3) format:
|
||||||
|
; %d/%b/%Y:%H:%M:%S %z (default)
|
||||||
|
; The strftime(3) format must be encapsulated in a %{<strftime_format>}t tag
|
||||||
|
; e.g. for a ISO8601 formatted timestring, use: %{%Y-%m-%dT%H:%M:%S%z}t
|
||||||
|
; %T: time the log has been written (the request has finished)
|
||||||
|
; it can accept a strftime(3) format:
|
||||||
|
; %d/%b/%Y:%H:%M:%S %z (default)
|
||||||
|
; The strftime(3) format must be encapsulated in a %{<strftime_format>}t tag
|
||||||
|
; e.g. for a ISO8601 formatted timestring, use: %{%Y-%m-%dT%H:%M:%S%z}t
|
||||||
|
; %u: remote user
|
||||||
|
;
|
||||||
|
; Default: "%R - %u %t \"%m %r\" %s"
|
||||||
|
;access.format = "%R - %u %t \"%m %r%Q%q\" %s %f %{milli}d %{kilo}M %C%%"
|
||||||
|
|
||||||
|
; A list of request_uri values which should be filtered from the access log.
|
||||||
|
;
|
||||||
|
; As a security precuation, this setting will be ignored if:
|
||||||
|
; - the request method is not GET or HEAD; or
|
||||||
|
; - there is a request body; or
|
||||||
|
; - there are query parameters; or
|
||||||
|
; - the response code is outwith the successful range of 200 to 299
|
||||||
|
;
|
||||||
|
; Note: The paths are matched against the output of the access.format tag "%r".
|
||||||
|
; On common configurations, this may look more like SCRIPT_NAME than the
|
||||||
|
; expected pre-rewrite URI.
|
||||||
|
;
|
||||||
|
; Default Value: not set
|
||||||
|
;access.suppress_path[] = /ping
|
||||||
|
;access.suppress_path[] = /health_check.php
|
||||||
|
|
||||||
|
; The log file for slow requests
|
||||||
|
; Default Value: not set
|
||||||
|
; Note: slowlog is mandatory if request_slowlog_timeout is set
|
||||||
|
;slowlog = log/php83/$pool.slow.log
|
||||||
|
|
||||||
|
; The timeout for serving a single request after which a PHP backtrace will be
|
||||||
|
; dumped to the 'slowlog' file. A value of '0s' means 'off'.
|
||||||
|
; Available units: s(econds)(default), m(inutes), h(ours), or d(ays)
|
||||||
|
; Default Value: 0
|
||||||
|
;request_slowlog_timeout = 0
|
||||||
|
|
||||||
|
; Depth of slow log stack trace.
|
||||||
|
; Default Value: 20
|
||||||
|
;request_slowlog_trace_depth = 20
|
||||||
|
|
||||||
|
; The timeout for serving a single request after which the worker process will
|
||||||
|
; be killed. This option should be used when the 'max_execution_time' ini option
|
||||||
|
; does not stop script execution for some reason. A value of '0' means 'off'.
|
||||||
|
; Available units: s(econds)(default), m(inutes), h(ours), or d(ays)
|
||||||
|
; Default Value: 0
|
||||||
|
;request_terminate_timeout = 0
|
||||||
|
|
||||||
|
; The timeout set by 'request_terminate_timeout' ini option is not engaged after
|
||||||
|
; application calls 'fastcgi_finish_request' or when application has finished and
|
||||||
|
; shutdown functions are being called (registered via register_shutdown_function).
|
||||||
|
; This option will enable timeout limit to be applied unconditionally
|
||||||
|
; even in such cases.
|
||||||
|
; Default Value: no
|
||||||
|
;request_terminate_timeout_track_finished = no
|
||||||
|
|
||||||
|
; Set open file descriptor rlimit.
|
||||||
|
; Default Value: system defined value
|
||||||
|
;rlimit_files = 1024
|
||||||
|
|
||||||
|
; Set max core size rlimit.
|
||||||
|
; Possible Values: 'unlimited' or an integer greater or equal to 0
|
||||||
|
; Default Value: system defined value
|
||||||
|
;rlimit_core = 0
|
||||||
|
|
||||||
|
; Chroot to this directory at the start. This value must be defined as an
|
||||||
|
; absolute path. When this value is not set, chroot is not used.
|
||||||
|
; Note: you can prefix with '$prefix' to chroot to the pool prefix or one
|
||||||
|
; of its subdirectories. If the pool prefix is not set, the global prefix
|
||||||
|
; will be used instead.
|
||||||
|
; Note: chrooting is a great security feature and should be used whenever
|
||||||
|
; possible. However, all PHP paths will be relative to the chroot
|
||||||
|
; (error_log, sessions.save_path, ...).
|
||||||
|
; Default Value: not set
|
||||||
|
;chroot =
|
||||||
|
|
||||||
|
; Chdir to this directory at the start.
|
||||||
|
; Note: relative path can be used.
|
||||||
|
; Default Value: current directory or / when chroot
|
||||||
|
;chdir = /var/www
|
||||||
|
|
||||||
|
; Redirect worker stdout and stderr into main error log. If not set, stdout and
|
||||||
|
; stderr will be redirected to /dev/null according to FastCGI specs.
|
||||||
|
; Note: on highloaded environment, this can cause some delay in the page
|
||||||
|
; process time (several ms).
|
||||||
|
; Default Value: no
|
||||||
|
;catch_workers_output = yes
|
||||||
|
|
||||||
|
; Decorate worker output with prefix and suffix containing information about
|
||||||
|
; the child that writes to the log and if stdout or stderr is used as well as
|
||||||
|
; log level and time. This options is used only if catch_workers_output is yes.
|
||||||
|
; Settings to "no" will output data as written to the stdout or stderr.
|
||||||
|
; Default value: yes
|
||||||
|
;decorate_workers_output = no
|
||||||
|
|
||||||
|
; Clear environment in FPM workers
|
||||||
|
; Prevents arbitrary environment variables from reaching FPM worker processes
|
||||||
|
; by clearing the environment in workers before env vars specified in this
|
||||||
|
; pool configuration are added.
|
||||||
|
; Setting to "no" will make all environment variables available to PHP code
|
||||||
|
; via getenv(), $_ENV and $_SERVER.
|
||||||
|
; Default Value: yes
|
||||||
|
;clear_env = no
|
||||||
|
|
||||||
|
; Limits the extensions of the main script FPM will allow to parse. This can
|
||||||
|
; prevent configuration mistakes on the web server side. You should only limit
|
||||||
|
; FPM to .php extensions to prevent malicious users to use other extensions to
|
||||||
|
; execute php code.
|
||||||
|
; Note: set an empty value to allow all extensions.
|
||||||
|
; Default Value: .php
|
||||||
|
;security.limit_extensions = .php .php3 .php4 .php5 .php7
|
||||||
|
|
||||||
|
; Pass environment variables like LD_LIBRARY_PATH. All $VARIABLEs are taken from
|
||||||
|
; the current environment.
|
||||||
|
; Default Value: clean env
|
||||||
|
;env[HOSTNAME] = $HOSTNAME
|
||||||
|
env[PATH] = /opt/venv:/usr/local/bin:/usr/bin:/bin
|
||||||
|
env[TMP] = /services/run/tmp
|
||||||
|
env[TMPDIR] = /services/run/tmp
|
||||||
|
env[TEMP] = /services/run/tmp
|
||||||
|
|
||||||
|
; Additional php.ini defines, specific to this pool of workers. These settings
|
||||||
|
; overwrite the values previously defined in the php.ini. The directives are the
|
||||||
|
; same as the PHP SAPI:
|
||||||
|
; php_value/php_flag - you can set classic ini defines which can
|
||||||
|
; be overwritten from PHP call 'ini_set'.
|
||||||
|
; php_admin_value/php_admin_flag - these directives won't be overwritten by
|
||||||
|
; PHP call 'ini_set'
|
||||||
|
; For php_*flag, valid values are on, off, 1, 0, true, false, yes or no.
|
||||||
|
|
||||||
|
; Defining 'extension' will load the corresponding shared extension from
|
||||||
|
; extension_dir. Defining 'disable_functions' or 'disable_classes' will not
|
||||||
|
; overwrite previously defined php.ini values, but will append the new value
|
||||||
|
; instead.
|
||||||
|
|
||||||
|
; Note: path INI options can be relative and will be expanded with the prefix
|
||||||
|
; (pool, global or /usr)
|
||||||
|
|
||||||
|
; Default Value: nothing is defined by default except the values in php.ini and
|
||||||
|
; specified at startup with the -d argument
|
||||||
|
;php_admin_value[sendmail_path] = /usr/sbin/sendmail -t -i -f www@my.domain.com
|
||||||
|
php_admin_value[sys_temp_dir] = /services/run/tmp
|
||||||
|
php_admin_value[upload_tmp_dir] = /services/run/tmp
|
||||||
|
php_admin_value[session.save_path] = /services/run/tmp
|
||||||
|
php_admin_value[output_buffering] = 262144
|
||||||
|
php_admin_flag[implicit_flush] = off
|
||||||
|
php_admin_value[realpath_cache_size] = 4096K
|
||||||
|
php_admin_value[realpath_cache_ttl] = 600
|
||||||
|
php_admin_value[memory_limit] = 256M
|
||||||
73
install/production-filesystem/services/healthcheck.sh
Executable file
73
install/production-filesystem/services/healthcheck.sh
Executable file
@@ -0,0 +1,73 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
# NetAlertX Comprehensive Health Check Script
|
||||||
|
# This script verifies all critical services and endpoints are functioning
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
EXIT_CODE=0
|
||||||
|
|
||||||
|
echo "[HEALTHCHECK] Starting comprehensive health check..."
|
||||||
|
|
||||||
|
# Function to log errors and set exit code
|
||||||
|
log_error() {
|
||||||
|
echo "[HEALTHCHECK] ERROR: $1" >&2
|
||||||
|
EXIT_CODE=1
|
||||||
|
}
|
||||||
|
|
||||||
|
# Function to log success
|
||||||
|
log_success() {
|
||||||
|
echo "[HEALTHCHECK] ✓ $1"
|
||||||
|
}
|
||||||
|
|
||||||
|
# 1. Check if crond is running
|
||||||
|
if pgrep -f "crond" > /dev/null; then
|
||||||
|
log_success "crond is running"
|
||||||
|
else
|
||||||
|
log_error "crond is not running"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 2. Check if php-fpm is running
|
||||||
|
if pgrep -f "php-fpm" > /dev/null; then
|
||||||
|
log_success "php-fpm is running"
|
||||||
|
else
|
||||||
|
log_error "php-fpm is not running"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 3. Check if nginx is running
|
||||||
|
if pgrep -f "nginx" > /dev/null; then
|
||||||
|
log_success "nginx is running"
|
||||||
|
else
|
||||||
|
log_error "nginx is not running"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 4. Check if python /app/server is running
|
||||||
|
if pgrep -f "python.*server" > /dev/null; then
|
||||||
|
log_success "python /app/server is running"
|
||||||
|
else
|
||||||
|
log_error "python /app/server is not running"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 5. Check port 20211 is open and contains "netalertx"
|
||||||
|
if curl -sf --max-time 10 "http://localhost:${PORT:-20211}" | grep -i "netalertx" > /dev/null; then
|
||||||
|
log_success "Port ${PORT:-20211} is responding and contains 'netalertx'"
|
||||||
|
else
|
||||||
|
log_error "Port ${PORT:-20211} is not responding or doesn't contain 'netalertx'"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# 6. Check port 20212/graphql returns "graphql" in first lines
|
||||||
|
GRAPHQL_PORT=${GRAPHQL_PORT:-20212}
|
||||||
|
if curl -sf --max-time 10 "http://localhost:${GRAPHQL_PORT}/graphql" | head -10 | grep -i "graphql" > /dev/null; then
|
||||||
|
log_success "Port ${GRAPHQL_PORT}/graphql is responding with GraphQL content"
|
||||||
|
else
|
||||||
|
log_error "Port ${GRAPHQL_PORT}/graphql is not responding or doesn't contain 'graphql'"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Summary
|
||||||
|
if [ $EXIT_CODE -eq 0 ]; then
|
||||||
|
echo "[HEALTHCHECK] ✅ All health checks passed"
|
||||||
|
else
|
||||||
|
echo "[HEALTHCHECK] ❌ One or more health checks failed"
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit $EXIT_CODE
|
||||||
31
install/production-filesystem/services/scripts/check-cap.sh
Normal file
31
install/production-filesystem/services/scripts/check-cap.sh
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# check-cap.sh - Uses a real nmap command to detect missing container
|
||||||
|
# privileges and warns the user. It is silent on success.
|
||||||
|
|
||||||
|
# Run a fast nmap command that requires raw sockets, capturing only stderr.
|
||||||
|
ERROR_OUTPUT=$(nmap --privileged -sS -p 20211 127.0.0.1 2>&1)
|
||||||
|
EXIT_CODE=$?
|
||||||
|
|
||||||
|
# Flag common capability errors regardless of exact exit code.
|
||||||
|
if [ "$EXIT_CODE" -ne 0 ] && \
|
||||||
|
echo "$ERROR_OUTPUT" | grep -q -e "Operation not permitted" -e "requires root privileges"
|
||||||
|
then
|
||||||
|
YELLOW=$(printf '\033[1;33m')
|
||||||
|
RESET=$(printf '\033[0m')
|
||||||
|
>&2 printf "%s" "${YELLOW}"
|
||||||
|
>&2 cat <<'EOF'
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
⚠️ ATTENTION: Raw network capabilities are missing.
|
||||||
|
|
||||||
|
Tools that rely on NET_RAW/NET_ADMIN/NET_BIND_SERVICE (e.g. nmap -sS,
|
||||||
|
arp-scan, nbtscan) will not function. Restart the container with:
|
||||||
|
|
||||||
|
--cap-add=NET_RAW --cap-add=NET_ADMIN --cap-add=NET_BIND_SERVICE
|
||||||
|
|
||||||
|
Without those caps, NetAlertX cannot inspect your network. Fix it before
|
||||||
|
trusting any results.
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
EOF
|
||||||
|
>&2 printf "%s" "${RESET}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# first-run-check.sh - Checks and initializes configuration files on first run
|
||||||
|
|
||||||
|
# Check for app.conf and deploy if required
|
||||||
|
if [ ! -f ${NETALERTX_CONFIG}/app.conf ]; then
|
||||||
|
mkdir -p "${NETALERTX_CONFIG}" || {
|
||||||
|
>&2 echo "ERROR: Failed to create config directory ${NETALERTX_CONFIG}"
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
cp /app/back/app.conf "${NETALERTX_CONFIG}/app.conf" || {
|
||||||
|
>&2 echo "ERROR: Failed to copy default config to ${NETALERTX_CONFIG}/app.conf"
|
||||||
|
exit 1
|
||||||
|
}
|
||||||
|
CYAN='\033[1;36m'
|
||||||
|
RESET='\033[0m'
|
||||||
|
>&2 printf "%s" "${CYAN}"
|
||||||
|
>&2 cat <<EOF
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
🆕 First run detected. Default configuration written to ${NETALERTX_CONFIG}/app.conf.
|
||||||
|
|
||||||
|
Review your settings in the UI or edit the file directly before trusting
|
||||||
|
this instance in production.
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
EOF
|
||||||
|
>&2 printf "%s" "${RESET}"
|
||||||
|
fi
|
||||||
|
|
||||||
@@ -0,0 +1,456 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# This script checks if the database file exists, and if not, creates it with the initial schema.
|
||||||
|
# It is intended to be run at the first start of the application.
|
||||||
|
|
||||||
|
# if the db exists, exit
|
||||||
|
test -f "${NETALERTX_DB_FILE}" && exit 0
|
||||||
|
|
||||||
|
CYAN='\033[1;36m'
|
||||||
|
RESET='\033[0m'
|
||||||
|
>&2 printf "%s" "${CYAN}"
|
||||||
|
>&2 cat <<EOF
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
🆕 First run detected. Building initial database schema in ${NETALERTX_DB_FILE}.
|
||||||
|
|
||||||
|
Do not interrupt this step. Once complete, consider backing up the fresh
|
||||||
|
database before onboarding sensitive networks.
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
EOF
|
||||||
|
>&2 printf "%s" "${RESET}"
|
||||||
|
|
||||||
|
# Write all text to db file until we see "end-of-database-schema"
|
||||||
|
sqlite3 "${NETALERTX_DB_FILE}" <<'end-of-database-schema'
|
||||||
|
CREATE TABLE Events (eve_MAC STRING (50) NOT NULL COLLATE NOCASE, eve_IP STRING (50) NOT NULL COLLATE NOCASE, eve_DateTime DATETIME NOT NULL, eve_EventType STRING (30) NOT NULL COLLATE NOCASE, eve_AdditionalInfo STRING (250) DEFAULT (''), eve_PendingAlertEmail BOOLEAN NOT NULL CHECK (eve_PendingAlertEmail IN (0, 1)) DEFAULT (1), eve_PairEventRowid INTEGER);
|
||||||
|
CREATE TABLE Sessions (ses_MAC STRING (50) COLLATE NOCASE, ses_IP STRING (50) COLLATE NOCASE, ses_EventTypeConnection STRING (30) COLLATE NOCASE, ses_DateTimeConnection DATETIME, ses_EventTypeDisconnection STRING (30) COLLATE NOCASE, ses_DateTimeDisconnection DATETIME, ses_StillConnected BOOLEAN, ses_AdditionalInfo STRING (250));
|
||||||
|
CREATE TABLE IF NOT EXISTS "Online_History" (
|
||||||
|
"Index" INTEGER,
|
||||||
|
"Scan_Date" TEXT,
|
||||||
|
"Online_Devices" INTEGER,
|
||||||
|
"Down_Devices" INTEGER,
|
||||||
|
"All_Devices" INTEGER,
|
||||||
|
"Archived_Devices" INTEGER,
|
||||||
|
"Offline_Devices" INTEGER,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE sqlite_sequence(name,seq);
|
||||||
|
CREATE TABLE Devices (
|
||||||
|
devMac STRING (50) PRIMARY KEY NOT NULL COLLATE NOCASE,
|
||||||
|
devName STRING (50) NOT NULL DEFAULT "(unknown)",
|
||||||
|
devOwner STRING (30) DEFAULT "(unknown)" NOT NULL,
|
||||||
|
devType STRING (30),
|
||||||
|
devVendor STRING (250),
|
||||||
|
devFavorite BOOLEAN CHECK (devFavorite IN (0, 1)) DEFAULT (0) NOT NULL,
|
||||||
|
devGroup STRING (10),
|
||||||
|
devComments TEXT,
|
||||||
|
devFirstConnection DATETIME NOT NULL,
|
||||||
|
devLastConnection DATETIME NOT NULL,
|
||||||
|
devLastIP STRING (50) NOT NULL COLLATE NOCASE,
|
||||||
|
devStaticIP BOOLEAN DEFAULT (0) NOT NULL CHECK (devStaticIP IN (0, 1)),
|
||||||
|
devScan INTEGER DEFAULT (1) NOT NULL,
|
||||||
|
devLogEvents BOOLEAN NOT NULL DEFAULT (1) CHECK (devLogEvents IN (0, 1)),
|
||||||
|
devAlertEvents BOOLEAN NOT NULL DEFAULT (1) CHECK (devAlertEvents IN (0, 1)),
|
||||||
|
devAlertDown BOOLEAN NOT NULL DEFAULT (0) CHECK (devAlertDown IN (0, 1)),
|
||||||
|
devSkipRepeated INTEGER DEFAULT 0 NOT NULL,
|
||||||
|
devLastNotification DATETIME,
|
||||||
|
devPresentLastScan BOOLEAN NOT NULL DEFAULT (0) CHECK (devPresentLastScan IN (0, 1)),
|
||||||
|
devIsNew BOOLEAN NOT NULL DEFAULT (1) CHECK (devIsNew IN (0, 1)),
|
||||||
|
devLocation STRING (250) COLLATE NOCASE,
|
||||||
|
devIsArchived BOOLEAN NOT NULL DEFAULT (0) CHECK (devIsArchived IN (0, 1)),
|
||||||
|
devParentMAC TEXT,
|
||||||
|
devParentPort INTEGER,
|
||||||
|
devIcon TEXT,
|
||||||
|
devGUID TEXT,
|
||||||
|
devSite TEXT,
|
||||||
|
devSSID TEXT,
|
||||||
|
devSyncHubNode TEXT,
|
||||||
|
devSourcePlugin TEXT
|
||||||
|
, "devCustomProps" TEXT);
|
||||||
|
CREATE TABLE IF NOT EXISTS "Settings" (
|
||||||
|
"setKey" TEXT,
|
||||||
|
"setName" TEXT,
|
||||||
|
"setDescription" TEXT,
|
||||||
|
"setType" TEXT,
|
||||||
|
"setOptions" TEXT,
|
||||||
|
"setGroup" TEXT,
|
||||||
|
"setValue" TEXT,
|
||||||
|
"setEvents" TEXT,
|
||||||
|
"setOverriddenByEnv" INTEGER
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS "Parameters" (
|
||||||
|
"par_ID" TEXT PRIMARY KEY,
|
||||||
|
"par_Value" TEXT
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_Objects(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Plugin TEXT NOT NULL,
|
||||||
|
Object_PrimaryID TEXT NOT NULL,
|
||||||
|
Object_SecondaryID TEXT NOT NULL,
|
||||||
|
DateTimeCreated TEXT NOT NULL,
|
||||||
|
DateTimeChanged TEXT NOT NULL,
|
||||||
|
Watched_Value1 TEXT NOT NULL,
|
||||||
|
Watched_Value2 TEXT NOT NULL,
|
||||||
|
Watched_Value3 TEXT NOT NULL,
|
||||||
|
Watched_Value4 TEXT NOT NULL,
|
||||||
|
Status TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
UserData TEXT NOT NULL,
|
||||||
|
ForeignKey TEXT NOT NULL,
|
||||||
|
SyncHubNodeName TEXT,
|
||||||
|
"HelpVal1" TEXT,
|
||||||
|
"HelpVal2" TEXT,
|
||||||
|
"HelpVal3" TEXT,
|
||||||
|
"HelpVal4" TEXT,
|
||||||
|
ObjectGUID TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_Events(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Plugin TEXT NOT NULL,
|
||||||
|
Object_PrimaryID TEXT NOT NULL,
|
||||||
|
Object_SecondaryID TEXT NOT NULL,
|
||||||
|
DateTimeCreated TEXT NOT NULL,
|
||||||
|
DateTimeChanged TEXT NOT NULL,
|
||||||
|
Watched_Value1 TEXT NOT NULL,
|
||||||
|
Watched_Value2 TEXT NOT NULL,
|
||||||
|
Watched_Value3 TEXT NOT NULL,
|
||||||
|
Watched_Value4 TEXT NOT NULL,
|
||||||
|
Status TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
UserData TEXT NOT NULL,
|
||||||
|
ForeignKey TEXT NOT NULL,
|
||||||
|
SyncHubNodeName TEXT,
|
||||||
|
"HelpVal1" TEXT,
|
||||||
|
"HelpVal2" TEXT,
|
||||||
|
"HelpVal3" TEXT,
|
||||||
|
"HelpVal4" TEXT, "ObjectGUID" TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_History(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Plugin TEXT NOT NULL,
|
||||||
|
Object_PrimaryID TEXT NOT NULL,
|
||||||
|
Object_SecondaryID TEXT NOT NULL,
|
||||||
|
DateTimeCreated TEXT NOT NULL,
|
||||||
|
DateTimeChanged TEXT NOT NULL,
|
||||||
|
Watched_Value1 TEXT NOT NULL,
|
||||||
|
Watched_Value2 TEXT NOT NULL,
|
||||||
|
Watched_Value3 TEXT NOT NULL,
|
||||||
|
Watched_Value4 TEXT NOT NULL,
|
||||||
|
Status TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
UserData TEXT NOT NULL,
|
||||||
|
ForeignKey TEXT NOT NULL,
|
||||||
|
SyncHubNodeName TEXT,
|
||||||
|
"HelpVal1" TEXT,
|
||||||
|
"HelpVal2" TEXT,
|
||||||
|
"HelpVal3" TEXT,
|
||||||
|
"HelpVal4" TEXT, "ObjectGUID" TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE Plugins_Language_Strings(
|
||||||
|
"Index" INTEGER,
|
||||||
|
Language_Code TEXT NOT NULL,
|
||||||
|
String_Key TEXT NOT NULL,
|
||||||
|
String_Value TEXT NOT NULL,
|
||||||
|
Extra TEXT NOT NULL,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE TABLE CurrentScan (
|
||||||
|
cur_MAC STRING(50) NOT NULL COLLATE NOCASE,
|
||||||
|
cur_IP STRING(50) NOT NULL COLLATE NOCASE,
|
||||||
|
cur_Vendor STRING(250),
|
||||||
|
cur_ScanMethod STRING(10),
|
||||||
|
cur_Name STRING(250),
|
||||||
|
cur_LastQuery STRING(250),
|
||||||
|
cur_DateTime STRING(250),
|
||||||
|
cur_SyncHubNodeName STRING(50),
|
||||||
|
cur_NetworkSite STRING(250),
|
||||||
|
cur_SSID STRING(250),
|
||||||
|
cur_NetworkNodeMAC STRING(250),
|
||||||
|
cur_PORT STRING(250),
|
||||||
|
cur_Type STRING(250),
|
||||||
|
UNIQUE(cur_MAC)
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS "AppEvents" (
|
||||||
|
"Index" INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||||
|
"GUID" TEXT UNIQUE,
|
||||||
|
"AppEventProcessed" BOOLEAN,
|
||||||
|
"DateTimeCreated" TEXT,
|
||||||
|
"ObjectType" TEXT,
|
||||||
|
"ObjectGUID" TEXT,
|
||||||
|
"ObjectPlugin" TEXT,
|
||||||
|
"ObjectPrimaryID" TEXT,
|
||||||
|
"ObjectSecondaryID" TEXT,
|
||||||
|
"ObjectForeignKey" TEXT,
|
||||||
|
"ObjectIndex" TEXT,
|
||||||
|
"ObjectIsNew" BOOLEAN,
|
||||||
|
"ObjectIsArchived" BOOLEAN,
|
||||||
|
"ObjectStatusColumn" TEXT,
|
||||||
|
"ObjectStatus" TEXT,
|
||||||
|
"AppEventType" TEXT,
|
||||||
|
"Helper1" TEXT,
|
||||||
|
"Helper2" TEXT,
|
||||||
|
"Helper3" TEXT,
|
||||||
|
"Extra" TEXT
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS "Notifications" (
|
||||||
|
"Index" INTEGER,
|
||||||
|
"GUID" TEXT UNIQUE,
|
||||||
|
"DateTimeCreated" TEXT,
|
||||||
|
"DateTimePushed" TEXT,
|
||||||
|
"Status" TEXT,
|
||||||
|
"JSON" TEXT,
|
||||||
|
"Text" TEXT,
|
||||||
|
"HTML" TEXT,
|
||||||
|
"PublishedVia" TEXT,
|
||||||
|
"Extra" TEXT,
|
||||||
|
PRIMARY KEY("Index" AUTOINCREMENT)
|
||||||
|
);
|
||||||
|
CREATE INDEX IDX_eve_DateTime ON Events (eve_DateTime);
|
||||||
|
CREATE INDEX IDX_eve_EventType ON Events (eve_EventType COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_eve_MAC ON Events (eve_MAC COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_eve_PairEventRowid ON Events (eve_PairEventRowid);
|
||||||
|
CREATE INDEX IDX_ses_EventTypeDisconnection ON Sessions (ses_EventTypeDisconnection COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_ses_EventTypeConnection ON Sessions (ses_EventTypeConnection COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_ses_DateTimeDisconnection ON Sessions (ses_DateTimeDisconnection);
|
||||||
|
CREATE INDEX IDX_ses_MAC ON Sessions (ses_MAC COLLATE NOCASE);
|
||||||
|
CREATE INDEX IDX_ses_DateTimeConnection ON Sessions (ses_DateTimeConnection);
|
||||||
|
CREATE INDEX IDX_dev_PresentLastScan ON Devices (devPresentLastScan);
|
||||||
|
CREATE INDEX IDX_dev_FirstConnection ON Devices (devFirstConnection);
|
||||||
|
CREATE INDEX IDX_dev_AlertDeviceDown ON Devices (devAlertDown);
|
||||||
|
CREATE INDEX IDX_dev_StaticIP ON Devices (devStaticIP);
|
||||||
|
CREATE INDEX IDX_dev_ScanCycle ON Devices (devScan);
|
||||||
|
CREATE INDEX IDX_dev_Favorite ON Devices (devFavorite);
|
||||||
|
CREATE INDEX IDX_dev_LastIP ON Devices (devLastIP);
|
||||||
|
CREATE INDEX IDX_dev_NewDevice ON Devices (devIsNew);
|
||||||
|
CREATE INDEX IDX_dev_Archived ON Devices (devIsArchived);
|
||||||
|
CREATE VIEW Events_Devices AS
|
||||||
|
SELECT *
|
||||||
|
FROM Events
|
||||||
|
LEFT JOIN Devices ON eve_MAC = devMac
|
||||||
|
/* Events_Devices(eve_MAC,eve_IP,eve_DateTime,eve_EventType,eve_AdditionalInfo,eve_PendingAlertEmail,eve_PairEventRowid,devMac,devName,devOwner,devType,devVendor,devFavorite,devGroup,devComments,devFirstConnection,devLastConnection,devLastIP,devStaticIP,devScan,devLogEvents,devAlertEvents,devAlertDown,devSkipRepeated,devLastNotification,devPresentLastScan,devIsNew,devLocation,devIsArchived,devParentMAC,devParentPort,devIcon,devGUID,devSite,devSSID,devSyncHubNode,devSourcePlugin,devCustomProps) */;
|
||||||
|
CREATE VIEW LatestEventsPerMAC AS
|
||||||
|
WITH RankedEvents AS (
|
||||||
|
SELECT
|
||||||
|
e.*,
|
||||||
|
ROW_NUMBER() OVER (PARTITION BY e.eve_MAC ORDER BY e.eve_DateTime DESC) AS row_num
|
||||||
|
FROM Events AS e
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
e.*,
|
||||||
|
d.*,
|
||||||
|
c.*
|
||||||
|
FROM RankedEvents AS e
|
||||||
|
LEFT JOIN Devices AS d ON e.eve_MAC = d.devMac
|
||||||
|
INNER JOIN CurrentScan AS c ON e.eve_MAC = c.cur_MAC
|
||||||
|
WHERE e.row_num = 1
|
||||||
|
/* LatestEventsPerMAC(eve_MAC,eve_IP,eve_DateTime,eve_EventType,eve_AdditionalInfo,eve_PendingAlertEmail,eve_PairEventRowid,row_num,devMac,devName,devOwner,devType,devVendor,devFavorite,devGroup,devComments,devFirstConnection,devLastConnection,devLastIP,devStaticIP,devScan,devLogEvents,devAlertEvents,devAlertDown,devSkipRepeated,devLastNotification,devPresentLastScan,devIsNew,devLocation,devIsArchived,devParentMAC,devParentPort,devIcon,devGUID,devSite,devSSID,devSyncHubNode,devSourcePlugin,devCustomProps,cur_MAC,cur_IP,cur_Vendor,cur_ScanMethod,cur_Name,cur_LastQuery,cur_DateTime,cur_SyncHubNodeName,cur_NetworkSite,cur_SSID,cur_NetworkNodeMAC,cur_PORT,cur_Type) */;
|
||||||
|
CREATE VIEW Sessions_Devices AS SELECT * FROM Sessions LEFT JOIN "Devices" ON ses_MAC = devMac
|
||||||
|
/* Sessions_Devices(ses_MAC,ses_IP,ses_EventTypeConnection,ses_DateTimeConnection,ses_EventTypeDisconnection,ses_DateTimeDisconnection,ses_StillConnected,ses_AdditionalInfo,devMac,devName,devOwner,devType,devVendor,devFavorite,devGroup,devComments,devFirstConnection,devLastConnection,devLastIP,devStaticIP,devScan,devLogEvents,devAlertEvents,devAlertDown,devSkipRepeated,devLastNotification,devPresentLastScan,devIsNew,devLocation,devIsArchived,devParentMAC,devParentPort,devIcon,devGUID,devSite,devSSID,devSyncHubNode,devSourcePlugin,devCustomProps) */;
|
||||||
|
CREATE VIEW Convert_Events_to_Sessions AS SELECT EVE1.eve_MAC,
|
||||||
|
EVE1.eve_IP,
|
||||||
|
EVE1.eve_EventType AS eve_EventTypeConnection,
|
||||||
|
EVE1.eve_DateTime AS eve_DateTimeConnection,
|
||||||
|
CASE WHEN EVE2.eve_EventType IN ('Disconnected', 'Device Down') OR
|
||||||
|
EVE2.eve_EventType IS NULL THEN EVE2.eve_EventType ELSE '<missing event>' END AS eve_EventTypeDisconnection,
|
||||||
|
CASE WHEN EVE2.eve_EventType IN ('Disconnected', 'Device Down') THEN EVE2.eve_DateTime ELSE NULL END AS eve_DateTimeDisconnection,
|
||||||
|
CASE WHEN EVE2.eve_EventType IS NULL THEN 1 ELSE 0 END AS eve_StillConnected,
|
||||||
|
EVE1.eve_AdditionalInfo
|
||||||
|
FROM Events AS EVE1
|
||||||
|
LEFT JOIN
|
||||||
|
Events AS EVE2 ON EVE1.eve_PairEventRowID = EVE2.RowID
|
||||||
|
WHERE EVE1.eve_EventType IN ('New Device', 'Connected','Down Reconnected')
|
||||||
|
UNION
|
||||||
|
SELECT eve_MAC,
|
||||||
|
eve_IP,
|
||||||
|
'<missing event>' AS eve_EventTypeConnection,
|
||||||
|
NULL AS eve_DateTimeConnection,
|
||||||
|
eve_EventType AS eve_EventTypeDisconnection,
|
||||||
|
eve_DateTime AS eve_DateTimeDisconnection,
|
||||||
|
0 AS eve_StillConnected,
|
||||||
|
eve_AdditionalInfo
|
||||||
|
FROM Events AS EVE1
|
||||||
|
WHERE (eve_EventType = 'Device Down' OR
|
||||||
|
eve_EventType = 'Disconnected') AND
|
||||||
|
EVE1.eve_PairEventRowID IS NULL
|
||||||
|
/* Convert_Events_to_Sessions(eve_MAC,eve_IP,eve_EventTypeConnection,eve_DateTimeConnection,eve_EventTypeDisconnection,eve_DateTimeDisconnection,eve_StillConnected,eve_AdditionalInfo) */;
|
||||||
|
CREATE TRIGGER "trg_insert_devices"
|
||||||
|
AFTER INSERT ON "Devices"
|
||||||
|
WHEN NOT EXISTS (
|
||||||
|
SELECT 1 FROM AppEvents
|
||||||
|
WHERE AppEventProcessed = 0
|
||||||
|
AND ObjectType = 'Devices'
|
||||||
|
AND ObjectGUID = NEW.devGUID
|
||||||
|
AND ObjectStatus = CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END
|
||||||
|
AND AppEventType = 'insert'
|
||||||
|
)
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO "AppEvents" (
|
||||||
|
"GUID",
|
||||||
|
"DateTimeCreated",
|
||||||
|
"AppEventProcessed",
|
||||||
|
"ObjectType",
|
||||||
|
"ObjectGUID",
|
||||||
|
"ObjectPrimaryID",
|
||||||
|
"ObjectSecondaryID",
|
||||||
|
"ObjectStatus",
|
||||||
|
"ObjectStatusColumn",
|
||||||
|
"ObjectIsNew",
|
||||||
|
"ObjectIsArchived",
|
||||||
|
"ObjectForeignKey",
|
||||||
|
"ObjectPlugin",
|
||||||
|
"AppEventType"
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
|
||||||
|
lower(
|
||||||
|
hex(randomblob(4)) || '-' || hex(randomblob(2)) || '-' || '4' ||
|
||||||
|
substr(hex( randomblob(2)), 2) || '-' ||
|
||||||
|
substr('AB89', 1 + (abs(random()) % 4) , 1) ||
|
||||||
|
substr(hex(randomblob(2)), 2) || '-' ||
|
||||||
|
hex(randomblob(6))
|
||||||
|
)
|
||||||
|
,
|
||||||
|
DATETIME('now'),
|
||||||
|
FALSE,
|
||||||
|
'Devices',
|
||||||
|
NEW.devGUID, -- ObjectGUID
|
||||||
|
NEW.devMac, -- ObjectPrimaryID
|
||||||
|
NEW.devLastIP, -- ObjectSecondaryID
|
||||||
|
CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END, -- ObjectStatus
|
||||||
|
'devPresentLastScan', -- ObjectStatusColumn
|
||||||
|
NEW.devIsNew, -- ObjectIsNew
|
||||||
|
NEW.devIsArchived, -- ObjectIsArchived
|
||||||
|
NEW.devGUID, -- ObjectForeignKey
|
||||||
|
'DEVICES', -- ObjectForeignKey
|
||||||
|
'insert'
|
||||||
|
);
|
||||||
|
END;
|
||||||
|
CREATE TRIGGER "trg_update_devices"
|
||||||
|
AFTER UPDATE ON "Devices"
|
||||||
|
WHEN NOT EXISTS (
|
||||||
|
SELECT 1 FROM AppEvents
|
||||||
|
WHERE AppEventProcessed = 0
|
||||||
|
AND ObjectType = 'Devices'
|
||||||
|
AND ObjectGUID = NEW.devGUID
|
||||||
|
AND ObjectStatus = CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END
|
||||||
|
AND AppEventType = 'update'
|
||||||
|
)
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO "AppEvents" (
|
||||||
|
"GUID",
|
||||||
|
"DateTimeCreated",
|
||||||
|
"AppEventProcessed",
|
||||||
|
"ObjectType",
|
||||||
|
"ObjectGUID",
|
||||||
|
"ObjectPrimaryID",
|
||||||
|
"ObjectSecondaryID",
|
||||||
|
"ObjectStatus",
|
||||||
|
"ObjectStatusColumn",
|
||||||
|
"ObjectIsNew",
|
||||||
|
"ObjectIsArchived",
|
||||||
|
"ObjectForeignKey",
|
||||||
|
"ObjectPlugin",
|
||||||
|
"AppEventType"
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
|
||||||
|
lower(
|
||||||
|
hex(randomblob(4)) || '-' || hex(randomblob(2)) || '-' || '4' ||
|
||||||
|
substr(hex( randomblob(2)), 2) || '-' ||
|
||||||
|
substr('AB89', 1 + (abs(random()) % 4) , 1) ||
|
||||||
|
substr(hex(randomblob(2)), 2) || '-' ||
|
||||||
|
hex(randomblob(6))
|
||||||
|
)
|
||||||
|
,
|
||||||
|
DATETIME('now'),
|
||||||
|
FALSE,
|
||||||
|
'Devices',
|
||||||
|
NEW.devGUID, -- ObjectGUID
|
||||||
|
NEW.devMac, -- ObjectPrimaryID
|
||||||
|
NEW.devLastIP, -- ObjectSecondaryID
|
||||||
|
CASE WHEN NEW.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END, -- ObjectStatus
|
||||||
|
'devPresentLastScan', -- ObjectStatusColumn
|
||||||
|
NEW.devIsNew, -- ObjectIsNew
|
||||||
|
NEW.devIsArchived, -- ObjectIsArchived
|
||||||
|
NEW.devGUID, -- ObjectForeignKey
|
||||||
|
'DEVICES', -- ObjectForeignKey
|
||||||
|
'update'
|
||||||
|
);
|
||||||
|
END;
|
||||||
|
CREATE TRIGGER "trg_delete_devices"
|
||||||
|
AFTER DELETE ON "Devices"
|
||||||
|
WHEN NOT EXISTS (
|
||||||
|
SELECT 1 FROM AppEvents
|
||||||
|
WHERE AppEventProcessed = 0
|
||||||
|
AND ObjectType = 'Devices'
|
||||||
|
AND ObjectGUID = OLD.devGUID
|
||||||
|
AND ObjectStatus = CASE WHEN OLD.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END
|
||||||
|
AND AppEventType = 'delete'
|
||||||
|
)
|
||||||
|
BEGIN
|
||||||
|
INSERT INTO "AppEvents" (
|
||||||
|
"GUID",
|
||||||
|
"DateTimeCreated",
|
||||||
|
"AppEventProcessed",
|
||||||
|
"ObjectType",
|
||||||
|
"ObjectGUID",
|
||||||
|
"ObjectPrimaryID",
|
||||||
|
"ObjectSecondaryID",
|
||||||
|
"ObjectStatus",
|
||||||
|
"ObjectStatusColumn",
|
||||||
|
"ObjectIsNew",
|
||||||
|
"ObjectIsArchived",
|
||||||
|
"ObjectForeignKey",
|
||||||
|
"ObjectPlugin",
|
||||||
|
"AppEventType"
|
||||||
|
)
|
||||||
|
VALUES (
|
||||||
|
|
||||||
|
lower(
|
||||||
|
hex(randomblob(4)) || '-' || hex(randomblob(2)) || '-' || '4' ||
|
||||||
|
substr(hex( randomblob(2)), 2) || '-' ||
|
||||||
|
substr('AB89', 1 + (abs(random()) % 4) , 1) ||
|
||||||
|
substr(hex(randomblob(2)), 2) || '-' ||
|
||||||
|
hex(randomblob(6))
|
||||||
|
)
|
||||||
|
,
|
||||||
|
DATETIME('now'),
|
||||||
|
FALSE,
|
||||||
|
'Devices',
|
||||||
|
OLD.devGUID, -- ObjectGUID
|
||||||
|
OLD.devMac, -- ObjectPrimaryID
|
||||||
|
OLD.devLastIP, -- ObjectSecondaryID
|
||||||
|
CASE WHEN OLD.devPresentLastScan = 1 THEN 'online' ELSE 'offline' END, -- ObjectStatus
|
||||||
|
'devPresentLastScan', -- ObjectStatusColumn
|
||||||
|
OLD.devIsNew, -- ObjectIsNew
|
||||||
|
OLD.devIsArchived, -- ObjectIsArchived
|
||||||
|
OLD.devGUID, -- ObjectForeignKey
|
||||||
|
'DEVICES', -- ObjectForeignKey
|
||||||
|
'delete'
|
||||||
|
);
|
||||||
|
END;
|
||||||
|
end-of-database-schema
|
||||||
|
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
RED='\033[1;31m'
|
||||||
|
RESET='\033[0m'
|
||||||
|
>&2 printf "%s" "${RED}"
|
||||||
|
>&2 cat <<EOF
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
❌ CRITICAL: Database schema creation failed for ${NETALERTX_DB_FILE}.
|
||||||
|
|
||||||
|
NetAlertX cannot start without a properly initialized database. This
|
||||||
|
failure typically indicates:
|
||||||
|
|
||||||
|
* Insufficient disk space or write permissions in the database directory
|
||||||
|
* Corrupted or inaccessible SQLite installation
|
||||||
|
* File system issues preventing database file creation
|
||||||
|
|
||||||
|
Check the logs for detailed SQLite error messages. Ensure the container
|
||||||
|
has write access to the database path and adequate storage space.
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
EOF
|
||||||
|
>&2 printf "%s" "${RESET}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# Initialize required directories and log files
|
||||||
|
# These must exist before services start to avoid permission/write errors
|
||||||
|
# TODO - improve with per-directory warning if creation fails
|
||||||
|
[ ! -d "${NETALERTX_PLUGINS_LOG}" ] && mkdir -p "${NETALERTX_PLUGINS_LOG}"
|
||||||
|
[ ! -d "${SYSTEM_SERVICES_RUN_LOG}" ] && mkdir -p "${SYSTEM_SERVICES_RUN_LOG}"
|
||||||
|
[ ! -d "${SYSTEM_SERVICES_RUN_TMP}" ] && mkdir -p "${SYSTEM_SERVICES_RUN_TMP}"
|
||||||
|
[ ! -f "${LOG_DB_IS_LOCKED}" ] && touch "${LOG_DB_IS_LOCKED}"
|
||||||
|
[ ! -f "${LOG_EXECUTION_QUEUE}" ] && touch "${LOG_EXECUTION_QUEUE}"
|
||||||
@@ -0,0 +1,14 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# TODO Add sanity checks here to ensure we can read from
|
||||||
|
# ${NETALERTX_APP}
|
||||||
|
# ${NETALERTX_SERVER}
|
||||||
|
# ${NETALERTX_FRONT}
|
||||||
|
# ${SYSTEM_SERVICES_CONFIG}
|
||||||
|
# ${VIRTUAL_ENV}
|
||||||
|
|
||||||
|
# And read/write tempdirs
|
||||||
|
# ${NETALERTX_API}
|
||||||
|
# ${NETALERTX_LOGS}
|
||||||
|
# ${SYSTEM_SERVICES_RUN}
|
||||||
|
|
||||||
@@ -0,0 +1,50 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# storage-check.sh - Verify critical paths use dedicated mounts.
|
||||||
|
|
||||||
|
warn_if_not_dedicated_mount() {
|
||||||
|
path="$1"
|
||||||
|
if awk -v target="${path}" '$5 == target {found=1} END {exit found ? 0 : 1}' /proc/self/mountinfo; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
failures=1
|
||||||
|
YELLOW=$(printf '\033[1;33m')
|
||||||
|
RESET=$(printf '\033[0m')
|
||||||
|
>&2 printf "%s" "${YELLOW}"
|
||||||
|
>&2 cat <<EOF
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
⚠️ ATTENTION: ${path} is not mounted separately inside this container.
|
||||||
|
|
||||||
|
NetAlertX runs as a single unprivileged process and pounds this directory
|
||||||
|
with writes. Leaving it on the container overlay will thrash storage and
|
||||||
|
slow the stack.
|
||||||
|
|
||||||
|
Fix: mount ${path} explicitly — tmpfs for ephemeral data, or bind/volume if
|
||||||
|
you want to preserve history:
|
||||||
|
--mount type=tmpfs,destination=${path}
|
||||||
|
# or
|
||||||
|
--mount type=bind,src=/path/on/host,dst=${path}
|
||||||
|
|
||||||
|
Apply the mount and restart the container.
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
EOF
|
||||||
|
>&2 printf "%s" "${RESET}"
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# If NETALERTX_DEBUG=1 then we will exit
|
||||||
|
if [ "${NETALERTX_DEBUG}" = "1" ]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
failures=0
|
||||||
|
warn_if_not_dedicated_mount "${NETALERTX_API}"
|
||||||
|
warn_if_not_dedicated_mount "${NETALERTX_LOG}"
|
||||||
|
|
||||||
|
if [ "${failures}" -ne 0 ]; then
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -f "${SYSTEM_NGINX_CONFIG}/conf.active" ]; then
|
||||||
|
echo "Note: Using default listen address ${LISTEN_ADDR}:${PORT} (no ${SYSTEM_NGINX_CONFIG}/conf.active override)."
|
||||||
|
fi
|
||||||
31
install/production-filesystem/services/scripts/check-root.sh
Normal file
31
install/production-filesystem/services/scripts/check-root.sh
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# check-root.sh - ensure the container is not running as root.
|
||||||
|
|
||||||
|
CURRENT_UID="$(id -u)"
|
||||||
|
|
||||||
|
if [ "${CURRENT_UID}" -eq 0 ]; then
|
||||||
|
YELLOW=$(printf '\033[1;33m')
|
||||||
|
RESET=$(printf '\033[0m')
|
||||||
|
>&2 printf "%s" "${YELLOW}"
|
||||||
|
>&2 cat <<'EOF'
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
⚠️ ATTENTION: NetAlertX is running as root (UID 0).
|
||||||
|
|
||||||
|
This defeats every hardening safeguard built into the image. You just
|
||||||
|
handed a high-value network monitoring appliance full control over your
|
||||||
|
host. If an attacker compromises NetAlertX now, the entire machine goes
|
||||||
|
with it.
|
||||||
|
|
||||||
|
Run the container as the dedicated 'netalertx' user instead:
|
||||||
|
* Keep the default USER in the image (20211:20211), or
|
||||||
|
* In docker-compose.yml, remove any 'user:' override that sets UID 0.
|
||||||
|
|
||||||
|
Bottom line: never run security tooling as root unless you are actively
|
||||||
|
trying to get pwned.
|
||||||
|
══════════════════════════════════════════════════════════════════════════════
|
||||||
|
EOF
|
||||||
|
>&2 printf "%s" "${RESET}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit 0
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
|
||||||
|
# TODO Sanity checks for storage paths
|
||||||
|
|
||||||
|
# Ensure we can read/write to
|
||||||
|
# ${NETALERTX_CONFIG}
|
||||||
|
# ${NETALERTX_DB}
|
||||||
16
install/production-filesystem/services/scripts/cron_script.sh
Executable file
16
install/production-filesystem/services/scripts/cron_script.sh
Executable file
@@ -0,0 +1,16 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
export INSTALL_DIR=/app
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Check if there are any entries with cron_restart_backend
|
||||||
|
if grep -q "cron_restart_backend" "${LOG_EXECUTION_QUEUE}"; then
|
||||||
|
killall python3
|
||||||
|
sleep 2
|
||||||
|
/services/start-backend.sh &
|
||||||
|
|
||||||
|
# Remove all lines containing cron_restart_backend from the log file
|
||||||
|
# Atomic replacement with temp file
|
||||||
|
grep -v "cron_restart_backend" "${LOG_EXECUTION_QUEUE}" > "${LOG_EXECUTION_QUEUE}.tmp" && \
|
||||||
|
mv "${LOG_EXECUTION_QUEUE}.tmp" "${LOG_EXECUTION_QUEUE}"
|
||||||
|
fi
|
||||||
44
install/production-filesystem/services/scripts/update_vendors.sh
Executable file
44
install/production-filesystem/services/scripts/update_vendors.sh
Executable file
@@ -0,0 +1,44 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
# NetAlertX
|
||||||
|
# Open Source Network Guard / WIFI & LAN intrusion detector
|
||||||
|
#
|
||||||
|
# update_vendors.sh - Back module. IEEE Vendors db update
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
# Puche 2021 / 2022+ jokob jokob@duck.com GNU GPLv3
|
||||||
|
# ------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
# ----------------------------------------------------------------------
|
||||||
|
# Main directories to update:
|
||||||
|
# /usr/share/arp-scan
|
||||||
|
# ----------------------------------------------------------------------
|
||||||
|
|
||||||
|
TEMP_FILE="/services/run/tmp/ieee-oui.txt.tmp"
|
||||||
|
OUTPUT_FILE="/services/run/tmp/ieee-oui.txt"
|
||||||
|
|
||||||
|
# Download the file using wget to stdout and process it
|
||||||
|
if ! wget --timeout=30 --tries=3 "https://standards-oui.ieee.org/oui/oui.txt" -O /dev/stdout | \
|
||||||
|
sed -E 's/ *\(base 16\)//' | \
|
||||||
|
awk -F' ' '{printf "%s\t%s\n", $1, substr($0, index($0, $2))}' | \
|
||||||
|
sort | \
|
||||||
|
awk '{$1=$1; print}' | \
|
||||||
|
sort -u | \
|
||||||
|
awk -F' ' '{printf "%s\t%s\n", $1, substr($0, index($0, $2))}' \
|
||||||
|
> "${TEMP_FILE}"; then
|
||||||
|
echo "ERROR: Failed to download or process OUI data" >&2
|
||||||
|
rm -f "${TEMP_FILE}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Validate we got actual content (should have hundreds of thousands of lines)
|
||||||
|
if [ ! -s "${TEMP_FILE}" ] || [ "$(wc -l < "${TEMP_FILE}")" -lt 1000 ]; then
|
||||||
|
echo "ERROR: OUI data appears invalid or incomplete" >&2
|
||||||
|
rm -f "${TEMP_FILE}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Atomic replacement
|
||||||
|
mv "${TEMP_FILE}" "${OUTPUT_FILE}"
|
||||||
|
echo "Successfully updated IEEE OUI database ($(wc -l < "${OUTPUT_FILE}") entries)"
|
||||||
15
install/production-filesystem/services/start-backend.sh
Executable file
15
install/production-filesystem/services/start-backend.sh
Executable file
@@ -0,0 +1,15 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
cd "${NETALERTX_APP}" || exit 1
|
||||||
|
max_attempts=50 # 10 seconds total (50 * 0.2s)
|
||||||
|
attempt=0
|
||||||
|
while ps ax | grep -v grep | grep -q python3 && [ $attempt -lt $max_attempts ]; do
|
||||||
|
killall -TERM python3 &>/dev/null
|
||||||
|
sleep 0.2
|
||||||
|
((attempt++))
|
||||||
|
done
|
||||||
|
# Force kill if graceful shutdown failed
|
||||||
|
killall -KILL python3 &>/dev/null
|
||||||
|
|
||||||
|
echo "python3 $(cat /services/config/python/backend-extra-launch-parameters 2>/dev/null) -m server > >(tee /app/log/stdout.log) 2> >(tee /app/log/stderr.log >&2)"
|
||||||
|
exec python3 $(cat /services/config/python/backend-extra-launch-parameters 2>/dev/null) -m server > >(tee /app/log/stdout.log) 2> >(tee /app/log/stderr.log >&2)
|
||||||
34
install/production-filesystem/services/start-crond.sh
Executable file
34
install/production-filesystem/services/start-crond.sh
Executable file
@@ -0,0 +1,34 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
echo "Starting crond..."
|
||||||
|
|
||||||
|
crond_pid=""
|
||||||
|
|
||||||
|
cleanup() {
|
||||||
|
status=$?
|
||||||
|
echo "Crond stopped! (exit ${status})"
|
||||||
|
}
|
||||||
|
|
||||||
|
forward_signal() {
|
||||||
|
if [[ -n "${crond_pid}" ]]; then
|
||||||
|
kill -TERM "${crond_pid}" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
while ps ax | grep -v -e grep -e '.sh' | grep crond >/dev/null 2>&1; do
|
||||||
|
killall crond &>/dev/null
|
||||||
|
sleep 0.2
|
||||||
|
done
|
||||||
|
|
||||||
|
trap cleanup EXIT
|
||||||
|
trap forward_signal INT TERM
|
||||||
|
|
||||||
|
echo "/usr/sbin/crond -c \"${SYSTEM_SERVICES_CROND}\" -f -L \"${LOG_CROND}\" >>\"${LOG_CROND}\" 2>&1 &"
|
||||||
|
|
||||||
|
/usr/sbin/crond -c "${SYSTEM_SERVICES_CROND}" -f -L "${LOG_CROND}" >>"${LOG_CROND}" 2>&1 &
|
||||||
|
crond_pid=$!
|
||||||
|
|
||||||
|
wait "${crond_pid}"; status=$?
|
||||||
|
echo -ne " done"
|
||||||
|
exit ${status}
|
||||||
61
install/production-filesystem/services/start-nginx.sh
Executable file
61
install/production-filesystem/services/start-nginx.sh
Executable file
@@ -0,0 +1,61 @@
|
|||||||
|
#! /bin/bash
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
LOG_DIR=${NETALERTX_LOG}
|
||||||
|
RUN_DIR=${SYSTEM_SERVICES_RUN}
|
||||||
|
TMP_DIR=${SYSTEM_SERVICES_RUN_TMP}
|
||||||
|
SYSTEM_NGINX_CONFIG_TEMPLATE="/services/config/nginx/netalertx.conf.template"
|
||||||
|
SYSTEM_NGINX_CONFIG_FILE="/services/config/nginx/conf.active/netalertx.conf"
|
||||||
|
|
||||||
|
# Create directories if they don't exist
|
||||||
|
mkdir -p "${LOG_DIR}" "${RUN_DIR}" "${TMP_DIR}"
|
||||||
|
|
||||||
|
echo "Starting nginx..."
|
||||||
|
|
||||||
|
nginx_pid=""
|
||||||
|
|
||||||
|
cleanup() {
|
||||||
|
status=$?
|
||||||
|
echo "nginx stopped! (exit ${status})"
|
||||||
|
}
|
||||||
|
|
||||||
|
forward_signal() {
|
||||||
|
if [[ -n "${nginx_pid}" ]]; then
|
||||||
|
kill -TERM "${nginx_pid}" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# When in devcontainer we must kill any existing nginx processes
|
||||||
|
while ps ax | grep -v -e "grep" -e "nginx.sh" | grep nginx >/dev/null 2>&1; do
|
||||||
|
killall nginx &>/dev/null || true
|
||||||
|
sleep 0.2
|
||||||
|
done
|
||||||
|
|
||||||
|
TEMP_CONFIG_FILE=$(mktemp "${TMP_DIR}/netalertx.conf.XXXXXX")
|
||||||
|
if envsubst '${LISTEN_ADDR} ${PORT}' < "${SYSTEM_NGINX_CONFIG_TEMPLATE}" > "${TEMP_CONFIG_FILE}" 2>/dev/null; then
|
||||||
|
mv "${TEMP_CONFIG_FILE}" "${SYSTEM_NGINX_CONFIG_FILE}"
|
||||||
|
else
|
||||||
|
echo "Note: Unable to write to ${SYSTEM_NGINX_CONFIG_FILE}. Using default configuration."
|
||||||
|
rm -f "${TEMP_CONFIG_FILE}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
trap cleanup EXIT
|
||||||
|
trap forward_signal INT TERM
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# Execute nginx with overrides
|
||||||
|
# echo the full nginx command then run it
|
||||||
|
echo "nginx -p \"${RUN_DIR}/\" -c \"${SYSTEM_NGINX_CONFIG_FILE}\" -g \"error_log ${NETALERTX_LOG}/nginx-error.log; pid ${RUN_DIR}/nginx.pid; daemon off;\" &"
|
||||||
|
nginx \
|
||||||
|
-p "${RUN_DIR}/" \
|
||||||
|
-c "${SYSTEM_NGINX_CONFIG_FILE}" \
|
||||||
|
-g "error_log ${NETALERTX_LOG}/nginx-error.log; pid ${RUN_DIR}/nginx.pid; daemon off;" &
|
||||||
|
nginx_pid=$!
|
||||||
|
|
||||||
|
wait "${nginx_pid}"
|
||||||
|
nginx_exit=$?
|
||||||
|
echo -ne " done"
|
||||||
|
exit ${nginx_exit}
|
||||||
33
install/production-filesystem/services/start-php-fpm.sh
Executable file
33
install/production-filesystem/services/start-php-fpm.sh
Executable file
@@ -0,0 +1,33 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
echo "Starting php-fpm..."
|
||||||
|
|
||||||
|
php_fpm_pid=""
|
||||||
|
|
||||||
|
cleanup() {
|
||||||
|
status=$?
|
||||||
|
echo "php-fpm stopped! (exit ${status})"
|
||||||
|
}
|
||||||
|
|
||||||
|
forward_signal() {
|
||||||
|
if [[ -n "${php_fpm_pid}" ]]; then
|
||||||
|
kill -TERM "${php_fpm_pid}" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
while ps ax | grep -v grep | grep php-fpm83 >/dev/null; do
|
||||||
|
killall php-fpm83 &>/dev/null
|
||||||
|
sleep 0.2
|
||||||
|
done
|
||||||
|
|
||||||
|
trap cleanup EXIT
|
||||||
|
trap forward_signal INT TERM
|
||||||
|
|
||||||
|
echo "/usr/sbin/php-fpm83 -y \"${PHP_FPM_CONFIG_FILE}\" -F >>\"${LOG_APP_PHP_ERRORS}\" 2>&1 &"
|
||||||
|
/usr/sbin/php-fpm83 -y "${PHP_FPM_CONFIG_FILE}" -F >>"${LOG_APP_PHP_ERRORS}" 2>&1 &
|
||||||
|
php_fpm_pid=$!
|
||||||
|
|
||||||
|
wait "${php_fpm_pid}"
|
||||||
|
echo -ne " done"
|
||||||
|
exit $?
|
||||||
@@ -15,6 +15,7 @@ echo "---------------------------------------------------------"
|
|||||||
INSTALL_DIR=/app
|
INSTALL_DIR=/app
|
||||||
INSTALL_SYSTEM_NAME=ubuntu24
|
INSTALL_SYSTEM_NAME=ubuntu24
|
||||||
INSTALLER_DIR=${INSTALL_DIR}/install/$INSTALL_SYSTEM_NAME
|
INSTALLER_DIR=${INSTALL_DIR}/install/$INSTALL_SYSTEM_NAME
|
||||||
|
REQUIREMENTS_FILE=${INSTALL_DIR}/requirements.txt
|
||||||
CONF_FILE=app.conf
|
CONF_FILE=app.conf
|
||||||
DB_FILE=app.db
|
DB_FILE=app.db
|
||||||
NGINX_CONF_FILE=netalertx.conf
|
NGINX_CONF_FILE=netalertx.conf
|
||||||
@@ -153,7 +154,12 @@ echo
|
|||||||
python3 -m venv "${VENV_DIR}"
|
python3 -m venv "${VENV_DIR}"
|
||||||
source "${VENV_DIR}/bin/activate"
|
source "${VENV_DIR}/bin/activate"
|
||||||
|
|
||||||
pip3 install -r "${INSTALLER_DIR}/requirements.txt" || {
|
if [[ ! -f "${REQUIREMENTS_FILE}" ]]; then
|
||||||
|
echo "[INSTALL] requirements.txt not found at ${REQUIREMENTS_FILE}"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
pip3 install -r "${REQUIREMENTS_FILE}" || {
|
||||||
echo "[INSTALL] Failed to install Python dependencies"
|
echo "[INSTALL] Failed to install Python dependencies"
|
||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
@@ -243,10 +249,10 @@ else
|
|||||||
echo "[INSTALL] The file ieee-oui.txt does not exist. Running update_vendors..."
|
echo "[INSTALL] The file ieee-oui.txt does not exist. Running update_vendors..."
|
||||||
|
|
||||||
# Run the update_vendors.sh script
|
# Run the update_vendors.sh script
|
||||||
if [ -f "${INSTALL_DIR}/back/update_vendors.sh" ]; then
|
if [ -f "${SYSTEM_SERVICES}/update_vendors.sh" ]; then
|
||||||
"${INSTALL_DIR}/back/update_vendors.sh"
|
"${SYSTEM_SERVICES}/update_vendors.sh"
|
||||||
else
|
else
|
||||||
echo "[INSTALL] update_vendors.sh script not found in ${INSTALL_DIR}."
|
echo "[INSTALL] update_vendors.sh script not found in ${SYSTEM_SERVICES}."
|
||||||
fi
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|||||||
32
requirements.txt
Normal file
32
requirements.txt
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
openwrt-luci-rpc
|
||||||
|
asusrouter
|
||||||
|
aiohttp
|
||||||
|
graphene
|
||||||
|
flask
|
||||||
|
flask-cors
|
||||||
|
unifi-sm-api
|
||||||
|
tplink-omada-client
|
||||||
|
wakeonlan
|
||||||
|
pycryptodome
|
||||||
|
requests
|
||||||
|
paho-mqtt
|
||||||
|
scapy
|
||||||
|
cron-converter
|
||||||
|
pytz
|
||||||
|
json2table
|
||||||
|
dhcp-leases
|
||||||
|
pyunifi
|
||||||
|
speedtest-cli
|
||||||
|
chardet
|
||||||
|
python-nmap
|
||||||
|
dnspython
|
||||||
|
librouteros
|
||||||
|
yattag
|
||||||
|
zeroconf
|
||||||
|
simplejson
|
||||||
|
future
|
||||||
|
six
|
||||||
|
urllib3
|
||||||
|
httplib2
|
||||||
|
gunicorn
|
||||||
|
git+https://github.com/foreign-sub/aiofreepybox.git
|
||||||
19
scripts/list-ports.sh
Executable file
19
scripts/list-ports.sh
Executable file
@@ -0,0 +1,19 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# Quick port/service enumerator for host-network dev container.
|
||||||
|
# Shows which processes are bound to key NetAlertX ports.
|
||||||
|
PORTS="20211 20212 9003 9000 5678"
|
||||||
|
printf "%-8s %-22s %-8s %s\n" PORT BIND LISTEN_PID COMMAND
|
||||||
|
for p in $PORTS; do
|
||||||
|
line=$(ss -ltnp 2>/dev/null | awk -v P=":${p}" '$4 ~ P {print $0; exit}')
|
||||||
|
if [ -n "$line" ]; then
|
||||||
|
addr=$(echo "$line" | awk '{print $4}')
|
||||||
|
pid=$(echo "$line" | sed -n 's/.*pid=\([0-9]*\).*/\1/p')
|
||||||
|
cmd="$( [ -n "$pid" ] && ps -o comm= -p "$pid" 2>/dev/null)"
|
||||||
|
printf "%-8s %-22s %-8s %s\n" "$p" "$addr" "${pid:-?}" "${cmd:-?}"
|
||||||
|
else
|
||||||
|
printf "%-8s %-22s %-8s %s\n" "$p" "(not listening)" "-" "-"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
# Show any other NetAlertX-related listeners (nginx, php-fpm, python backend)
|
||||||
|
ss -ltnp 2>/dev/null | egrep 'nginx|php-fpm|python' || true
|
||||||
@@ -1,8 +1,11 @@
|
|||||||
""" CONSTANTS for NetAlertX """
|
""" CONSTANTS for NetAlertX """
|
||||||
|
import os
|
||||||
|
|
||||||
#===============================================================================
|
#===============================================================================
|
||||||
# PATHS
|
# PATHS
|
||||||
#===============================================================================
|
#===============================================================================
|
||||||
|
|
||||||
|
|
||||||
applicationPath = '/app'
|
applicationPath = '/app'
|
||||||
dbFileName = 'app.db'
|
dbFileName = 'app.db'
|
||||||
confFileName = 'app.conf'
|
confFileName = 'app.conf'
|
||||||
@@ -17,8 +20,8 @@ reportTemplatesPath = applicationPath + '/front/report_templates/'
|
|||||||
fullConfFolder = applicationPath + '/config'
|
fullConfFolder = applicationPath + '/config'
|
||||||
fullConfPath = applicationPath + confPath
|
fullConfPath = applicationPath + confPath
|
||||||
fullDbPath = applicationPath + dbPath
|
fullDbPath = applicationPath + dbPath
|
||||||
vendorsPath = '/usr/share/arp-scan/ieee-oui.txt'
|
vendorsPath = os.getenv('VENDORSPATH', '/usr/share/arp-scan/ieee-oui.txt')
|
||||||
vendorsPathNewest = '/usr/share/arp-scan/ieee-oui_all_filtered.txt'
|
vendorsPathNewest = os.getenv('VENDORSPATH_NEWEST', '/usr/share/arp-scan/ieee-oui_all_filtered.txt')
|
||||||
|
|
||||||
default_tz = 'Europe/Berlin'
|
default_tz = 'Europe/Berlin'
|
||||||
|
|
||||||
|
|||||||
@@ -194,20 +194,7 @@ def fixPermissions():
|
|||||||
# Try fixing access rights if needed
|
# Try fixing access rights if needed
|
||||||
chmodCommands = []
|
chmodCommands = []
|
||||||
|
|
||||||
chmodCommands.append(['sudo', 'chmod', 'a+rw', '-R', fullDbPath])
|
|
||||||
chmodCommands.append(['sudo', 'chmod', 'a+rw', '-R', fullConfPath])
|
|
||||||
|
|
||||||
for com in chmodCommands:
|
|
||||||
# Execute command
|
|
||||||
mylog('none', ["[Setup] Attempting to fix permissions."])
|
|
||||||
try:
|
|
||||||
# try runnning a subprocess
|
|
||||||
result = subprocess.check_output (com, universal_newlines=True)
|
|
||||||
except subprocess.CalledProcessError as e:
|
|
||||||
# An error occured, handle it
|
|
||||||
mylog('none', ["[Setup] Fix Failed. Execute this command manually inside of the container: ", ' '.join(com)])
|
|
||||||
mylog('none', [e.output])
|
|
||||||
|
|
||||||
|
|
||||||
#-------------------------------------------------------------------------------
|
#-------------------------------------------------------------------------------
|
||||||
def initialiseFile(pathToCheck, defaultFile):
|
def initialiseFile(pathToCheck, defaultFile):
|
||||||
|
|||||||
582
xdebug.out
Normal file
582
xdebug.out
Normal file
File diff suppressed because one or more lines are too long
Reference in New Issue
Block a user