commit c0b4ca1021b38251efb1c00829ed87f74d336048 Author: Ana Custura Date: Tue Feb 17 08:42:33 2026 +0000 Initial commit diff --git a/app/__init__.py b/app/__init__.py new file mode 100644 index 0000000..65badd7 --- /dev/null +++ b/app/__init__.py @@ -0,0 +1,23 @@ +from flask import Flask +from config import Config +from flask_sqlalchemy import SQLAlchemy +from flask_migrate import Migrate +from flask_login import LoginManager +from flask import request +from flask_babel import Babel +from flask_babel import lazy_gettext as _l + +def get_locale(): + return request.accept_languages.best_match(app.config['LANGUAGES']) +app = Flask(__name__) +babel = Babel(app, locale_selector=get_locale) +app.config.from_object(Config) +db = SQLAlchemy(app) +migrate = Migrate(app, db) +login = LoginManager(app) +login.login_view = 'login' +login.login_message = _l('Please log in to access this page.') +from app import routes, models + +from app.commands import seed_settings_command +app.cli.add_command(seed_settings_command) \ No newline at end of file diff --git a/app/commands.py b/app/commands.py new file mode 100644 index 0000000..1e02ec2 --- /dev/null +++ b/app/commands.py @@ -0,0 +1,49 @@ +import click +from app import db +from flask import current_app +from app.models import Setting, User + + +def seed_defaults(): + defaults = { + "butterbox_name": current_app.config["BUTTERBOX_NAME"], + "butterbox_logo": current_app.config["BUTTERBOX_LOGO"], + "ssid": current_app.config["BUTTERBOX_SSID"], + "wifi_password": current_app.config["BUTTERBOX_WIFI_PASSWORD"], + "disable_access_point": current_app.config["DISABLE_ACCESS_POINT"], + "apply_changes": "false", + "onboarding_complete": "false", + "lock_root_password": "false", + "disable_file_viewer": current_app.config["DISABLE_FILE_VIEWER"], + "disable_map_viewer": current_app.config["DISABLE_MAP_VIEWER"], + "disable_chat": current_app.config["DISABLE_CHAT"], + "disable_app_store": current_app.config["DISABLE_APP_STORE"], + "ssh_password": "", + "admin_password": current_app.config["ADMIN_PASSWORD"], + } + + for key, value in defaults.items(): + exists = Setting.query.filter_by(key=key).first() + if not exists: + db.session.add(Setting(key=key, value=value)) + click.echo("Created new setting {}".format(key)) + else: + click.echo("Found existing setting {}".format(key)) + db.session.commit() + + admin_user_exists = User.query.filter_by(username='admin').first() + if not admin_user_exists: + u = User(username='admin') + u.set_password('admin') + db.session.add(u) + db.session.commit() + click.echo("Created new admin user") + else: + click.echo("Found existing admin user") + + +@click.command("seed-settings") +def seed_settings_command(): + """Seed default settings into the database (only if missing).""" + seed_defaults() + click.echo("Finished seeding default settings.") diff --git a/app/forms.py b/app/forms.py new file mode 100644 index 0000000..43548bd --- /dev/null +++ b/app/forms.py @@ -0,0 +1,32 @@ +from flask_wtf import FlaskForm +from flask_wtf.file import FileAllowed, FileRequired +from wtforms import StringField, PasswordField, SubmitField, BooleanField, FileField +from wtforms.validators import DataRequired +from flask_babel import lazy_gettext as _l + +class LoginForm(FlaskForm): + username = StringField(_l('Username'), validators=[DataRequired()]) + password = PasswordField(_l('Password'), validators=[DataRequired()]) + submit = SubmitField(_l('Sign In')) + remember_me = BooleanField('Remember Me') + + +class SettingsForm(FlaskForm): + # Access point settings + ssid = StringField('SSID', validators=[DataRequired()]) + wifi_password = PasswordField(_l('WiFi Password')) + disable_access_point = BooleanField(_l('Disable Access Point')) + # Customisation settings + butterbox_name = StringField(_l('Butterbox Name'), validators=[DataRequired()]) + butterbox_logo = FileField((_l('Butterbox Logo')), validators=[FileAllowed(['jpg', 'png', 'svg'], 'Images only!')]) + # Services settings + disable_file_viewer = BooleanField(_l('Disable File Viewer')) + disable_map_viewer = BooleanField(_l('Disable Map Viewer')) + disable_chat = BooleanField(_l('Disable Chat')) + disable_app_store = BooleanField(_l('Disable App Store')) + # Access Settings + admin_password = PasswordField(_l('Admin Password')) + ssh_password = PasswordField(_l('SSH Password')) + + submit = SubmitField(_l('Submit')) + apply_changes = SubmitField(_l('Apply Changes')) diff --git a/app/models.py b/app/models.py new file mode 100644 index 0000000..c1a4d28 --- /dev/null +++ b/app/models.py @@ -0,0 +1,35 @@ +from typing import Optional +import sqlalchemy as sa +import sqlalchemy.orm as so +from app import db +from werkzeug.security import generate_password_hash, check_password_hash +from flask_login import UserMixin +from app import login + +class User(UserMixin, db.Model): + id: so.Mapped[int] = so.mapped_column(primary_key=True) + username: so.Mapped[str] = so.mapped_column(sa.String(64), index=True, + unique=True) + password_hash: so.Mapped[Optional[str]] = so.mapped_column(sa.String(256)) + + def set_password(self, password: str) -> None: + self.password_hash = generate_password_hash(password) + + def check_password(self, password: str) -> bool: + return check_password_hash(self.password_hash, password) + + def __repr__(self): + return ''.format(self.username) + +class Setting(db.Model): + id: so.Mapped[int] = so.mapped_column(primary_key=True) + key: so.Mapped[str] = so.mapped_column(sa.String(255), index=True, + unique=True, nullable=False) + value: so.Mapped[str] = so.mapped_column(sa.String(255)) + + def __repr__(self): + return ''.format(self.key) + +@login.user_loader +def load_user(user_id: str) -> User: + return db.session.get(User, int(user_id)) \ No newline at end of file diff --git a/app/routes.py b/app/routes.py new file mode 100644 index 0000000..f715551 --- /dev/null +++ b/app/routes.py @@ -0,0 +1,140 @@ +from email.mime import image + +from alembic.util import obfuscate_url_pw + +from app import app +from flask import render_template, flash, redirect, url_for, request, session +from app.forms import LoginForm, SettingsForm +from flask_login import login_user, current_user, logout_user, login_required +import sqlalchemy as sa +from app import db +from app.models import User, Setting +from werkzeug.datastructures import FileStorage +import json +from flask_babel import _ +import base64 + +def get_setting(name) -> str: + setting = db.session.scalar(sa.select(Setting).where(Setting.key == name)) + return str(setting.value) + +def set_setting(name: str, value: str): + setting = db.session.scalar(sa.select(Setting).where(Setting.key == name)) + print(f"I have changed {setting.key}") + setting.value = value + db.session.add(setting) + +def dump_settings(filename: str) -> None: + settings = db.session.execute(sa.select(Setting)).scalars().all() + settings_dict = {s.key: s.value for s in settings} + print(settings_dict) + with open(filename, "w") as f: + json.dump(settings_dict, f, indent=4) +@app.route('/') +@app.route('/index') +def index(): + disable_chat = get_setting("disable_chat") + disable_app_store = get_setting("disable_app_store") + disable_map_viewer = get_setting("disable_map_viewer") + disable_file_viewer = get_setting("disable_file_viewer") + service_array = [] + usb_inserted = False # actual test of whether USB is inserted + usb_has_maps = False # actual test of whether USB has maps folder + usb_has_appstore = False # actual test of whether USB has an appstore + if disable_chat == 'false': + service_array.append({"name": "Message Board", "image": url_for("static", filename="images/chat-icon.png"), "url": app.config["CONVENE_INSTALL_PATH"] }) + if disable_app_store == 'false' and usb_has_appstore: + service_array.append({"name": "Apps", "image": url_for("static", filename="images/appstore-icon.svg")}) + if disable_map_viewer == 'false' and usb_has_maps: + service_array.append({"name": "Offline Maps", "image": url_for("static", filename="images/maps-icon.png")}) + if disable_file_viewer == 'false': + name = "Files" + if not usb_inserted: + name = "Insert USB to browse files" + service_array.append({ + "name": name, + "image": url_for("static", filename="images/explore-icon.svg"), + "url": url_for("usb")}) + return render_template('index.html', title='Home', get_setting=get_setting, services=service_array) + +@app.route('/usb') +def usb(): + return render_template('usb-file-viewer.html', title='File Viewer') + +@app.route('/login', methods=['GET', 'POST']) +def login(): + if current_user.is_authenticated: + return redirect(url_for('admin')) + form = LoginForm() + form.remember_me.data = False + if form.validate_on_submit(): + user = db.session.scalar(sa.select(User).where(User.username == form.username.data)) + if user is None or not user.check_password(form.password.data): + flash(_('Invalid username or password')) + return redirect(url_for('login')) + login_user(user) + return redirect(url_for('admin')) + return render_template('login.html', title='Sign in', form=form, get_setting=get_setting) + + +@app.route('/logout') +def logout(): + logout_user() + return redirect(url_for('index')) +@app.route('/admin', methods=['GET', 'POST']) +@login_required +def admin(): + form = SettingsForm() + populate_settings = ['butterbox_name', 'wifi_password', 'disable_access_point', 'ssid', 'disable_file_viewer', 'disable_map_viewer', 'disable_app_store', 'disable_chat' ] + + bool_settings = ['disable_access_point','disable_file_viewer', 'disable_map_viewer', 'disable_app_store', 'disable_chat'] + + if not form.is_submitted(): + for s in populate_settings: + if s in bool_settings: + getattr(form, s).data = (get_setting(s) == "true") + else: + getattr(form, s).data = get_setting(s) + + if form.validate_on_submit(): + if form.submit.data: + for s in populate_settings: + new_value = getattr(form, s).data + if s in bool_settings: + new_value = str(new_value).lower() # all settings are str fow now + existing_value = get_setting(s) + if new_value != existing_value: + print(f"New value was changed for {s}. Existing value was {existing_value}, new value was {new_value}") + set_setting(s, new_value) + if s in ['butterbox_name', 'wifi_password', 'ssid', 'disable_access_point']: + app.config['SETTINGS_CHANGED'] = True + + new_logo = form.butterbox_logo.data + if new_logo.filename: + logo_stream = form.butterbox_logo.data.stream + b64_logo = base64.b64encode(logo_stream.read()).decode('utf-8') + file_mimetype = form.butterbox_logo.data.mimetype + new_value = f"data:{file_mimetype};base64,{b64_logo}" + existing_value = get_setting('butterbox_logo') + if new_value != existing_value: + print( f"New value was changed for logo") + set_setting('butterbox_logo', new_value) + new_admin_password = form.admin_password.data + if new_admin_password: + existing_admin_password = get_setting('admin_password') + if new_admin_password != existing_admin_password: + print( f"New value was changed for admin password") + set_setting('admin_password', new_admin_password) + print(get_setting('admin_password')) + + if app.config['SETTINGS_CHANGED']: + flash(_("⚠️ Some settings won't take effect until the Butter Box restarts. Click 'Apply Changes' to restart.")) + db.session.commit() + + if form.apply_changes.data: + set_setting('apply_changes', "true") + dump_settings("settings.txt") + flash(_("⚠️ Changes applied! Please wait for the box to restart.")) + + + return render_template('admin.html', get_setting=get_setting, form=form) diff --git a/app/static/butter_styles.css b/app/static/butter_styles.css new file mode 100644 index 0000000..c45feec --- /dev/null +++ b/app/static/butter_styles.css @@ -0,0 +1,21 @@ +.butter-title { + text-align: center; +} + +.butter-service { + border-radius: 20px; +} + +.butter-service__image { + margin: 0 auto; +} + +.butter-service__content { + display: block; +} + +@media (max-width: 960px) { + html { + padding: 10px; + } +} \ No newline at end of file diff --git a/app/static/images/appstore-icon.svg b/app/static/images/appstore-icon.svg new file mode 100644 index 0000000..df8a694 --- /dev/null +++ b/app/static/images/appstore-icon.svg @@ -0,0 +1,15 @@ + + + + + + + \ No newline at end of file diff --git a/app/static/images/chat-icon.png b/app/static/images/chat-icon.png new file mode 100644 index 0000000..b14bdd4 Binary files /dev/null and b/app/static/images/chat-icon.png differ diff --git a/app/static/images/explore-icon.svg b/app/static/images/explore-icon.svg new file mode 100644 index 0000000..3354fbe --- /dev/null +++ b/app/static/images/explore-icon.svg @@ -0,0 +1,5 @@ + + + + + \ No newline at end of file diff --git a/app/static/images/maps-icon.png b/app/static/images/maps-icon.png new file mode 100644 index 0000000..b294641 Binary files /dev/null and b/app/static/images/maps-icon.png differ diff --git a/app/templates/admin.html b/app/templates/admin.html new file mode 100644 index 0000000..2ed362e --- /dev/null +++ b/app/templates/admin.html @@ -0,0 +1,64 @@ +{% extends "base.html" %} + +{% block content %} +

{{ _('Application Settings') }}

+{% import "bulma_wtf.html" as wtf %} +
+ {{ form.hidden_tag() }} + {% if config['SETTINGS_CHANGED'] %} +

{{ form.apply_changes(class="button is-warning") }}

+ {% endif %} +
+ {{ wtf.form_input_field(form.ssid) }} +

This is the name of the advertised Wi-Fi network. Current SSID: {{ get_setting('ssid') }}

+
+
+ {{ wtf.form_input_field(form.wifi_password) }} +

This is the secret key needed to connect to the Wi-Fi network. By default, this is not set and everyone can join. + Current password: {{ get_setting('wifi_password') or 'Not set' }}

+
+
+ {{ wtf.form_input_field(form.butterbox_name) }} +

This is the name shown in the UI, and used to access the box locally by adding .local or .lan in your browser. + Current name: {{ get_setting('butterbox_name') }}, accessed at {{ get_setting('butterbox_name') }}.local.

+
+
+ {{ wtf.form_bool_field(form.disable_access_point) }} +

Whether this box will advertise a WiFi network.

+
+
+ {{ wtf.form_bool_field(form.disable_map_viewer) }} +

Whether map services are enabled.

+
+
+ {{ wtf.form_bool_field(form.disable_chat) }} +

Whether chat services are enabled.

+
+
+ {{ wtf.form_bool_field(form.disable_file_viewer) }} +

Whether files services via USB are enabled.

+
+
+ {{ wtf.form_bool_field(form.disable_app_store) }} +

Whether app store services are enabled.

+
+
+ {{ wtf.form_input_field(form.admin_password) }} +

Password for accessing this interface.

+
+
+ +
{{ form.butterbox_logo(class='label', style="width: 280px") }}
+ {% for error in form.butterbox_logo.errors %} +

{{ error }}

+ {% endfor %} +

This is the logo shown in the UI. Current logo:
+

+
+

{{ form.submit( class="button is-link") }}

+
+ + Logout + + +{% endblock %} \ No newline at end of file diff --git a/app/templates/base.html b/app/templates/base.html new file mode 100644 index 0000000..c869fc7 --- /dev/null +++ b/app/templates/base.html @@ -0,0 +1,36 @@ + + + + + + {% if title %} + {{ title }} + {% else %} + "{{ get_setting('butterbox_name') }}" + {% endif %} + + + + + +
+
+ +

{{ get_setting('butterbox_name') }}

+
+ +
{% block content %}{% endblock %}
+ {% with messages = get_flashed_messages() %} + {% if messages %} +
+
    + {% for message in messages %} +
  • {{ message }}
  • + {% endfor %} +
+
+ {% endif %} + {% endwith %} +
+ + diff --git a/app/templates/bulma_wtf.html b/app/templates/bulma_wtf.html new file mode 100644 index 0000000..42aab63 --- /dev/null +++ b/app/templates/bulma_wtf.html @@ -0,0 +1,20 @@ +{% macro form_input_field(field) %} +
+ {{ field.label(class='label')}} + {{ field(class='input' + (' is-danger' if field.errors else ' is_success'), type="text") }} + {% for error in field.errors %} +

{{ error }}

+ {% endfor %} +
+{% endmacro %} + + +{% macro form_bool_field(field) %} +
+ {{ field.label(class='label')}} + {{ field(class='checkbox', type="checkbox") }} + {% for error in field.errors %} +

{{ error }}

+ {% endfor %} +
+{% endmacro %} \ No newline at end of file diff --git a/app/templates/index.html b/app/templates/index.html new file mode 100644 index 0000000..c258b68 --- /dev/null +++ b/app/templates/index.html @@ -0,0 +1,17 @@ +{% extends "base.html" %} + +{% block content %} +

Hi, welcome to the {{get_setting('butterbox_name')}}.

+

View and download the information you want from this offline box.

+
+ {% for service in services %} + +
{{ service.name }}
+ +
+
+ {% endfor %} + + +
+{% endblock %} \ No newline at end of file diff --git a/app/templates/login.html b/app/templates/login.html new file mode 100644 index 0000000..46b7146 --- /dev/null +++ b/app/templates/login.html @@ -0,0 +1,14 @@ +{% extends "base.html" %} + +{% block content %} +{% import "bulma_wtf.html" as wtf %} + +

Sign In

+
+ {{ form.hidden_tag() }} +
{{ wtf.form_input_field(form.username) }}
+
{{ wtf.form_input_field(form.password) }}
+ +
{{ form.submit(class="button is-link") }}
+
+{% endblock %} diff --git a/app/translation_refs.py b/app/translation_refs.py new file mode 100644 index 0000000..398006e --- /dev/null +++ b/app/translation_refs.py @@ -0,0 +1,6 @@ +from flask_babel import gettext as _ + +_('ssid_description') +_('butterbox_name_description') +_('wifi_password_description') +_('logo_description') \ No newline at end of file diff --git a/app/translations/en/LC_MESSAGES/messages.po b/app/translations/en/LC_MESSAGES/messages.po new file mode 100644 index 0000000..d1fb43c --- /dev/null +++ b/app/translations/en/LC_MESSAGES/messages.po @@ -0,0 +1,88 @@ +# English translations for PROJECT. +# Copyright (C) 2026 ORGANIZATION +# This file is distributed under the same license as the PROJECT project. +# FIRST AUTHOR , 2026. +# +msgid "" +msgstr "" +"Project-Id-Version: PROJECT VERSION\n" +"Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" +"POT-Creation-Date: 2026-02-10 15:23+0000\n" +"PO-Revision-Date: 2026-02-10 15:29+0000\n" +"Last-Translator: FULL NAME \n" +"Language: en\n" +"Language-Team: en \n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=utf-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Generated-By: Babel 2.18.0\n" + +#: app/__init__.py:19 +msgid "Please log in to access this page." +msgstr "Please log in to access this page." + +#: app/forms.py:6 +msgid "Username" +msgstr "Username" + +#: app/forms.py:7 +msgid "Password" +msgstr "Password" + +#: app/forms.py:8 +msgid "Sign In" +msgstr "Sign In" + +#: app/forms.py:14 +msgid "WiFi Password" +msgstr "WiFi Password" + +#: app/forms.py:15 +msgid "Butterbox Name" +msgstr "Butterbox Name" + +#: app/forms.py:16 +msgid "Butterbox Logo" +msgstr "Butterbox Logo" + +#: app/forms.py:18 +msgid "Submit" +msgstr "Submit" + +#: app/routes.py:39 +msgid "Invalid username or password" +msgstr "Invalid username or password" + +#: app/translation_refs.py:3 +msgid "ssid_description" +msgstr "This is the name of the advertised Wi-Fi network." + +#: app/translation_refs.py:4 +msgid "butterbox_name_description" +msgstr "This is the secret key needed to connect to the Wi-Fi network. By default, this is not set and everyone can join." + +#: app/translation_refs.py:5 +msgid "wifi_password_description" +msgstr "This is the name shown in the UI, and used to access the box locally by adding .local or .lan in your browser." + +#: app/translation_refs.py:6 +msgid "logo_description" +msgstr "An image that will be used as the logo." + +#: app/templates/admin.html:4 +msgid "Application Settings" +msgstr "Application Settings" + +#: app/templates/base.html:15 +msgid "Home" +msgstr "Home" + +#: app/templates/base.html:32 +msgid "" +"Admin settings have changed! Restart the butterbox for changes to take " +"effect." +msgstr "" +"Admin settings have changed! Restart the butterbox for changes to take " +"effect." + diff --git a/app/translations/ro/LC_MESSAGES/messages.po b/app/translations/ro/LC_MESSAGES/messages.po new file mode 100644 index 0000000..a4e0abd --- /dev/null +++ b/app/translations/ro/LC_MESSAGES/messages.po @@ -0,0 +1,87 @@ +# Romanian translations for PROJECT. +# Copyright (C) 2026 ORGANIZATION +# This file is distributed under the same license as the PROJECT project. +# FIRST AUTHOR , 2026. +# +msgid "" +msgstr "" +"Project-Id-Version: PROJECT VERSION\n" +"Report-Msgid-Bugs-To: EMAIL@ADDRESS\n" +"POT-Creation-Date: 2026-02-10 15:23+0000\n" +"PO-Revision-Date: 2026-02-10 16:28+0000\n" +"Last-Translator: FULL NAME \n" +"Language: ro\n" +"Language-Team: ro \n" +"Plural-Forms: nplurals=3; plural=(n==1 ? 0 : (n==0 || (n%100 > 0 && n%100" +" < 20)) ? 1 : 2);\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=utf-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Generated-By: Babel 2.18.0\n" + +#: app/__init__.py:19 +msgid "Please log in to access this page." +msgstr "" + +#: app/forms.py:6 +msgid "Username" +msgstr "" + +#: app/forms.py:7 +msgid "Password" +msgstr "Password" + +#: app/forms.py:8 +msgid "Sign In" +msgstr "Sign In" + +#: app/forms.py:14 +msgid "WiFi Password" +msgstr "WiFi Password" + +#: app/forms.py:15 +msgid "Butterbox Name" +msgstr "Butterbox Name" + +#: app/forms.py:16 +msgid "Butterbox Logo" +msgstr "Butterbox Logo" + +#: app/forms.py:18 +msgid "Submit" +msgstr "Submit" + +#: app/routes.py:39 +msgid "Invalid username or password" +msgstr "" + +#: app/translation_refs.py:3 +msgid "ssid_description" +msgstr "" + +#: app/translation_refs.py:4 +msgid "butterbox_name_description" +msgstr "" + +#: app/translation_refs.py:5 +msgid "wifi_password_description" +msgstr "" + +#: app/translation_refs.py:6 +msgid "logo_description" +msgstr "" + +#: app/templates/admin.html:4 +msgid "Application Settings" +msgstr "" + +#: app/templates/base.html:15 +msgid "Home" +msgstr "" + +#: app/templates/base.html:32 +msgid "" +"Admin settings have changed! Restart the butterbox for changes to take " +"effect." +msgstr "" + diff --git a/babel.cfg b/babel.cfg new file mode 100644 index 0000000..b42c3d5 --- /dev/null +++ b/babel.cfg @@ -0,0 +1,2 @@ +[python: app/**.py] +[jinja2: app/templates/**.html] diff --git a/butter-portal.py b/butter-portal.py new file mode 100644 index 0000000..d099b92 --- /dev/null +++ b/butter-portal.py @@ -0,0 +1 @@ +from app import app diff --git a/config.py b/config.py new file mode 100644 index 0000000..1b77d7e --- /dev/null +++ b/config.py @@ -0,0 +1,22 @@ +import os +basedir = os.path.abspath(os.path.dirname(__file__)) + +class Config: + SECRET_KEY = os.environ.get('SECRET_KEY') or 'msdskdoopancjsduinabemseruulmf781' + SQLALCHEMY_DATABASE_URI = os.environ.get('DATABASE_URL') or \ + 'sqlite:///' + os.path.join(basedir, 'app.db') + SETTINGS_CHANGED = False + BUTTERBOX_SSID= "butterbox" + BUTTERBOX_WIFI_PASSWORD= "" + LANGUAGES = ['en', 'ro'] + + BUTTERBOX_NAME = "Butter Box" + BUTTERBOX_LOGO = "data:image/svg+xml;base64,<svg width="279" height="236" viewBox="0 0 279 236" fill="none" xmlns="http://www.w3.org/2000/svg">
<g opacity="0.9">
<path opacity="0.9" d="M139.388 197.636C172.284 197.636 198.952 170.968 198.952 138.072C198.952 105.175 172.284 78.507 139.388 78.507C106.491 78.507 79.823 105.175 79.823 138.072C79.823 170.968 106.491 197.636 139.388 197.636Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M139.612 4.03259V60.3754C134.817 57.4534 131.146 45.9151 131.146 32.204C131.221 18.4929 134.817 6.95463 139.612 4.03259Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M60.4178 139.42H4C6.92204 144.215 18.4603 147.887 32.1714 147.887C45.9574 147.887 57.4957 144.215 60.4178 139.42Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M275 139.645H218.657C221.579 134.85 233.118 131.179 246.829 131.179C260.54 131.254 272.078 134.85 275 139.645Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M235.365 43.8173L195.506 83.6769C194.157 78.2074 199.776 67.4933 209.442 57.8281C219.257 48.088 229.896 42.4687 235.365 43.8173Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M83.6442 83.5269L43.7847 43.6674C42.4361 49.1368 48.0554 59.8509 57.7205 69.5161C67.4607 79.2562 78.1748 84.8755 83.6442 83.5269Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M182.918 34.9012L167.559 72.1385C165.161 68.9167 165.911 60.3005 169.657 51.2347C173.403 42.1689 178.947 35.5006 182.918 34.9012Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M73.0051 114.021L35.8428 98.6616C36.4422 102.633 43.1104 108.177 52.1762 111.923C61.242 115.669 69.7833 116.419 73.0051 114.021Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M247.653 184.524L210.416 169.24C213.637 166.842 222.254 167.592 231.319 171.338C240.385 175.084 247.053 180.553 247.653 184.524Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M67.2359 170.888L30.0736 186.323C33.2953 188.72 41.8367 187.971 50.9025 184.225C60.0432 180.404 66.6365 174.859 67.2359 170.888Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M245.105 95.2899L207.943 110.799C208.542 106.828 215.136 101.284 224.202 97.5376C233.267 93.7165 241.809 92.8924 245.105 95.2899Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
<path opacity="0.9" d="M110.092 73.0377L94.658 35.8754C92.2604 39.0971 93.0846 47.7134 96.8308 56.7792C100.577 65.845 106.121 72.4383 110.092 73.0377Z" fill="#FEF5B8" stroke="#FEF5B8" stroke-width="4" stroke-miterlimit="10"/>
</g>
<path d="M203.13 178.936C198.119 178.22 192.393 177.791 190.246 175.572C188.457 173.711 192.894 167.627 188.743 161.83C187.884 160.685 186.524 159.539 184.95 158.609V139.713C184.95 136.135 183.16 132.842 180.154 130.838L144.439 108.078C140.932 105.787 136.423 105.787 132.915 108.078L97.1285 130.91C94.1224 132.842 92.333 136.206 92.333 139.713V158.752C91.2594 159.396 90.329 159.969 89.7564 160.398C83.9589 164.335 88.0386 171.492 85.8198 175.286C83.0284 179.866 80.4517 178.864 73.008 180.94C66.4948 182.658 63.4171 194.181 86.5355 188.455C89.1122 187.811 90.5436 187.525 93.4782 189.457C100.564 193.966 93.3351 208.21 99.9914 210.428C109.869 213.721 102.425 189.457 108.151 191.032C110.871 192.392 113.304 194.396 114.306 197.617C117.169 205.204 112.732 233.261 120.032 234.477C128.478 235.766 124.112 203.7 128.048 202.269C130.41 201.41 132.701 205.561 138.498 206.206C143.151 206.707 147.946 203.7 149.664 202.269C154.531 198.118 157.895 196.543 161.831 196.114C172.353 194.897 163.191 220.95 172.997 219.805C181.657 216.942 171.208 195.04 186.811 189.243C192.894 187.095 205.062 190.03 209.285 188.312C213.365 186.523 211.647 179.795 203.13 178.936Z" fill="#FCE363"/>
<path d="M120.533 235.909C120.318 235.909 120.032 235.909 119.817 235.837C114.163 234.907 114.092 224.099 114.02 212.647C114.02 206.778 113.948 200.694 112.946 198.046V197.975C112.231 195.684 110.513 193.823 107.721 192.32C107.149 193.108 107.077 196.829 107.077 199.12C107.077 204.13 107.006 209.355 104.214 211.359C102.998 212.218 101.423 212.361 99.562 211.788C95.2676 210.357 95.3392 205.561 95.4823 200.981C95.5539 196.829 95.6254 192.535 92.7625 190.674C90.2574 189.028 89.2554 189.243 86.965 189.887C70.7177 193.895 67.64 189.386 67.139 187.239C66.3517 184.161 68.9999 180.582 72.7933 179.58C74.869 179.008 76.5868 178.721 77.9467 178.435C81.6685 177.719 83.0284 177.433 84.7462 174.641C85.3904 173.568 85.3188 171.779 85.2472 169.918C85.104 166.482 84.8893 162.188 89.1122 159.325C89.5416 158.967 90.1858 158.537 91.0447 158.036V139.785C91.0447 135.777 93.1203 131.912 96.4843 129.765L132.271 106.932C136.208 104.356 141.361 104.356 145.369 106.932L181.085 129.693C184.52 131.983 186.524 135.705 186.524 139.785V157.893C187.956 158.824 189.173 159.969 190.032 161.042C192.966 165.122 192.179 169.273 191.678 172.065C191.535 172.995 191.248 174.355 191.463 174.641C192.966 176.216 197.404 176.789 201.34 177.29C202.056 177.361 202.772 177.504 203.416 177.576C209.786 178.292 212.506 181.727 212.935 184.447C213.293 186.666 212.148 188.742 210.001 189.672C207.71 190.602 203.917 190.388 199.622 190.101C195.328 189.887 190.389 189.6 187.455 190.602C179.439 193.609 179.081 201.195 178.794 208.568C178.508 214.293 178.293 219.662 173.641 221.236L173.355 221.308C171.78 221.522 170.42 221.165 169.347 220.234C166.77 218.087 166.627 213.363 166.412 208.424C166.269 204.345 166.054 199.692 164.336 198.189C163.979 197.903 163.406 197.545 162.189 197.688C158.396 198.118 155.247 199.692 150.809 203.486C150.523 203.701 144.797 208.424 138.57 207.78C135.134 207.422 132.844 205.919 131.126 204.846C130.124 204.202 129.265 203.701 128.836 203.772C128.048 204.846 127.762 210.715 127.619 215.009C127.261 223.527 126.903 232.33 123.396 235.193C122.394 235.551 121.464 235.909 120.533 235.909ZM115.666 197.116C116.811 200.265 116.883 206.277 116.883 212.647C116.883 220.52 116.954 232.473 120.247 233.046C120.676 233.117 121.034 233.046 121.321 232.76C123.826 230.756 124.183 221.451 124.47 214.651C124.828 205.991 125.114 201.768 127.476 200.909C129.265 200.265 130.768 201.267 132.414 202.269C134.061 203.271 135.85 204.488 138.57 204.774C142.793 205.204 147.302 202.341 148.662 201.195C153.529 197.044 157.179 195.183 161.617 194.682C163.334 194.467 164.838 194.897 165.983 195.827C168.703 198.118 168.917 202.985 169.06 208.138C169.204 212.075 169.418 216.584 170.993 217.944C171.279 218.158 171.709 218.445 172.639 218.373C175.144 217.443 175.43 213.936 175.645 208.353C176.003 200.838 176.361 191.533 186.238 187.883C189.745 186.666 194.755 186.952 199.551 187.239C203.201 187.453 206.995 187.668 208.641 187.024C209.571 186.666 210.001 185.879 209.858 184.877C209.571 183.087 207.281 180.868 202.843 180.439H202.772C202.056 180.367 201.412 180.224 200.696 180.153C195.829 179.509 191.32 178.936 189.101 176.646C187.884 175.357 188.242 173.496 188.6 171.564C189.101 168.987 189.674 165.766 187.455 162.76C186.739 161.83 185.522 160.756 184.091 159.969L183.375 159.539V139.857C183.375 136.779 181.801 133.916 179.224 132.198L143.652 109.223C140.645 107.219 136.709 107.219 133.703 109.223L97.9158 132.055C95.3391 133.701 93.7645 136.636 93.7645 139.642V159.468L93.0488 159.897C92.1183 160.47 91.1163 161.042 90.6152 161.472C87.6807 163.476 87.8239 166.482 87.967 169.631C88.0386 171.85 88.1818 174.14 87.1081 176.001C84.7462 179.938 82.2411 180.439 78.4477 181.155C77.0162 181.441 75.4416 181.727 73.5091 182.3C71.0756 182.944 69.5725 185.091 69.9304 186.451C70.3598 188.097 74.2964 190.03 86.3208 187.095C89.1122 186.38 91.0447 186.093 94.4087 188.312C98.7031 191.032 98.56 196.4 98.4884 201.124C98.4168 205.633 98.4884 208.496 100.636 209.212C102.067 209.713 102.568 209.355 102.711 209.212C104.357 208.067 104.357 202.985 104.357 199.335C104.357 194.968 104.429 191.605 106.218 190.245C106.719 189.887 107.507 189.529 108.723 189.815L109.01 189.887C112.445 191.533 114.736 193.966 115.666 197.116Z" fill="#010004"/>
<path d="M150.022 166.124C152.813 170.347 146.085 194.11 138.284 194.11C128.693 194.11 123.897 169.846 126.474 166.124C127.905 164.049 131.413 168.343 138.641 168.343C145.369 168.343 148.59 163.905 150.022 166.124Z" fill="#010004"/>
<path d="M175.216 159.683C176.934 160.112 178.293 158.323 177.077 157.178C175.287 155.603 172.138 153.169 166.555 153.169C161.116 153.169 157.752 155.603 155.676 157.249C154.245 158.394 155.676 160.327 157.537 159.826C160.042 159.182 163.263 157.607 166.698 157.607C170.134 157.75 173.069 159.182 175.216 159.683Z" fill="#010004"/>
<path d="M119.102 159.683C120.819 160.112 122.179 158.323 120.963 157.178C119.173 155.603 116.024 153.169 110.441 153.169C105.002 153.169 101.638 155.603 99.562 157.249C98.1305 158.394 99.562 160.327 101.423 159.826C103.928 159.182 107.149 157.607 110.584 157.607C114.092 157.75 116.955 159.182 119.102 159.683Z" fill="#010004"/>
<path d="M109.869 182.586C116.154 182.586 121.249 178.517 121.249 173.496C121.249 168.476 116.154 164.406 109.869 164.406C103.583 164.406 98.4883 168.476 98.4883 173.496C98.4883 178.517 103.583 182.586 109.869 182.586Z" fill="#FEF5B8"/>
<path d="M166.698 182.586C172.983 182.586 178.079 178.517 178.079 173.496C178.079 168.476 172.983 164.406 166.698 164.406C160.413 164.406 155.318 168.476 155.318 173.496C155.318 178.517 160.413 182.586 166.698 182.586Z" fill="#FEF5B8"/>
<path d="M133.202 152.096L107.435 136.206C105.932 135.276 105.932 133.057 107.435 132.127L133.202 116.237C136.351 114.305 140.288 114.305 143.437 116.237L169.203 132.127C170.706 133.057 170.706 135.276 169.203 136.206L143.437 152.096C140.288 154.028 136.279 154.028 133.202 152.096Z" fill="#FEF5B8"/>
<path d="M135.635 190.388C132.271 187.811 133.202 178.578 139.142 177.504C144.153 176.574 146.515 178.22 143.365 184.948C140.216 191.891 137.21 191.533 135.635 190.388Z" fill="#FEF5B8"/>
</svg>
" + ADMIN_PASSWORD = "admin" + + DISABLE_ACCESS_POINT = "false" + DISABLE_CHAT = "false" + DISABLE_APP_STORE = "false" + DISABLE_MAP_VIEWER = "false" + DISABLE_FILE_VIEWER = "false" + CONVENE_INSTALL_PATH = "/chat" \ No newline at end of file