mirror of
https://github.com/Nikolay-Shirokov/cc-1c-skills.git
synced 2026-06-12 17:04:57 +03:00
161 lines
5.4 KiB
Python
161 lines
5.4 KiB
Python
#!/usr/bin/env python3
|
|
# web-info v1.0 — Apache & 1C publication status
|
|
# Source: https://github.com/Nikolay-Shirokov/cc-1c-skills
|
|
|
|
"""
|
|
Статус Apache HTTP Server и публикаций 1С.
|
|
Показывает состояние Apache, список опубликованных баз
|
|
и последние ошибки из error.log.
|
|
"""
|
|
|
|
import argparse
|
|
import os
|
|
import re
|
|
import sys
|
|
|
|
import psutil
|
|
|
|
|
|
def get_httpd_by_exe(httpd_exe_norm):
|
|
"""Get httpd processes matching our exe path."""
|
|
ours = []
|
|
foreign = []
|
|
for p in psutil.process_iter(['pid', 'name', 'exe']):
|
|
try:
|
|
if p.info['name'] and 'httpd' in p.info['name'].lower():
|
|
if p.info['exe'] and os.path.normcase(os.path.normpath(p.info['exe'])) == httpd_exe_norm:
|
|
ours.append(p)
|
|
else:
|
|
foreign.append(p)
|
|
except (psutil.NoSuchProcess, psutil.AccessDenied):
|
|
pass
|
|
return ours, foreign
|
|
|
|
|
|
def main():
|
|
sys.stdout.reconfigure(encoding="utf-8")
|
|
sys.stderr.reconfigure(encoding="utf-8")
|
|
parser = argparse.ArgumentParser(description='Apache & 1C publication status', allow_abbrev=False)
|
|
parser.add_argument('-ApachePath', type=str, default='', help='Apache root (default: tools\\apache24)')
|
|
args = parser.parse_args()
|
|
|
|
# --- Resolve ApachePath ---
|
|
apache_path = args.ApachePath
|
|
if not apache_path:
|
|
script_dir = os.path.dirname(os.path.abspath(__file__))
|
|
project_root = os.path.dirname(os.path.dirname(os.path.dirname(os.path.dirname(script_dir))))
|
|
apache_path = os.path.join(project_root, 'tools', 'apache24')
|
|
|
|
# --- Check Apache installation ---
|
|
httpd_exe = os.path.join(apache_path, 'bin', 'httpd.exe')
|
|
|
|
print('=== Apache Web Server ===')
|
|
|
|
if not os.path.exists(httpd_exe):
|
|
print('Status: Не установлен')
|
|
print(f'Path: {apache_path} (не найден)')
|
|
print('')
|
|
print('Используйте /web-publish для установки Apache.')
|
|
sys.exit(0)
|
|
|
|
# --- Check process (only our Apache) ---
|
|
httpd_exe_norm = os.path.normcase(os.path.normpath(os.path.realpath(httpd_exe)))
|
|
our_proc, foreign_proc = get_httpd_by_exe(httpd_exe_norm)
|
|
|
|
if our_proc:
|
|
pids = ', '.join(str(p.pid) for p in our_proc)
|
|
print(f'Status: Запущен (PID: {pids})')
|
|
else:
|
|
print('Status: Остановлен')
|
|
|
|
if foreign_proc:
|
|
fp = foreign_proc[0]
|
|
try:
|
|
fpath = fp.info['exe'] or '?'
|
|
except Exception:
|
|
fpath = '?'
|
|
print(f'[WARN] Обнаружен сторонний Apache (PID: {fp.pid}, {fpath})')
|
|
|
|
print(f'Path: {apache_path}')
|
|
|
|
# --- Parse httpd.conf ---
|
|
conf_file = os.path.join(apache_path, 'conf', 'httpd.conf')
|
|
if not os.path.exists(conf_file):
|
|
print('Config: httpd.conf не найден')
|
|
sys.exit(0)
|
|
|
|
with open(conf_file, 'r', encoding='utf-8-sig') as f:
|
|
conf_content = f.read()
|
|
|
|
# Extract port from global block
|
|
port = '\u2014'
|
|
m = re.search(r'(?m)^Listen\s+(\d+)', conf_content)
|
|
if m:
|
|
port = m.group(1)
|
|
print(f'Port: {port}')
|
|
|
|
# Extract wsap24 path
|
|
m = re.search(r'LoadModule\s+_1cws_module\s+"([^"]+)"', conf_content)
|
|
if m:
|
|
print(f'Module: {m.group(1)}')
|
|
|
|
# --- Publications ---
|
|
print('')
|
|
print('=== Опубликованные базы ===')
|
|
|
|
pub_pattern = r'# --- 1C Publication: (.+?) ---'
|
|
pub_matches = re.findall(pub_pattern, conf_content)
|
|
|
|
if not pub_matches:
|
|
print('(нет публикаций)')
|
|
else:
|
|
for app_name in pub_matches:
|
|
# Read default.vrd for this publication
|
|
vrd_path = os.path.join(apache_path, 'publish', app_name, 'default.vrd')
|
|
ib_info = '\u2014'
|
|
vrd_content = ''
|
|
if os.path.exists(vrd_path):
|
|
with open(vrd_path, 'r', encoding='utf-8-sig') as f:
|
|
vrd_content = f.read()
|
|
m = re.search(r'ib="([^"]*)"', vrd_content)
|
|
if m:
|
|
ib_info = m.group(1).replace('"', '"')
|
|
|
|
# Detect published services
|
|
svc_tags = []
|
|
if vrd_content:
|
|
if re.search(r'<ws\s', vrd_content):
|
|
svc_tags.append('WS')
|
|
if re.search(r'<httpServices\s', vrd_content):
|
|
svc_tags.append('HTTP')
|
|
if re.search(r'enableStandardOdata\s*=\s*"true"', vrd_content):
|
|
svc_tags.append('OData')
|
|
svc_label = ' [' + ' '.join(svc_tags) + ']' if svc_tags else ''
|
|
|
|
url = f'http://localhost:{port}/{app_name}'
|
|
print(f' {app_name} {url} {ib_info}{svc_label}')
|
|
|
|
# --- Error log ---
|
|
print('')
|
|
print('=== Последние ошибки ===')
|
|
|
|
error_log = os.path.join(apache_path, 'logs', 'error.log')
|
|
if os.path.exists(error_log):
|
|
try:
|
|
with open(error_log, 'r', encoding='utf-8-sig', errors='replace') as f:
|
|
all_lines = f.readlines()
|
|
tail_lines = all_lines[-5:] if len(all_lines) >= 5 else all_lines
|
|
if tail_lines:
|
|
for line in tail_lines:
|
|
print(f' {line.rstrip()}')
|
|
else:
|
|
print('(пусто)')
|
|
except Exception:
|
|
print('(ошибка чтения)')
|
|
else:
|
|
print('(нет файла)')
|
|
|
|
|
|
if __name__ == '__main__':
|
|
main()
|