Spaces:
Running
Running
File size: 32,441 Bytes
a2e32ba e9edc80 82f1c72 7563ce6 047a355 b3f165a e9edc80 047a355 bfbefd8 19ac8e6 5579d71 122c6e9 b341f74 faf5324 aa339ab 94b54d8 e324722 b0f9081 f834392 4c2371e b0f9081 a0cad35 b341f74 bfbefd8 b341f74 e9edc80 ec447d0 7563ce6 82f1c72 907705d 82f1c72 907705d 82f1c72 dd0cd24 3e39985 9c9c963 907705d dd0cd24 907705d 82f1c72 907705d 82f1c72 8d825b0 7563ce6 bfbefd8 48865cf bfbefd8 37bd79c dd0cd24 37bd79c d1ee52b 7563ce6 2e2d1ae 445a4b4 5513b00 edd90d3 40bd491 2906c61 02a45e5 e4b33d5 3f126f3 127f786 93d3551 5513b00 86238c5 7563ce6 ad7838f 7563ce6 9c9c963 7563ce6 ad7838f 7563ce6 dd0cd24 9c9c963 7563ce6 0adfca0 e1a89b1 86238c5 dd0cd24 1a43a47 86238c5 7563ce6 1a43a47 7563ce6 86238c5 16c78e8 86238c5 493367f 0caa43f 493367f 0caa43f efd4578 0caa43f 410f05c 0caa43f 97f055d 48865cf 97f055d 0caa43f 493367f 86238c5 7563ce6 86238c5 493367f 038b0ae 445a4b4 ec447d0 2923221 330477a 5d56be8 612c460 9a40b68 e0a4dec 9c2e351 612c460 5d56be8 3489455 612c460 5d56be8 a9ab5af 21be93f a9ab5af e0a4dec e383080 9c2e351 612c460 e0a4dec a98b469 e0a4dec 5d56be8 9c2e351 5d56be8 9c2e351 e0a4dec 5d56be8 8dd6d44 9c2e351 79bd99a e0a4dec a98b469 9c2e351 e0a4dec 330477a 9c2e351 5d56be8 e0a4dec 330477a 5d56be8 29691bc 79bd99a 4e98c77 b157933 9ad0d79 a98b469 30cb684 7a42f0c 2923221 1bfefb1 71e318b 336718b 2ab4f80 2923221 5a51109 2923221 2a9b787 247757c 7b09368 8f4c812 7b09368 09441e0 2923221 856645f 1bfefb1 2923221 856645f e40f660 ff195d4 5e334b8 dd91eb9 0a545c6 856645f 2923221 bc09640 1321c2f 856645f 2923221 bc09640 bfd6aac 4de0c07 2923221 bc09640 12a6081 a1a3ecc 4de0c07 a1a3ecc 856645f bc09640 6fcfbd6 74b1f4e 6fcfbd6 bc09640 ba064ac bc09640 0cac539 ba064ac 2923221 bc09640 a365b41 2923221 bc09640 3bcb31f 2923221 bc09640 481eafd d08d9bf 481eafd 2923221 23526fa 0db1c35 23526fa e85045f 23526fa 26420ac 23526fa 2923221 1a7f16a c7511da 2923221 7f47b57 a207445 5584ab1 2923221 23526fa 2923221 02a45e5 4c8e202 18d36c3 4c8e202 18d36c3 02a45e5 18d36c3 02a45e5 18d36c3 02a45e5 18d36c3 02a45e5 18d36c3 02a45e5 4c8e202 3e477c1 4595fee e4b33d5 491c891 d8b02a6 e4b33d5 491c891 e4b33d5 d8b02a6 e4b33d5 491c891 e4b33d5 4595fee d8b02a6 e4b33d5 d8b02a6 e4b33d5 3839853 d8b02a6 491c891 d8b02a6 491c891 228a361 3ed682f 2325df5 01d3120 ed1c345 282eefd e721ff8 93f4c9c e721ff8 1ca248d f614c3f c235662 1ca248d e721ff8 282eefd e721ff8 7c1d440 24c29c0 7c1d440 c217f21 7dbf3d5 c217f21 24c29c0 40d1c0a 24c29c0 8cfe43d 24c29c0 2085c6d f8e33fa 195ab37 f5fb09b 9e1fef7 7c1d440 24c29c0 7c1d440 24c29c0 7c1d440 1ca248d 7c1d440 313dd15 7c1d440 f74215d 7c1d440 f74215d 7c1d440 2085c6d 99d4fdf 24c29c0 ce1f267 7dbf3d5 ce1f267 7dbf3d5 24c29c0 b9a247b c217f21 2085c6d 24c29c0 7c1d440 2085c6d 3a3ebb3 7dbf3d5 3a3ebb3 8b26076 2085c6d 7c1d440 24c29c0 2085c6d c217f21 f74215d a6303d1 7c1d440 544f49f 7dbf3d5 7ffcd2c 7dbf3d5 b9a247b 7ffcd2c 0084cd6 838f241 7dbf3d5 838f241 7c1d440 838f241 689b83c 7dbf3d5 d98e04e e436956 dbc6aa8 4f40621 c5b5633 3905113 c5b5633 3905113 c5b5633 dbc6aa8 3905113 9868d92 0084cd6 62fa030 9868d92 c5b5633 9868d92 3905113 dbc6aa8 0084cd6 dbc6aa8 0084cd6 dbc6aa8 e436956 c5b5633 0bc943e dbc6aa8 0bc943e dbc6aa8 0084cd6 dbc6aa8 0bc943e d98e04e a72e499 d98e04e 1ca248d 14165cc ca5570f ea96c20 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 |
from flask import Flask, request, jsonify, redirect, url_for, render_template, send_from_directory, send_file, render_template_string
import os
import sqlite3
from datetime import datetime
import pytz
import io
import base64
from dotenv import load_dotenv
import requests
import json
import logging
import uuid
from io import BytesIO
from tabulate import tabulate
import numpy as np
from scipy.optimize import linprog
from typing import Dict, Any, Union, List
import itertools
import traceback
import logging
from werkzeug.utils import secure_filename
import globs
from api_logic import api
from urllib.parse import urlencode # Добавлен правильный импорт
load_dotenv()
# Инициализация базы данных
def init_db(db_name):
conn = sqlite3.connect(db_name)
cursor = conn.cursor()
# Таблица с системными данными (твоя старая таблица)
cursor.execute('''
CREATE TABLE IF NOT EXISTS system_data (
id INTEGER PRIMARY KEY AUTOINCREMENT,
date_time TEXT,
dey TEXT,
wek TEXT,
v_hid TEXT,
v_min TEXT,
ph TEXT,
ec TEXT,
tS TEXT,
tA TEXT,
hDm TEXT,
sVen TEXT,
onA TEXT,
onB TEXT,
onC TEXT,
nPh TEXT,
nEC TEXT,
nTa TEXT,
nLon TEXT,
nLoff TEXT
)
''')
# **Новая таблица для пользователей бота**
cursor.execute('''
CREATE TABLE IF NOT EXISTS bot_users (
id INTEGER PRIMARY KEY AUTOINCREMENT, -- Уникальный ID
chat_id INTEGER UNIQUE, -- Telegram ID пользователя
created_at TEXT -- Время добавления (ISO формат)
)
''')
conn.commit()
conn.close()
# Глобальные переменные
api_key_sys = os.getenv('api_key') # Берём значение API-ключа
btg_key = os.getenv('btg_key') # Берём значение
btg_id = os.getenv('btg_id') # Берём значение
btg_on = os.getenv('btg_on') # Берём значение
globs.dey = 0
globs.wek = 0
globs.v_hid = 0
globs.v_min = 0
globs.ph = 0
globs.ec = 0
globs.tS = 0
globs.tA = 0
globs.hDm = 0
globs.sVen = 0
globs.onA = 0
globs.onB = 0
globs.onC = 0
globs.ph_eep = 0
globs.ph_on_eep = 0
globs.ec_eep = 0
globs.ec_A_eep = 0
globs.ec_B_eep = 0
globs.ec_C_eep = 0
globs.l_ON_h_eep = 0
globs.l_ON_m_eep = 0
globs.l_OFF_h_eep = 0
globs.l_OFF_m_eep = 0
globs.t_Voz_eep = 0
# Создаем экземпляр Flask-приложения
app = Flask(__name__, template_folder="./")
app.config['DEBUG'] = True
UPLOAD_FOLDER = 'uploads'
ALLOWED_EXTENSIONS = {'jpg', 'jpeg', 'png', 'gif'}
# Создаем папку для загрузок если ее нет
os.makedirs(UPLOAD_FOLDER, exist_ok=True)
# Глобальная переменная для хранения последнего изображения в памяти
latest_image = {"data": None, "filename": None}
# Настроим логирование
logging.basicConfig(level=logging.DEBUG)
# Функция сохранения в базу данных системы автоматизации гидропоники
def save_data_to_db(db_name, data):
try:
conn = sqlite3.connect(db_name)
cursor = conn.cursor()
# ✅ Устанавливаем московское время (UTC+3)
moscow_tz = pytz.timezone("Europe/Moscow")
current_time = datetime.now(moscow_tz).strftime('%Y-%m-%d %H:%M:%S')
# Вставляем данные в таблицу
cursor.execute('''
INSERT INTO system_data (
date_time, dey, wek, v_hid, v_min, ph, ec, tS, tA, hDm, sVen, onA, onB, onC, nPh, nEC, nTa, nLon, nLoff
) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
''', (
current_time, # ✅ Дата и время по Москве
data['dey'], data['wek'], data['v_hid'], data['v_min'], data['ph'], data['ec'],
data['tS'], data['tA'], data['hDm'], data['sVen'], data['onA'], data['onB'],
data['onC'], data['nPh'], data['nEC'], data['nTa'], data['nLon'], data['nLoff']
))
conn.commit()
conn.close()
except Exception as e:
return jsonify({'status': 'error', 'message': str(e)}), 500
# Маршрут сохранения в базу
@app.route('/sav_db_api', methods=['GET'])
def sav_db_api():
# Инициализируем базу данных
init_db('system_data.db')
# Получаем данные из запроса
data = {
'dey': request.args.get('dey'),
'wek': request.args.get('wek'),
'v_hid': request.args.get('v_hid'),
'v_min': request.args.get('v_min'),
'ph': request.args.get('ph'),
'ec': request.args.get('ec'),
'tS': request.args.get('tS'),
'tA': request.args.get('tA'),
'hDm': request.args.get('hDm'),
'sVen': request.args.get('sVen'),
'onA': request.args.get('onA'),
'onB': request.args.get('onB'),
'onC': request.args.get('onC'),
'nPh': request.args.get('nPh'),
'nEC': request.args.get('nEC'),
'nTa': request.args.get('nTa'),
'nLon': request.args.get('nLon'),
'nLoff': request.args.get('nLoff')
}
# Проверяем, что все необходимые параметры переданы
required_params = ['dey', 'wek', 'v_hid', 'v_min', 'ph', 'ec', 'tS', 'tA', 'hDm', 'sVen', 'onA', 'onB', 'onC', 'nPh', 'nEC', 'nTa', 'nLon', 'nLoff']
for param in required_params:
if data[param] is None:
return jsonify({'status': 'error', 'message': f'Отсутствует параметр: {param}'}), 400
# Сохраняем данные в базу
save_data_to_db('system_data.db', data)
# Возвращаем ответ
return jsonify({'status': 'success', 'message': 'Save OK'})
# Проверка входа на страницы
@app.route('/page_key', methods=['GET'])
def check_api_key():
api_sys_param = request.args.get('api_sys') # Получаем параметр из запроса
if api_sys_param == api_key_sys:
return jsonify({"status": "ok"}), 200 # ✅ Совпадает — отправляем "ok"
else:
return jsonify({"status": "error", "message": "Invalid API key"}), 403 # ❌ Ошибка 403
# Тестовый запрос с установки
@app.route('/test_server', methods=['GET'])
def test_server():
api_key_param = request.args.get('api_sys') # Получаем параметр из запроса
err_ser = 1 if api_key_param == api_key_sys else 0 # Проверяем совпадение ключей
return jsonify(err_ser=err_ser)
@app.route('/test_server_str', methods=['GET'])
def test_server_str():
api_key_param = request.args.get('api_sys')
err_ser = "1" if api_key_param == api_key_sys else "0"
return err_ser # Возвращаем строку "1" или "0"
# Тестовый запрос с установки
@app.route('/btg_teleg', methods=['GET'])
def btg_teleg():
api_key_param = request.args.get('api_sys') # Получаем параметр из запроса
return jsonify(btg_key_ser=btg_key,btg_id_ser=btg_id,btg_on_ser=btg_on)
# Маршрут для вывода всех данных из таблицы
@app.route('/get_all_data', methods=['GET'])
def get_all_data():
try:
conn = sqlite3.connect('system_data.db')
cursor = conn.cursor()
# Выполняем запрос для получения всех данных из таблицы
cursor.execute('SELECT * FROM system_data')
rows = cursor.fetchall()
# Получаем названия столбцов
column_names = [description[0] for description in cursor.description]
# Преобразуем данные в формат JSON
data = []
for row in rows:
data.append(dict(zip(column_names, row)))
conn.close()
# Возвращаем данные в формате JSON
return jsonify(data)
except Exception as e:
return jsonify({'status': 'error', 'message': str(e)}), 500
# Удаление базы
@app.route('/delite_db', methods=['GET'])
def delete_db():
try:
conn = sqlite3.connect("system_data.db") # Используем вашу БД
cursor = conn.cursor()
# ✅ Удаляем все записи из таблицы
cursor.execute("DELETE FROM system_data")
# ✅ Сбрасываем автоинкрементный счётчик ID (для SQLite)
cursor.execute("DELETE FROM sqlite_sequence WHERE name='system_data'")
conn.commit()
conn.close()
return jsonify({'status': 'ok', 'message': 'База данных успешно очищена'})
except Exception as e:
return jsonify({'status': 'error', 'message': str(e)}), 500
@app.route('/plot_week', methods=['GET'])
def plot_week():
try:
# Получаем номер недели из параметров запроса
week_number = request.args.get('week', default=1, type=int)
week_number = max(1, min(30, week_number)) # Ограничиваем диапазон 1-30
# Подключаемся к базе данных
conn = sqlite3.connect('system_data.db')
cursor = conn.cursor()
# Проверяем существование таблицы
cursor.execute("SELECT name FROM sqlite_master WHERE type='table' AND name='system_data'")
table_exists = cursor.fetchone()
if not table_exists:
conn.close()
return render_template('plot_week.html', data=None, week_number=week_number, table_exists=False)
# Запрашиваем данные за выбранную неделю
cursor.execute('''
SELECT date_time, dey, ph, ec, tS, tA, hDm, sVen, onA, onB, onC, v_hid, v_min
FROM system_data
WHERE wek = ?
ORDER BY date_time
''', (str(week_number),)) # Приводим week_number к строке, так как wek имеет тип TEXT
rows = cursor.fetchall()
conn.close()
# Если данных нет
if not rows:
return render_template('plot_week.html', data=None, week_number=week_number, table_exists=True)
# Формируем данные для JSON
data = {
'week': week_number,
'dates': [row[0] for row in rows],
'days_of_week': [int(row[1]) if row[1] else 0 for row in rows], # Преобразуем dey в int
'ph': [float(row[2]) if row[2] else 0.0 for row in rows], # pH
'ec': [float(row[3]) if row[3] else 0.0 for row in rows], # EC
'tS': [float(row[4]) if row[4] else 0.0 for row in rows], # Температура раствора
'tA': [float(row[5]) if row[5] else 0.0 for row in rows], # Температура воздуха
'hDm': [float(row[6]) if row[6] else 0.0 for row in rows], # Влажность воздуха
'sVen': [float(row[7]) if row[7] else 0.0 for row in rows], # Обороты вентилятора
'onA': [float(row[8]) if row[8] else 0.0 for row in rows], # Насос A
'onB': [float(row[9]) if row[9] else 0.0 for row in rows], # Насос B
'onC': [float(row[10]) if row[10] else 0.0 for row in rows], # Насос C
'sus': [f"{row[11]}:{row[12]}" if row[11] and row[12] else "0:0" for row in rows] # Объединяем v_hid и v_min
}
# Отправляем данные в HTML-шаблон
return render_template('plot_week.html', data=data, week_number=week_number, table_exists=True)
except Exception as e:
# Логируем ошибку в консоль для отладки
print(f"Ошибка: {str(e)}")
return render_template('plot_week.html', data=None, week_number=week_number, table_exists=True, message=f"Ошибка: {str(e)}")
@app.route("/")
def index():
return flask.render_template('index.html')
@app.route('/online', methods=['GET'])
def online():
return render_template('online.html')
@app.route('/table', methods=['GET'])
def table():
return render_template('table.html')
@app.route('/online_api', methods=['GET'])
def online_api():
# Устанавливаем московское время (UTC+3)
moscow_tz = pytz.timezone("Europe/Moscow")
current_time = datetime.now(moscow_tz)
# Форматируем дату и время отдельно
date = current_time.strftime('%Y-%m-%d') # Например, "2025-03-23"
time = current_time.strftime('%H:%M:%S') # Например, "14:35:42"
return jsonify(
dey=globs.dey,
wek=globs.wek,
v_hid=globs.v_hid,
v_min=globs.v_min,
ph=globs.ph,
ec=globs.ec,
tS=globs.tS,
tA=globs.tA,
hDm=globs.hDm,
sVen=globs.sVen,
rFul=globs.rFul,
rLi=globs.rLi,
rWat=globs.rWat,
rRas=globs.rRas,
rPH=globs.rPH,
rEC=globs.rEC,
rSl=globs.rSl,
rLe=globs.rLe,
alW=globs.alW,
ec_A_eep=globs.ec_A_eep,
ec_B_eep=globs.ec_B_eep,
ec_C_eep=globs.ec_C_eep,
date=date, # Добавляем дату
time=time # Добавляем время
)
@app.route('/settings', methods=['GET'])
def settings():
return render_template('settings.html')
@app.route('/settings_api', methods=['GET'])
def settings_api():
return jsonify(ph_eep=globs.ph_eep,
ph_on_eep=globs.ph_on_eep,
ec_eep=globs.ec_eep,
ec_A_eep=globs.ec_A_eep,
ec_B_eep=globs.ec_B_eep,
ec_C_eep=globs.ec_C_eep,
l_ON_h_eep=globs.l_ON_h_eep,
l_ON_m_eep=globs.l_ON_m_eep,
l_OFF_h_eep=globs.l_OFF_h_eep,
l_OFF_m_eep=globs.l_OFF_m_eep,
t_Voz_eep=globs.t_Voz_eep,
set_st=globs.set_status
)
@app.route('/pH_set', methods=['GET'])
def set_pH_value():
ph_value = request.args.get('value')
globs.ph_set = ph_value
globs.eep_set = 1
return "pH value set successfully"
@app.route('/ph_on_set', methods=['GET'])
def ph_on_value():
ph_on_value = request.args.get('value')
globs.ph_on_set = ph_on_value
globs.eep_set = 2
return "EC value set successfully"
@app.route('/EC_set', methods=['GET'])
def set_EC_value():
ec_value = request.args.get('value')
globs.ec_set = ec_value
globs.eep_set = 3
return "EC value set successfully"
@app.route('/ec_A_set', methods=['GET'])
def ec_A_setValue():
ec_A_setValue = request.args.get('value')
globs.ec_A_set = ec_A_setValue
globs.eep_set = 4
return "EC value set successfully"
@app.route('/ec_B_set', methods=['GET'])
def ec_B_setValue():
ec_B_setValue = request.args.get('value')
globs.ec_B_set = ec_B_setValue
globs.eep_set = 5
return "EC value set successfully"
@app.route('/ec_C_set', methods=['GET'])
def ec_C_setValue():
ec_C_setValue = request.args.get('value')
globs.ec_C_set = ec_C_setValue
globs.eep_set = 6
return "EC value set successfully"
@app.route('/l_ON_set', methods=['GET'])
def l_ON_set():
globs.l_ON_h_set = request.args.get('l_ON_h_set')
globs.l_ON_m_set = request.args.get('l_ON_m_set')
globs.eep_set = 7
return "EC value set successfully"
@app.route('/l_OFF_set', methods=['GET'])
def l_OFF_set():
globs.l_OFF_h_set = request.args.get('l_OFF_h_set')
globs.l_OFF_m_set = request.args.get('l_OFF_m_set')
globs.eep_set = 8
return "EC value set successfully"
@app.route('/t_Voz_eep_set', methods=['GET'])
def t_Voz_eep_set():
t_Voz_eep_set = request.args.get('value')
globs.t_Voz_set = t_Voz_eep_set
globs.eep_set = 9
return "EC value set successfully"
@app.route('/but_start', methods=['GET'])
def but_start():
globs.eep_set = 10
return jsonify(value_set="start")
@app.route('/but_stop', methods=['GET'])
def but_stop():
globs.eep_set = 11
return jsonify(value_set="stop")
@app.route('/but_res', methods=['GET'])
def but_res():
globs.eep_set = 12
return jsonify(value_set="res")
@app.route('/but_sliv', methods=['GET'])
def but_sliv():
globs.eep_set = 13
return jsonify(value_set="sliv")
@app.route("/api", methods=['GET'])
def handle_api():
response = api()
return response
@app.route("/save_db", methods=['GET'])
def handle_save_db():
response = save_db()
return response
@app.route('/set_res')
def set_res():
globs.eep_set = 0
return jsonify(value_set="reset")
def allowed_file(filename):
return '.' in filename and \
filename.rsplit('.', 1)[1].lower() in ALLOWED_EXTENSIONS
@app.route('/upload', methods=['POST'])
def upload_file():
if 'file' not in request.files:
return jsonify({"error": "No file part"}), 400
file = request.files['file']
if file.filename == '':
return jsonify({"error": "No selected file"}), 400
if not allowed_file(file.filename):
return jsonify({"error": "Invalid file type"}), 400
# Генерируем имя для файла с использованием времени
timestamp = datetime.now().strftime('%Y.%m.%d_%H:%M:%S_')
filename = timestamp + file.filename
save_path = os.path.join(UPLOAD_FOLDER, filename)
# Открываем файл для записи и собираем его части
try:
with open(save_path, 'wb') as f:
while chunk := file.read(1024): # Чтение и запись данных частями
f.write(chunk)
return jsonify({
"message": "File uploaded successfully",
"filename": filename,
"path": save_path
}), 200
except Exception as e:
return jsonify({"error": str(e)}), 500
@app.route('/uploads/<filename>', methods=['GET'])
def uploaded_file(filename):
return send_from_directory(UPLOAD_FOLDER, filename)
# 🧠 2. Маршрут: сохраняет файл только в память (BytesIO)
# Маршрут для загрузки файла в память
# Загрузка изображения в память (в виде байтов)
@app.route('/upload_memory', methods=['POST'])
def upload_file_to_memory():
if 'file' not in request.files:
return jsonify({"error": "No file part"}), 400
file = request.files['file']
if file.filename == '':
return jsonify({"error": "No selected file"}), 400
if not file.filename.lower().endswith(('.jpg', '.jpeg', '.png')):
return jsonify({"error": "Invalid file type"}), 400
timestamp = datetime.now().strftime('%Y.%m.%d_%H:%M:%S_')
filename = timestamp + file.filename
try:
# Читаем весь файл в байты
file_bytes = file.read()
# Сохраняем в переменную
latest_image["data"] = file_bytes
latest_image["filename"] = filename
return jsonify({
"message": "Image uploaded successfully",
"filename": filename
}), 200
except Exception as e:
return jsonify({"error": str(e)}), 500
# Получение последнего изображения
@app.route('/last_image', methods=['GET'])
def get_last_image():
if not latest_image["data"]:
return jsonify({"error": "No image available"}), 404
# Возвращаем через новый BytesIO каждый раз
return send_file(
BytesIO(latest_image["data"]),
mimetype='image/jpeg',
download_name=latest_image["filename"]
)
@app.route('/view_image', methods=['GET'])
def view_image():
return render_template('show_image.html')
@app.route('/nutri_call', methods=['GET'])
def nutri_call():
return render_template('nutri_call.html')
from tabulate import tabulate
# Входные данные (пример)
INPUT_DATA = {
"fertilizerConstants": {
"Кальциевая селитра": {"N (NO3-)": 0.11863, "Ca": 0.16972},
"Калий азотнокислый": {"N (NO3-)": 0.13854, "K": 0.36672},
"Аммоний азотнокислый": {"N (NO3-)": 0.17499, "N (NH4+)": 0.17499},
"Сульфат магния": {"Mg": 0.10220, "S": 0.13483},
"Монофосфат калия": {"P": 0.22761, "K": 0.28731},
"Калий сернокислый": {"K": 0.44874, "S": 0.18401},
"Кальций хлорид": {"Ca": 0.272, "Cl": 0.483}
},
"profileSettings": {
"P": 31, "K": 210, "Mg": 24, "Ca": 82, "S": 57.5, "CaCl": 18.5,
"NO3_RAT": 8.25, "TOTAL_NITROG": 125, "liters": 100,
"activation_cacl": 5, # Активация CaCl
"enhancement_cacl": 0.1
}
}
from tabulate import tabulate
# Входные данные (пример)
INPUT_DATA = {
"fertilizerConstants": {
"Кальциевая селитра": {"N (NO3-)": 0.11863, "Ca": 0.16972},
"Калий азотнокислый": {"N (NO3-)": 0.13854, "K": 0.36672},
"Аммоний азотнокислый": {"N (NO3-)": 0.17499, "N (NH4+)": 0.17499},
"Сульфат магния": {"Mg": 0.10220, "S": 0.13483},
"Монофосфат калия": {"P": 0.22761, "K": 0.28731},
"Калий сернокислый": {"K": 0.44874, "S": 0.18401}
},
"profileSettings": {
"P": 31, "K": 210, "Mg": 24, "Ca": 82, "S": 57.5,
"NO3_RAT": 8.25, "TOTAL_NITROG": 125, "liters": 100
}
}
class NutrientCalculator:
def __init__(self, fertilizer_constants, profile_settings, liters, rounding_precision):
self.fertilizers = fertilizer_constants
self.profile = profile_settings
self.volume = liters
self.rounding_precision = rounding_precision
total_parts = self.profile["NO3_RAT"] + 1
self.target = {
'P': self.profile["P"],
'K': self.profile["K"],
'Mg': self.profile["Mg"],
'Ca': self.profile["Ca"], # Общий кальций
'S': self.profile["S"],
'N (NO3-)': self.profile["TOTAL_NITROG"] * (self.profile["NO3_RAT"] / total_parts),
'N (NH4+)': self.profile["TOTAL_NITROG"] * (1 / total_parts)
}
self.actual = {k: 0.0 for k in self.target} # CaCl не учитываем в фактическом балансе
self.results = {fert: {'граммы': 0.0} for fert in self.fertilizers}
def calculate(self):
# Добавляем аммонийный азот
self._apply_fertilizer("Аммоний азотнокислый", "N (NH4+)", self.target["N (NH4+)"])
# Добавляем фосфор
self._apply_fertilizer("Монофосфат калия", "P", self.target["P"])
# Добавляем магний
self._apply_fertilizer("Сульфат магния", "Mg", self.target["Mg"])
# Балансируем калий и серу
self._balance_k_s()
# Распределяем кальций
self._distribute_calcium()
# Балансируем нитратный азот
no3_needed = self.target["N (NO3-)"] - self.actual["N (NO3-)"]
if no3_needed > 0:
self._apply_fertilizer("Калий азотнокислый", "N (NO3-)", no3_needed)
return self._verify_results()
def _apply_fertilizer(self, name, element, target_ppm):
if name not in self.fertilizers:
raise KeyError(f"Удобрение '{name}' не найдено!")
content = self.fertilizers[name].get(element, 0)
if content == 0:
print(f"ПРЕДУПРЕЖДЕНИЕ: Удобрение '{name}' не содержит элемент '{element}'")
return
grams = (target_ppm * self.volume) / (content * 1000)
self.results[name]['граммы'] += round(grams, self.rounding_precision)
for el, val in self.fertilizers[name].items():
added_ppm = (grams * val * 1000) / self.volume
if el in self.actual:
self.actual[el] += round(added_ppm, self.rounding_precision)
def _balance_k_s(self):
k_needed = self.target["K"] - self.actual["K"]
s_needed = self.target["S"] - self.actual["S"]
# Используем "Калий сернокислый" для баланса K и S
if k_needed > 0 and s_needed > 0:
k_fraction = self.fertilizers["Калий сернокислый"].get("K", 0)
s_fraction = self.fertilizers["Калий сернокислый"].get("S", 0)
if k_fraction == 0 or s_fraction == 0:
print("ПРЕДУПРЕЖДЕНИЕ: Удобрение 'Калий сернокислый' содержит нулевые значения!")
return
k_from_k2so4 = min(k_needed, s_needed * k_fraction / s_fraction)
self._apply_fertilizer("Калий сернокислый", "K", k_from_k2so4)
# Оставшийся калий добавляем через "Калий азотнокислый"
remaining_k = self.target["K"] - self.actual["K"]
if remaining_k > 0:
self._apply_fertilizer("Калий азотнокислый", "K", remaining_k)
def _distribute_calcium(self):
ca_target = self.target["Ca"] # Общий кальций
# Весь кальций добавляется через "Кальциевую селитру"
if ca_target > 0:
self._apply_fertilizer("Кальциевая селитра", "Ca", ca_target)
def _verify_results(self):
deficits = {}
for el in self.target:
diff = self.target[el] - self.actual[el]
if abs(diff) > 0.1:
deficits[el] = round(diff, self.rounding_precision)
return {
'fertilizers': {k: round(v['граммы'], self.rounding_precision) for k, v in self.results.items()},
'actual_profile': {k: round(v, self.rounding_precision) for k, v in self.actual.items()},
'deficits': deficits,
'total_ppm': round(sum(self.actual.values()), self.rounding_precision)
}
def generate_report(self, results):
fert_table = []
for name, data in results['fertilizers'].items():
fert_table.append([name, f"{data} г"])
element_table = []
for el in sorted(self.target.keys()):
if el == "CaCl":
continue
element_table.append([
el,
f"{self.target[el]} ppm",
f"{results['actual_profile'][el]} ppm",
f"{results['actual_profile'][el] - self.target[el]:+.2f} ppm"
])
report = "РЕКОМЕНДУЕМЫЕ УДОБРЕНИЯ:\n"
report += tabulate(fert_table, headers=["Удобрение", "Количество"], tablefmt="grid")
report += "\n\nБАЛАНС ЭЛЕМЕНТОВ:\n"
report += tabulate(element_table,
headers=["Элемент", "Цель", "Факт", "Отклонение"],
tablefmt="grid")
report += f"\n\nОбщая концентрация: {results['total_ppm']} ppm"
if results['deficits']:
report += "\n\nВНИМАНИЕ: Обнаружены небольшие отклонения:"
for el, diff in results['deficits'].items():
report += f"\n- {el}: не хватает {abs(diff)} ppm"
return report
# Извлекаем данные из INPUT_DATA
fertilizer_constants = INPUT_DATA["fertilizerConstants"]
profile_settings = INPUT_DATA["profileSettings"]
liters = profile_settings["liters"]
rounding_precision = 3
# Создаем калькулятор
calculator = NutrientCalculator(
fertilizer_constants=fertilizer_constants,
profile_settings=profile_settings,
liters=liters,
rounding_precision=rounding_precision
)
# Запуск расчета
results = calculator.calculate()
print(calculator.generate_report(results))
@app.route('/calculation', methods=['POST'])
def handle_calculation():
try:
# 1. Получаем и парсим данные
data = request.get_json()
if not data:
return jsonify({"error": "No JSON data received"}), 400
# Логируем входные данные для отладки
print("\n=== ВХОДНЫЕ ДАННЫЕ ===")
print(json.dumps(data, indent=2, ensure_ascii=False))
# 2. Извлекаем основные параметры
fertilizer_constants = data.get("fertilizerConstants", {})
profile_settings = data.get("profileSettings", {})
# Проверяем наличие необходимых полей
if not fertilizer_constants or not profile_settings:
return jsonify({"error": "Missing fertilizerConstants or profileSettings"}), 400
# 3. Извлекаем дополнительные параметры
liters = profile_settings.get("liters", 100) # Объем раствора (литры)
rounding_precision = profile_settings.get("rounding_precision", 3) # Точность округления
# 4. Создаем и запускаем калькулятор
calculator = NutrientCalculator(
fertilizer_constants=fertilizer_constants,
profile_settings=profile_settings,
liters=liters,
rounding_precision=rounding_precision
)
results = calculator.calculate()
# 5. Формируем дополнительные данные
element_contributions = {}
for fert_name in calculator.fertilizers.keys():
grams = calculator.results[fert_name]['граммы']
element_contributions[fert_name] = {}
for element, percent in calculator.fertilizers[fert_name].items():
added_ppm = (grams * percent * 1000) / calculator.volume
element_contributions[fert_name][element] = round(added_ppm, rounding_precision)
# 6. Формируем полный ответ
response = {
"fertilizers": results['fertilizers'],
"actual_profile": results['actual_profile'],
"deficits": results['deficits'],
"total_ppm": results['total_ppm'],
"element_contributions": element_contributions,
"nitrogen_ratios": {
"NH4_RATIO": 1,
"NO3_RATIO": profile_settings.get("NO3_RAT", 0),
"TOTAL_NITROGEN": profile_settings.get("TOTAL_NITROG", 0)
}
}
return jsonify(response), 200
except Exception as e:
error_msg = f"Server error: {str(e)}"
print(error_msg)
return jsonify({
"error": error_msg,
"fertilizers": {},
"actual_profile": {},
"deficits": {},
"element_contributions": {},
"total_ppm": 0
}), 500
if __name__ == '__main__':
app.run(host='0.0.0.0', port=int(os.environ.get('PORT', 7860)))
|