diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000000000000000000000000000000000000..e16235b2eb61750c6f9902f9e24ab92f6ed2ad9f --- /dev/null +++ b/Dockerfile @@ -0,0 +1,63 @@ +FROM swaggerapi/swagger-ui:v4.18.2 AS swagger-ui +FROM nvidia/cuda:11.8.0-cudnn8-runtime-ubuntu22.04 + +ARG SERVICE_USER=service +ARG SERVICE_UID=1001 +ARG SERVICE_GID=1001 + +ENV PYTHON_VERSION=3.10 +ENV POETRY_VENV=/app/.venv + +RUN export DEBIAN_FRONTEND=noninteractive \ + && apt-get -qq update \ + && apt-get -qq install --no-install-recommends \ + python${PYTHON_VERSION} \ + python${PYTHON_VERSION}-venv \ + python3-pip \ + lua5.3 \ + lua5.4 \ + lua-check \ + fswatch \ + make \ + ffmpeg \ + redis \ + && rm -rf /var/lib/apt/lists/* + +RUN ln -s -f /usr/bin/python${PYTHON_VERSION} /usr/bin/python3 && \ + ln -s -f /usr/bin/python${PYTHON_VERSION} /usr/bin/python && \ + ln -s -f /usr/bin/pip3 /usr/bin/pip + +RUN python3 -m venv $POETRY_VENV \ + && $POETRY_VENV/bin/pip install -U pip setuptools \ + && $POETRY_VENV/bin/pip install poetry==1.6.1 + +ENV PATH="${PATH}:${POETRY_VENV}/bin" + +WORKDIR /app + +COPY poetry.lock pyproject.toml ./ + +RUN poetry config virtualenvs.in-project true +RUN poetry install --no-root + +COPY . . +COPY --from=swagger-ui /usr/share/nginx/html/swagger-ui.css swagger-ui-assets/swagger-ui.css +COPY --from=swagger-ui /usr/share/nginx/html/swagger-ui-bundle.js swagger-ui-assets/swagger-ui-bundle.js + +RUN poetry install && rm -rf /root/.cache/pypoetry +RUN $POETRY_VENV/bin/pip install --no-cache-dir torch==1.13.1+cu117 -f https://download.pytorch.org/whl/torch + +WORKDIR /app/reascripts/ReaSpeech +RUN make publish +WORKDIR /app +RUN rm -rf reascripts + +RUN groupadd -g $SERVICE_GID $SERVICE_USER || true \ + && useradd -u $SERVICE_UID -g $SERVICE_GID -d /app -s /usr/sbin/nologin $SERVICE_USER || true \ + && chown -R $SERVICE_UID:$SERVICE_GID /app + +USER $SERVICE_USER + +ENTRYPOINT ["python3", "app/run.py"] + +EXPOSE 9000 diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000000000000000000000000000000000000..c7f040222e7de35e43c14a1c7aa0018d3611691a --- /dev/null +++ b/LICENSE @@ -0,0 +1,22 @@ +MIT License + +Copyright (c) 2022 Ahmet Oner & Besim Alibegovic +Portions Copyright (c) 2024 TeamAudio + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md index 627aa7dee82e02de5089d8aed1d9168243faabf1..1defa4e6fb2d81b383293f6bb3647b7aec9884d2 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,9 @@ --- -title: ReaSpeech Cloud -emoji: 🚀 -colorFrom: indigo +title: Reaspeech Eval +emoji: 🗣️📝 +colorFrom: gray colorTo: yellow sdk: docker pinned: false -license: mit +app_port: 9000 --- - -Check out the configuration reference at https://huggingface.co/docs/hub/spaces-config-reference diff --git a/app/.gitignore b/app/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..52d26d4ecb7d6d0eab778b58093d1fe838b59335 --- /dev/null +++ b/app/.gitignore @@ -0,0 +1 @@ +output/**/*/ \ No newline at end of file diff --git a/app/__pycache__/webservice.cpython-310.pyc b/app/__pycache__/webservice.cpython-310.pyc new file mode 100644 index 0000000000000000000000000000000000000000..7fdb961c16dab9c6028d0790cfab93b4751ae7cd Binary files /dev/null and b/app/__pycache__/webservice.cpython-310.pyc differ diff --git a/app/__pycache__/worker.cpython-310.pyc b/app/__pycache__/worker.cpython-310.pyc new file mode 100644 index 0000000000000000000000000000000000000000..6e4644ab5278781559686553fb581811b6cd4249 Binary files /dev/null and b/app/__pycache__/worker.cpython-310.pyc differ diff --git a/app/faster_whisper/__pycache__/core.cpython-310.pyc b/app/faster_whisper/__pycache__/core.cpython-310.pyc new file mode 100644 index 0000000000000000000000000000000000000000..7969b71e8d126f5419aaddb3eff603fe12154503 Binary files /dev/null and b/app/faster_whisper/__pycache__/core.cpython-310.pyc differ diff --git a/app/faster_whisper/__pycache__/utils.cpython-310.pyc b/app/faster_whisper/__pycache__/utils.cpython-310.pyc new file mode 100644 index 0000000000000000000000000000000000000000..60f062df8836fb6a875a78a81d74475449ec1354 Binary files /dev/null and b/app/faster_whisper/__pycache__/utils.cpython-310.pyc differ diff --git a/app/faster_whisper/core.py b/app/faster_whisper/core.py new file mode 100644 index 0000000000000000000000000000000000000000..ada1402962085773bd8c2d11ed9f5d63354db3be --- /dev/null +++ b/app/faster_whisper/core.py @@ -0,0 +1,98 @@ +import os +from io import StringIO +from threading import Lock +from typing import Union, BinaryIO + +import torch +import tqdm +import whisper +from faster_whisper import WhisperModel + +from .utils import ResultWriter, WriteTXT, WriteSRT, WriteVTT, WriteTSV, WriteJSON + +ASR_ENGINE_OPTIONS = frozenset([ + "task", + "language", + "initial_prompt", + "vad_filter", + "word_timestamps", +]) + +model_name = os.getenv("ASR_MODEL", "small") +model_path = os.getenv("ASR_MODEL_PATH", os.path.join(os.path.expanduser("~"), ".cache", "whisper")) + +model_lock = Lock() + +model = None +def load_model(next_model_name: str): + with model_lock: + global model_name, model + + if model and next_model_name == model_name: + return model + + if torch.cuda.is_available(): + model = WhisperModel(model_size_or_path=next_model_name, device="cuda", compute_type="float32", download_root=model_path) + else: + model = WhisperModel(model_size_or_path=next_model_name, device="cpu", compute_type="int8", download_root=model_path) + + model_name = next_model_name + + return model + + +def transcribe(audio, asr_options, output): + options_dict = {k: v for k, v in asr_options.items() if k in ASR_ENGINE_OPTIONS} + + with model_lock: + segments = [] + text = "" + segment_generator, info = model.transcribe(audio, beam_size=5, **options_dict) + with tqdm.tqdm(total=round(info.duration), unit='sec') as tqdm_pbar: + for segment in segment_generator: + segment_dict = segment._asdict() + if segment.words: + segment_dict["words"] = [word._asdict() for word in segment.words] + segments.append(segment_dict) + text = text + segment.text + tqdm_pbar.update(segment.end - segment.start) + result = { + "language": options_dict.get("language", info.language), + "segments": segments, + "text": text + } + + output_file = StringIO() + write_result(result, output_file, output) + output_file.seek(0) + + return output_file + + +def language_detection(audio): + # load audio and pad/trim it to fit 30 seconds + audio = whisper.pad_or_trim(audio) + + # detect the spoken language + with model_lock: + segments, info = model.transcribe(audio, beam_size=5) + detected_lang_code = info.language + + return detected_lang_code + + +def write_result( + result: dict, file: BinaryIO, output: Union[str, None] +): + if output == "srt": + WriteSRT(ResultWriter).write_result(result, file=file) + elif output == "vtt": + WriteVTT(ResultWriter).write_result(result, file=file) + elif output == "tsv": + WriteTSV(ResultWriter).write_result(result, file=file) + elif output == "json": + WriteJSON(ResultWriter).write_result(result, file=file) + elif output == "txt": + WriteTXT(ResultWriter).write_result(result, file=file) + else: + return 'Please select an output method!' diff --git a/app/faster_whisper/utils.py b/app/faster_whisper/utils.py new file mode 100644 index 0000000000000000000000000000000000000000..c85ff5a704474eac45063e6e286a4d52a1ed5e8b --- /dev/null +++ b/app/faster_whisper/utils.py @@ -0,0 +1,86 @@ +import json +import os +from typing import TextIO + +from faster_whisper.utils import format_timestamp + + +class ResultWriter: + extension: str + + def __init__(self, output_dir: str): + self.output_dir = output_dir + + def __call__(self, result: dict, audio_path: str): + audio_basename = os.path.basename(audio_path) + output_path = os.path.join(self.output_dir, audio_basename + "." + self.extension) + + with open(output_path, "w", encoding="utf-8") as f: + self.write_result(result, file=f) + + def write_result(self, result: dict, file: TextIO): + raise NotImplementedError + + +class WriteTXT(ResultWriter): + extension: str = "txt" + + def write_result(self, result: dict, file: TextIO): + for segment in result["segments"]: + print(segment['text'].strip(), file=file, flush=True) + + +class WriteVTT(ResultWriter): + extension: str = "vtt" + + def write_result(self, result: dict, file: TextIO): + print("WEBVTT\n", file=file) + for segment in result["segments"]: + print( + f"{format_timestamp(segment['start'])} --> {format_timestamp(segment['end'])}\n" + f"{segment['text'].strip().replace('-->', '->')}\n", + file=file, + flush=True, + ) + + +class WriteSRT(ResultWriter): + extension: str = "srt" + + def write_result(self, result: dict, file: TextIO): + for i, segment in enumerate(result["segments"], start=1): + # write srt lines + print( + f"{i}\n" + f"{format_timestamp(segment['start'], always_include_hours=True, decimal_marker=',')} --> " + f"{format_timestamp(segment['end'], always_include_hours=True, decimal_marker=',')}\n" + f"{segment['text'].strip().replace('-->', '->')}\n", + file=file, + flush=True, + ) + + +class WriteTSV(ResultWriter): + """ + Write a transcript to a file in TSV (tab-separated values) format containing lines like: + \t\t + + Using integer milliseconds as start and end times means there's no chance of interference from + an environment setting a language encoding that causes the decimal in a floating point number + to appear as a comma; also is faster and more efficient to parse & store, e.g., in C++. + """ + extension: str = "tsv" + + def write_result(self, result: dict, file: TextIO): + print("start", "end", "text", sep="\t", file=file) + for segment in result["segments"]: + print(round(1000 * segment['start']), file=file, end="\t") + print(round(1000 * segment['end']), file=file, end="\t") + print(segment['text'].strip().replace("\t", " "), file=file, flush=True) + + +class WriteJSON(ResultWriter): + extension: str = "json" + + def write_result(self, result: dict, file: TextIO): + json.dump(result, file) diff --git a/app/openai_whisper/core.py b/app/openai_whisper/core.py new file mode 100644 index 0000000000000000000000000000000000000000..eee49d3a6beb71549af81d51d7bd641728397bc7 --- /dev/null +++ b/app/openai_whisper/core.py @@ -0,0 +1,87 @@ +import os +from io import StringIO +from threading import Lock +from typing import BinaryIO, Union + +import torch +import whisper +from whisper.utils import ResultWriter, WriteTXT, WriteSRT, WriteVTT, WriteTSV, WriteJSON + +ASR_ENGINE_OPTIONS = frozenset([ + "task", + "language", + "initial_prompt", + "word_timestamps", +]) + +model_name = os.getenv("ASR_MODEL", "small") +model_path = os.getenv("ASR_MODEL_PATH", os.path.join(os.path.expanduser("~"), ".cache", "whisper")) + +model_lock = Lock() + +model = None +def load_model(next_model_name: str): + with model_lock: + global model_name, model + + if model and next_model_name == model_name: + return model + + if torch.cuda.is_available(): + model = whisper.load_model(next_model_name, download_root=model_path).cuda() + else: + model = whisper.load_model(next_model_name, download_root=model_path) + + model_name = next_model_name + + return model + + +def transcribe(audio, asr_options, output): + options_dict = {k: v for k, v in asr_options.items() if k in ASR_ENGINE_OPTIONS} + + with model_lock: + result = model.transcribe(audio, **options_dict) + + output_file = StringIO() + write_result(result, output_file, output) + output_file.seek(0) + + return output_file + + +def language_detection(audio): + # load audio and pad/trim it to fit 30 seconds + audio = whisper.pad_or_trim(audio) + + # make log-Mel spectrogram and move to the same device as the model + mel = whisper.log_mel_spectrogram(audio).to(model.device) + + # detect the spoken language + with model_lock: + _, probs = model.detect_language(mel) + detected_lang_code = max(probs, key=probs.get) + + return detected_lang_code + + +def write_result( + result: dict, file: BinaryIO, output: Union[str, None] +): + options = { + 'max_line_width': 1000, + 'max_line_count': 10, + 'highlight_words': False + } + if output == "srt": + WriteSRT(ResultWriter).write_result(result, file=file, options=options) + elif output == "vtt": + WriteVTT(ResultWriter).write_result(result, file=file, options=options) + elif output == "tsv": + WriteTSV(ResultWriter).write_result(result, file=file, options=options) + elif output == "json": + WriteJSON(ResultWriter).write_result(result, file=file, options=options) + elif output == "txt": + WriteTXT(ResultWriter).write_result(result, file=file, options=options) + else: + return 'Please select an output method!' diff --git a/app/output/.gitkeep b/app/output/.gitkeep new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/app/run.py b/app/run.py new file mode 100644 index 0000000000000000000000000000000000000000..83176988a36403479bc4eeb3730994943866905a --- /dev/null +++ b/app/run.py @@ -0,0 +1,71 @@ +#!/usr/bin/env python + +import os +import subprocess +import sys +import argparse + +argmap = { + '--redis-bin': { + 'default': 'redis-server', + 'help': 'Path to Redis server binary (default: %(default)s)' }, + '--celery-broker-url': { + 'default': 'redis://localhost:6379/0', + 'help': 'Celery broker URL (default: %(default)s)' }, + '--celery-result-backend-url': { + 'default': 'redis://localhost:6379/0', + 'help': 'Celery result backend URL (default: %(default)s)' }, + '--output-directory': { + 'default': 'app/output', + 'help': 'Output directory (default: %(default)s)' }, + '--output-url-prefix': { + 'default': '/output', + 'help': 'Output URL prefix (default: %(default)s)' }, + '--ffmpeg-bin': { + 'default': 'ffmpeg', + 'help': 'Path to ffmpeg binary (default: %(default)s)' }, + '--asr-engine': { + 'default': os.getenv('ASR_ENGINE', 'faster_whisper'), + 'help': 'ASR engine to use (default: %(default)s)' }, + '--asr-model': { + 'default': os.getenv('ASR_MODEL', 'small'), + 'help': 'ASR model to use (default: %(default)s)' }, +} + +parser = argparse.ArgumentParser() +for arg, kwargs in argmap.items(): + parser.add_argument(arg, **kwargs) + +args = parser.parse_args() + +os.environ['CELERY_BROKER_URL'] = args.celery_broker_url +os.environ['CELERY_RESULT_BACKEND'] = args.celery_result_backend_url +os.environ['OUTPUT_DIRECTORY'] = args.output_directory +os.environ['OUTPUT_URL_PREFIX'] = args.output_url_prefix +os.environ['FFMPEG_BIN'] = args.ffmpeg_bin +os.environ['ASR_ENGINE'] = args.asr_engine +os.environ['ASR_MODEL'] = args.asr_model + +# Start Redis +print('Starting database...', file=sys.stderr) +subprocess.Popen([args.redis_bin], stdout=subprocess.DEVNULL) + +# Start Celery +print('Starting worker...', file=sys.stderr) +subprocess.Popen(['celery', '-A', 'app.worker.celery', 'worker', '--pool=solo', '--loglevel=info']) + +# Start Gunicorn +print('Starting application...', file=sys.stderr) +subprocess.Popen(['gunicorn', '--bind', '0.0.0.0:9000', '--workers', '1', '--timeout', '0', 'app.webservice:app', '-k', 'uvicorn.workers.UvicornWorker']) + +# Wait for any process to exit +status = os.WEXITSTATUS(os.wait()[1]) +print('Process exited with status', status, file=sys.stderr) + +# Terminate any child processes +print('Terminating child processes...', file=sys.stderr) +os.system('pkill -P %d' % os.getpid()) + +# Exit with status of process that exited +print('Exiting with status', status, file=sys.stderr) +sys.exit(status) diff --git a/app/static/css/bootstrap-icons.min.css b/app/static/css/bootstrap-icons.min.css new file mode 100644 index 0000000000000000000000000000000000000000..088ba56931d308cd50354cb40acbc1d027a674aa --- /dev/null +++ b/app/static/css/bootstrap-icons.min.css @@ -0,0 +1,5 @@ +/*! + * Bootstrap Icons v1.10.5 (https://icons.getbootstrap.com/) + * Copyright 2019-2023 The Bootstrap Authors + * Licensed under MIT (https://github.com/twbs/icons/blob/main/LICENSE) + */@font-face{font-display:block;font-family:bootstrap-icons;src:url("fonts/bootstrap-icons.woff2?1fa40e8900654d2863d011707b9fb6f2") format("woff2"),url("fonts/bootstrap-icons.woff?1fa40e8900654d2863d011707b9fb6f2") format("woff")}.bi::before,[class*=" bi-"]::before,[class^=bi-]::before{display:inline-block;font-family:bootstrap-icons!important;font-style:normal;font-weight:400!important;font-variant:normal;text-transform:none;line-height:1;vertical-align:-.125em;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.bi-123::before{content:"\f67f"}.bi-alarm-fill::before{content:"\f101"}.bi-alarm::before{content:"\f102"}.bi-align-bottom::before{content:"\f103"}.bi-align-center::before{content:"\f104"}.bi-align-end::before{content:"\f105"}.bi-align-middle::before{content:"\f106"}.bi-align-start::before{content:"\f107"}.bi-align-top::before{content:"\f108"}.bi-alt::before{content:"\f109"}.bi-app-indicator::before{content:"\f10a"}.bi-app::before{content:"\f10b"}.bi-archive-fill::before{content:"\f10c"}.bi-archive::before{content:"\f10d"}.bi-arrow-90deg-down::before{content:"\f10e"}.bi-arrow-90deg-left::before{content:"\f10f"}.bi-arrow-90deg-right::before{content:"\f110"}.bi-arrow-90deg-up::before{content:"\f111"}.bi-arrow-bar-down::before{content:"\f112"}.bi-arrow-bar-left::before{content:"\f113"}.bi-arrow-bar-right::before{content:"\f114"}.bi-arrow-bar-up::before{content:"\f115"}.bi-arrow-clockwise::before{content:"\f116"}.bi-arrow-counterclockwise::before{content:"\f117"}.bi-arrow-down-circle-fill::before{content:"\f118"}.bi-arrow-down-circle::before{content:"\f119"}.bi-arrow-down-left-circle-fill::before{content:"\f11a"}.bi-arrow-down-left-circle::before{content:"\f11b"}.bi-arrow-down-left-square-fill::before{content:"\f11c"}.bi-arrow-down-left-square::before{content:"\f11d"}.bi-arrow-down-left::before{content:"\f11e"}.bi-arrow-down-right-circle-fill::before{content:"\f11f"}.bi-arrow-down-right-circle::before{content:"\f120"}.bi-arrow-down-right-square-fill::before{content:"\f121"}.bi-arrow-down-right-square::before{content:"\f122"}.bi-arrow-down-right::before{content:"\f123"}.bi-arrow-down-short::before{content:"\f124"}.bi-arrow-down-square-fill::before{content:"\f125"}.bi-arrow-down-square::before{content:"\f126"}.bi-arrow-down-up::before{content:"\f127"}.bi-arrow-down::before{content:"\f128"}.bi-arrow-left-circle-fill::before{content:"\f129"}.bi-arrow-left-circle::before{content:"\f12a"}.bi-arrow-left-right::before{content:"\f12b"}.bi-arrow-left-short::before{content:"\f12c"}.bi-arrow-left-square-fill::before{content:"\f12d"}.bi-arrow-left-square::before{content:"\f12e"}.bi-arrow-left::before{content:"\f12f"}.bi-arrow-repeat::before{content:"\f130"}.bi-arrow-return-left::before{content:"\f131"}.bi-arrow-return-right::before{content:"\f132"}.bi-arrow-right-circle-fill::before{content:"\f133"}.bi-arrow-right-circle::before{content:"\f134"}.bi-arrow-right-short::before{content:"\f135"}.bi-arrow-right-square-fill::before{content:"\f136"}.bi-arrow-right-square::before{content:"\f137"}.bi-arrow-right::before{content:"\f138"}.bi-arrow-up-circle-fill::before{content:"\f139"}.bi-arrow-up-circle::before{content:"\f13a"}.bi-arrow-up-left-circle-fill::before{content:"\f13b"}.bi-arrow-up-left-circle::before{content:"\f13c"}.bi-arrow-up-left-square-fill::before{content:"\f13d"}.bi-arrow-up-left-square::before{content:"\f13e"}.bi-arrow-up-left::before{content:"\f13f"}.bi-arrow-up-right-circle-fill::before{content:"\f140"}.bi-arrow-up-right-circle::before{content:"\f141"}.bi-arrow-up-right-square-fill::before{content:"\f142"}.bi-arrow-up-right-square::before{content:"\f143"}.bi-arrow-up-right::before{content:"\f144"}.bi-arrow-up-short::before{content:"\f145"}.bi-arrow-up-square-fill::before{content:"\f146"}.bi-arrow-up-square::before{content:"\f147"}.bi-arrow-up::before{content:"\f148"}.bi-arrows-angle-contract::before{content:"\f149"}.bi-arrows-angle-expand::before{content:"\f14a"}.bi-arrows-collapse::before{content:"\f14b"}.bi-arrows-expand::before{content:"\f14c"}.bi-arrows-fullscreen::before{content:"\f14d"}.bi-arrows-move::before{content:"\f14e"}.bi-aspect-ratio-fill::before{content:"\f14f"}.bi-aspect-ratio::before{content:"\f150"}.bi-asterisk::before{content:"\f151"}.bi-at::before{content:"\f152"}.bi-award-fill::before{content:"\f153"}.bi-award::before{content:"\f154"}.bi-back::before{content:"\f155"}.bi-backspace-fill::before{content:"\f156"}.bi-backspace-reverse-fill::before{content:"\f157"}.bi-backspace-reverse::before{content:"\f158"}.bi-backspace::before{content:"\f159"}.bi-badge-3d-fill::before{content:"\f15a"}.bi-badge-3d::before{content:"\f15b"}.bi-badge-4k-fill::before{content:"\f15c"}.bi-badge-4k::before{content:"\f15d"}.bi-badge-8k-fill::before{content:"\f15e"}.bi-badge-8k::before{content:"\f15f"}.bi-badge-ad-fill::before{content:"\f160"}.bi-badge-ad::before{content:"\f161"}.bi-badge-ar-fill::before{content:"\f162"}.bi-badge-ar::before{content:"\f163"}.bi-badge-cc-fill::before{content:"\f164"}.bi-badge-cc::before{content:"\f165"}.bi-badge-hd-fill::before{content:"\f166"}.bi-badge-hd::before{content:"\f167"}.bi-badge-tm-fill::before{content:"\f168"}.bi-badge-tm::before{content:"\f169"}.bi-badge-vo-fill::before{content:"\f16a"}.bi-badge-vo::before{content:"\f16b"}.bi-badge-vr-fill::before{content:"\f16c"}.bi-badge-vr::before{content:"\f16d"}.bi-badge-wc-fill::before{content:"\f16e"}.bi-badge-wc::before{content:"\f16f"}.bi-bag-check-fill::before{content:"\f170"}.bi-bag-check::before{content:"\f171"}.bi-bag-dash-fill::before{content:"\f172"}.bi-bag-dash::before{content:"\f173"}.bi-bag-fill::before{content:"\f174"}.bi-bag-plus-fill::before{content:"\f175"}.bi-bag-plus::before{content:"\f176"}.bi-bag-x-fill::before{content:"\f177"}.bi-bag-x::before{content:"\f178"}.bi-bag::before{content:"\f179"}.bi-bar-chart-fill::before{content:"\f17a"}.bi-bar-chart-line-fill::before{content:"\f17b"}.bi-bar-chart-line::before{content:"\f17c"}.bi-bar-chart-steps::before{content:"\f17d"}.bi-bar-chart::before{content:"\f17e"}.bi-basket-fill::before{content:"\f17f"}.bi-basket::before{content:"\f180"}.bi-basket2-fill::before{content:"\f181"}.bi-basket2::before{content:"\f182"}.bi-basket3-fill::before{content:"\f183"}.bi-basket3::before{content:"\f184"}.bi-battery-charging::before{content:"\f185"}.bi-battery-full::before{content:"\f186"}.bi-battery-half::before{content:"\f187"}.bi-battery::before{content:"\f188"}.bi-bell-fill::before{content:"\f189"}.bi-bell::before{content:"\f18a"}.bi-bezier::before{content:"\f18b"}.bi-bezier2::before{content:"\f18c"}.bi-bicycle::before{content:"\f18d"}.bi-binoculars-fill::before{content:"\f18e"}.bi-binoculars::before{content:"\f18f"}.bi-blockquote-left::before{content:"\f190"}.bi-blockquote-right::before{content:"\f191"}.bi-book-fill::before{content:"\f192"}.bi-book-half::before{content:"\f193"}.bi-book::before{content:"\f194"}.bi-bookmark-check-fill::before{content:"\f195"}.bi-bookmark-check::before{content:"\f196"}.bi-bookmark-dash-fill::before{content:"\f197"}.bi-bookmark-dash::before{content:"\f198"}.bi-bookmark-fill::before{content:"\f199"}.bi-bookmark-heart-fill::before{content:"\f19a"}.bi-bookmark-heart::before{content:"\f19b"}.bi-bookmark-plus-fill::before{content:"\f19c"}.bi-bookmark-plus::before{content:"\f19d"}.bi-bookmark-star-fill::before{content:"\f19e"}.bi-bookmark-star::before{content:"\f19f"}.bi-bookmark-x-fill::before{content:"\f1a0"}.bi-bookmark-x::before{content:"\f1a1"}.bi-bookmark::before{content:"\f1a2"}.bi-bookmarks-fill::before{content:"\f1a3"}.bi-bookmarks::before{content:"\f1a4"}.bi-bookshelf::before{content:"\f1a5"}.bi-bootstrap-fill::before{content:"\f1a6"}.bi-bootstrap-reboot::before{content:"\f1a7"}.bi-bootstrap::before{content:"\f1a8"}.bi-border-all::before{content:"\f1a9"}.bi-border-bottom::before{content:"\f1aa"}.bi-border-center::before{content:"\f1ab"}.bi-border-inner::before{content:"\f1ac"}.bi-border-left::before{content:"\f1ad"}.bi-border-middle::before{content:"\f1ae"}.bi-border-outer::before{content:"\f1af"}.bi-border-right::before{content:"\f1b0"}.bi-border-style::before{content:"\f1b1"}.bi-border-top::before{content:"\f1b2"}.bi-border-width::before{content:"\f1b3"}.bi-border::before{content:"\f1b4"}.bi-bounding-box-circles::before{content:"\f1b5"}.bi-bounding-box::before{content:"\f1b6"}.bi-box-arrow-down-left::before{content:"\f1b7"}.bi-box-arrow-down-right::before{content:"\f1b8"}.bi-box-arrow-down::before{content:"\f1b9"}.bi-box-arrow-in-down-left::before{content:"\f1ba"}.bi-box-arrow-in-down-right::before{content:"\f1bb"}.bi-box-arrow-in-down::before{content:"\f1bc"}.bi-box-arrow-in-left::before{content:"\f1bd"}.bi-box-arrow-in-right::before{content:"\f1be"}.bi-box-arrow-in-up-left::before{content:"\f1bf"}.bi-box-arrow-in-up-right::before{content:"\f1c0"}.bi-box-arrow-in-up::before{content:"\f1c1"}.bi-box-arrow-left::before{content:"\f1c2"}.bi-box-arrow-right::before{content:"\f1c3"}.bi-box-arrow-up-left::before{content:"\f1c4"}.bi-box-arrow-up-right::before{content:"\f1c5"}.bi-box-arrow-up::before{content:"\f1c6"}.bi-box-seam::before{content:"\f1c7"}.bi-box::before{content:"\f1c8"}.bi-braces::before{content:"\f1c9"}.bi-bricks::before{content:"\f1ca"}.bi-briefcase-fill::before{content:"\f1cb"}.bi-briefcase::before{content:"\f1cc"}.bi-brightness-alt-high-fill::before{content:"\f1cd"}.bi-brightness-alt-high::before{content:"\f1ce"}.bi-brightness-alt-low-fill::before{content:"\f1cf"}.bi-brightness-alt-low::before{content:"\f1d0"}.bi-brightness-high-fill::before{content:"\f1d1"}.bi-brightness-high::before{content:"\f1d2"}.bi-brightness-low-fill::before{content:"\f1d3"}.bi-brightness-low::before{content:"\f1d4"}.bi-broadcast-pin::before{content:"\f1d5"}.bi-broadcast::before{content:"\f1d6"}.bi-brush-fill::before{content:"\f1d7"}.bi-brush::before{content:"\f1d8"}.bi-bucket-fill::before{content:"\f1d9"}.bi-bucket::before{content:"\f1da"}.bi-bug-fill::before{content:"\f1db"}.bi-bug::before{content:"\f1dc"}.bi-building::before{content:"\f1dd"}.bi-bullseye::before{content:"\f1de"}.bi-calculator-fill::before{content:"\f1df"}.bi-calculator::before{content:"\f1e0"}.bi-calendar-check-fill::before{content:"\f1e1"}.bi-calendar-check::before{content:"\f1e2"}.bi-calendar-date-fill::before{content:"\f1e3"}.bi-calendar-date::before{content:"\f1e4"}.bi-calendar-day-fill::before{content:"\f1e5"}.bi-calendar-day::before{content:"\f1e6"}.bi-calendar-event-fill::before{content:"\f1e7"}.bi-calendar-event::before{content:"\f1e8"}.bi-calendar-fill::before{content:"\f1e9"}.bi-calendar-minus-fill::before{content:"\f1ea"}.bi-calendar-minus::before{content:"\f1eb"}.bi-calendar-month-fill::before{content:"\f1ec"}.bi-calendar-month::before{content:"\f1ed"}.bi-calendar-plus-fill::before{content:"\f1ee"}.bi-calendar-plus::before{content:"\f1ef"}.bi-calendar-range-fill::before{content:"\f1f0"}.bi-calendar-range::before{content:"\f1f1"}.bi-calendar-week-fill::before{content:"\f1f2"}.bi-calendar-week::before{content:"\f1f3"}.bi-calendar-x-fill::before{content:"\f1f4"}.bi-calendar-x::before{content:"\f1f5"}.bi-calendar::before{content:"\f1f6"}.bi-calendar2-check-fill::before{content:"\f1f7"}.bi-calendar2-check::before{content:"\f1f8"}.bi-calendar2-date-fill::before{content:"\f1f9"}.bi-calendar2-date::before{content:"\f1fa"}.bi-calendar2-day-fill::before{content:"\f1fb"}.bi-calendar2-day::before{content:"\f1fc"}.bi-calendar2-event-fill::before{content:"\f1fd"}.bi-calendar2-event::before{content:"\f1fe"}.bi-calendar2-fill::before{content:"\f1ff"}.bi-calendar2-minus-fill::before{content:"\f200"}.bi-calendar2-minus::before{content:"\f201"}.bi-calendar2-month-fill::before{content:"\f202"}.bi-calendar2-month::before{content:"\f203"}.bi-calendar2-plus-fill::before{content:"\f204"}.bi-calendar2-plus::before{content:"\f205"}.bi-calendar2-range-fill::before{content:"\f206"}.bi-calendar2-range::before{content:"\f207"}.bi-calendar2-week-fill::before{content:"\f208"}.bi-calendar2-week::before{content:"\f209"}.bi-calendar2-x-fill::before{content:"\f20a"}.bi-calendar2-x::before{content:"\f20b"}.bi-calendar2::before{content:"\f20c"}.bi-calendar3-event-fill::before{content:"\f20d"}.bi-calendar3-event::before{content:"\f20e"}.bi-calendar3-fill::before{content:"\f20f"}.bi-calendar3-range-fill::before{content:"\f210"}.bi-calendar3-range::before{content:"\f211"}.bi-calendar3-week-fill::before{content:"\f212"}.bi-calendar3-week::before{content:"\f213"}.bi-calendar3::before{content:"\f214"}.bi-calendar4-event::before{content:"\f215"}.bi-calendar4-range::before{content:"\f216"}.bi-calendar4-week::before{content:"\f217"}.bi-calendar4::before{content:"\f218"}.bi-camera-fill::before{content:"\f219"}.bi-camera-reels-fill::before{content:"\f21a"}.bi-camera-reels::before{content:"\f21b"}.bi-camera-video-fill::before{content:"\f21c"}.bi-camera-video-off-fill::before{content:"\f21d"}.bi-camera-video-off::before{content:"\f21e"}.bi-camera-video::before{content:"\f21f"}.bi-camera::before{content:"\f220"}.bi-camera2::before{content:"\f221"}.bi-capslock-fill::before{content:"\f222"}.bi-capslock::before{content:"\f223"}.bi-card-checklist::before{content:"\f224"}.bi-card-heading::before{content:"\f225"}.bi-card-image::before{content:"\f226"}.bi-card-list::before{content:"\f227"}.bi-card-text::before{content:"\f228"}.bi-caret-down-fill::before{content:"\f229"}.bi-caret-down-square-fill::before{content:"\f22a"}.bi-caret-down-square::before{content:"\f22b"}.bi-caret-down::before{content:"\f22c"}.bi-caret-left-fill::before{content:"\f22d"}.bi-caret-left-square-fill::before{content:"\f22e"}.bi-caret-left-square::before{content:"\f22f"}.bi-caret-left::before{content:"\f230"}.bi-caret-right-fill::before{content:"\f231"}.bi-caret-right-square-fill::before{content:"\f232"}.bi-caret-right-square::before{content:"\f233"}.bi-caret-right::before{content:"\f234"}.bi-caret-up-fill::before{content:"\f235"}.bi-caret-up-square-fill::before{content:"\f236"}.bi-caret-up-square::before{content:"\f237"}.bi-caret-up::before{content:"\f238"}.bi-cart-check-fill::before{content:"\f239"}.bi-cart-check::before{content:"\f23a"}.bi-cart-dash-fill::before{content:"\f23b"}.bi-cart-dash::before{content:"\f23c"}.bi-cart-fill::before{content:"\f23d"}.bi-cart-plus-fill::before{content:"\f23e"}.bi-cart-plus::before{content:"\f23f"}.bi-cart-x-fill::before{content:"\f240"}.bi-cart-x::before{content:"\f241"}.bi-cart::before{content:"\f242"}.bi-cart2::before{content:"\f243"}.bi-cart3::before{content:"\f244"}.bi-cart4::before{content:"\f245"}.bi-cash-stack::before{content:"\f246"}.bi-cash::before{content:"\f247"}.bi-cast::before{content:"\f248"}.bi-chat-dots-fill::before{content:"\f249"}.bi-chat-dots::before{content:"\f24a"}.bi-chat-fill::before{content:"\f24b"}.bi-chat-left-dots-fill::before{content:"\f24c"}.bi-chat-left-dots::before{content:"\f24d"}.bi-chat-left-fill::before{content:"\f24e"}.bi-chat-left-quote-fill::before{content:"\f24f"}.bi-chat-left-quote::before{content:"\f250"}.bi-chat-left-text-fill::before{content:"\f251"}.bi-chat-left-text::before{content:"\f252"}.bi-chat-left::before{content:"\f253"}.bi-chat-quote-fill::before{content:"\f254"}.bi-chat-quote::before{content:"\f255"}.bi-chat-right-dots-fill::before{content:"\f256"}.bi-chat-right-dots::before{content:"\f257"}.bi-chat-right-fill::before{content:"\f258"}.bi-chat-right-quote-fill::before{content:"\f259"}.bi-chat-right-quote::before{content:"\f25a"}.bi-chat-right-text-fill::before{content:"\f25b"}.bi-chat-right-text::before{content:"\f25c"}.bi-chat-right::before{content:"\f25d"}.bi-chat-square-dots-fill::before{content:"\f25e"}.bi-chat-square-dots::before{content:"\f25f"}.bi-chat-square-fill::before{content:"\f260"}.bi-chat-square-quote-fill::before{content:"\f261"}.bi-chat-square-quote::before{content:"\f262"}.bi-chat-square-text-fill::before{content:"\f263"}.bi-chat-square-text::before{content:"\f264"}.bi-chat-square::before{content:"\f265"}.bi-chat-text-fill::before{content:"\f266"}.bi-chat-text::before{content:"\f267"}.bi-chat::before{content:"\f268"}.bi-check-all::before{content:"\f269"}.bi-check-circle-fill::before{content:"\f26a"}.bi-check-circle::before{content:"\f26b"}.bi-check-square-fill::before{content:"\f26c"}.bi-check-square::before{content:"\f26d"}.bi-check::before{content:"\f26e"}.bi-check2-all::before{content:"\f26f"}.bi-check2-circle::before{content:"\f270"}.bi-check2-square::before{content:"\f271"}.bi-check2::before{content:"\f272"}.bi-chevron-bar-contract::before{content:"\f273"}.bi-chevron-bar-down::before{content:"\f274"}.bi-chevron-bar-expand::before{content:"\f275"}.bi-chevron-bar-left::before{content:"\f276"}.bi-chevron-bar-right::before{content:"\f277"}.bi-chevron-bar-up::before{content:"\f278"}.bi-chevron-compact-down::before{content:"\f279"}.bi-chevron-compact-left::before{content:"\f27a"}.bi-chevron-compact-right::before{content:"\f27b"}.bi-chevron-compact-up::before{content:"\f27c"}.bi-chevron-contract::before{content:"\f27d"}.bi-chevron-double-down::before{content:"\f27e"}.bi-chevron-double-left::before{content:"\f27f"}.bi-chevron-double-right::before{content:"\f280"}.bi-chevron-double-up::before{content:"\f281"}.bi-chevron-down::before{content:"\f282"}.bi-chevron-expand::before{content:"\f283"}.bi-chevron-left::before{content:"\f284"}.bi-chevron-right::before{content:"\f285"}.bi-chevron-up::before{content:"\f286"}.bi-circle-fill::before{content:"\f287"}.bi-circle-half::before{content:"\f288"}.bi-circle-square::before{content:"\f289"}.bi-circle::before{content:"\f28a"}.bi-clipboard-check::before{content:"\f28b"}.bi-clipboard-data::before{content:"\f28c"}.bi-clipboard-minus::before{content:"\f28d"}.bi-clipboard-plus::before{content:"\f28e"}.bi-clipboard-x::before{content:"\f28f"}.bi-clipboard::before{content:"\f290"}.bi-clock-fill::before{content:"\f291"}.bi-clock-history::before{content:"\f292"}.bi-clock::before{content:"\f293"}.bi-cloud-arrow-down-fill::before{content:"\f294"}.bi-cloud-arrow-down::before{content:"\f295"}.bi-cloud-arrow-up-fill::before{content:"\f296"}.bi-cloud-arrow-up::before{content:"\f297"}.bi-cloud-check-fill::before{content:"\f298"}.bi-cloud-check::before{content:"\f299"}.bi-cloud-download-fill::before{content:"\f29a"}.bi-cloud-download::before{content:"\f29b"}.bi-cloud-drizzle-fill::before{content:"\f29c"}.bi-cloud-drizzle::before{content:"\f29d"}.bi-cloud-fill::before{content:"\f29e"}.bi-cloud-fog-fill::before{content:"\f29f"}.bi-cloud-fog::before{content:"\f2a0"}.bi-cloud-fog2-fill::before{content:"\f2a1"}.bi-cloud-fog2::before{content:"\f2a2"}.bi-cloud-hail-fill::before{content:"\f2a3"}.bi-cloud-hail::before{content:"\f2a4"}.bi-cloud-haze-fill::before{content:"\f2a6"}.bi-cloud-haze::before{content:"\f2a7"}.bi-cloud-haze2-fill::before{content:"\f2a8"}.bi-cloud-lightning-fill::before{content:"\f2a9"}.bi-cloud-lightning-rain-fill::before{content:"\f2aa"}.bi-cloud-lightning-rain::before{content:"\f2ab"}.bi-cloud-lightning::before{content:"\f2ac"}.bi-cloud-minus-fill::before{content:"\f2ad"}.bi-cloud-minus::before{content:"\f2ae"}.bi-cloud-moon-fill::before{content:"\f2af"}.bi-cloud-moon::before{content:"\f2b0"}.bi-cloud-plus-fill::before{content:"\f2b1"}.bi-cloud-plus::before{content:"\f2b2"}.bi-cloud-rain-fill::before{content:"\f2b3"}.bi-cloud-rain-heavy-fill::before{content:"\f2b4"}.bi-cloud-rain-heavy::before{content:"\f2b5"}.bi-cloud-rain::before{content:"\f2b6"}.bi-cloud-slash-fill::before{content:"\f2b7"}.bi-cloud-slash::before{content:"\f2b8"}.bi-cloud-sleet-fill::before{content:"\f2b9"}.bi-cloud-sleet::before{content:"\f2ba"}.bi-cloud-snow-fill::before{content:"\f2bb"}.bi-cloud-snow::before{content:"\f2bc"}.bi-cloud-sun-fill::before{content:"\f2bd"}.bi-cloud-sun::before{content:"\f2be"}.bi-cloud-upload-fill::before{content:"\f2bf"}.bi-cloud-upload::before{content:"\f2c0"}.bi-cloud::before{content:"\f2c1"}.bi-clouds-fill::before{content:"\f2c2"}.bi-clouds::before{content:"\f2c3"}.bi-cloudy-fill::before{content:"\f2c4"}.bi-cloudy::before{content:"\f2c5"}.bi-code-slash::before{content:"\f2c6"}.bi-code-square::before{content:"\f2c7"}.bi-code::before{content:"\f2c8"}.bi-collection-fill::before{content:"\f2c9"}.bi-collection-play-fill::before{content:"\f2ca"}.bi-collection-play::before{content:"\f2cb"}.bi-collection::before{content:"\f2cc"}.bi-columns-gap::before{content:"\f2cd"}.bi-columns::before{content:"\f2ce"}.bi-command::before{content:"\f2cf"}.bi-compass-fill::before{content:"\f2d0"}.bi-compass::before{content:"\f2d1"}.bi-cone-striped::before{content:"\f2d2"}.bi-cone::before{content:"\f2d3"}.bi-controller::before{content:"\f2d4"}.bi-cpu-fill::before{content:"\f2d5"}.bi-cpu::before{content:"\f2d6"}.bi-credit-card-2-back-fill::before{content:"\f2d7"}.bi-credit-card-2-back::before{content:"\f2d8"}.bi-credit-card-2-front-fill::before{content:"\f2d9"}.bi-credit-card-2-front::before{content:"\f2da"}.bi-credit-card-fill::before{content:"\f2db"}.bi-credit-card::before{content:"\f2dc"}.bi-crop::before{content:"\f2dd"}.bi-cup-fill::before{content:"\f2de"}.bi-cup-straw::before{content:"\f2df"}.bi-cup::before{content:"\f2e0"}.bi-cursor-fill::before{content:"\f2e1"}.bi-cursor-text::before{content:"\f2e2"}.bi-cursor::before{content:"\f2e3"}.bi-dash-circle-dotted::before{content:"\f2e4"}.bi-dash-circle-fill::before{content:"\f2e5"}.bi-dash-circle::before{content:"\f2e6"}.bi-dash-square-dotted::before{content:"\f2e7"}.bi-dash-square-fill::before{content:"\f2e8"}.bi-dash-square::before{content:"\f2e9"}.bi-dash::before{content:"\f2ea"}.bi-diagram-2-fill::before{content:"\f2eb"}.bi-diagram-2::before{content:"\f2ec"}.bi-diagram-3-fill::before{content:"\f2ed"}.bi-diagram-3::before{content:"\f2ee"}.bi-diamond-fill::before{content:"\f2ef"}.bi-diamond-half::before{content:"\f2f0"}.bi-diamond::before{content:"\f2f1"}.bi-dice-1-fill::before{content:"\f2f2"}.bi-dice-1::before{content:"\f2f3"}.bi-dice-2-fill::before{content:"\f2f4"}.bi-dice-2::before{content:"\f2f5"}.bi-dice-3-fill::before{content:"\f2f6"}.bi-dice-3::before{content:"\f2f7"}.bi-dice-4-fill::before{content:"\f2f8"}.bi-dice-4::before{content:"\f2f9"}.bi-dice-5-fill::before{content:"\f2fa"}.bi-dice-5::before{content:"\f2fb"}.bi-dice-6-fill::before{content:"\f2fc"}.bi-dice-6::before{content:"\f2fd"}.bi-disc-fill::before{content:"\f2fe"}.bi-disc::before{content:"\f2ff"}.bi-discord::before{content:"\f300"}.bi-display-fill::before{content:"\f301"}.bi-display::before{content:"\f302"}.bi-distribute-horizontal::before{content:"\f303"}.bi-distribute-vertical::before{content:"\f304"}.bi-door-closed-fill::before{content:"\f305"}.bi-door-closed::before{content:"\f306"}.bi-door-open-fill::before{content:"\f307"}.bi-door-open::before{content:"\f308"}.bi-dot::before{content:"\f309"}.bi-download::before{content:"\f30a"}.bi-droplet-fill::before{content:"\f30b"}.bi-droplet-half::before{content:"\f30c"}.bi-droplet::before{content:"\f30d"}.bi-earbuds::before{content:"\f30e"}.bi-easel-fill::before{content:"\f30f"}.bi-easel::before{content:"\f310"}.bi-egg-fill::before{content:"\f311"}.bi-egg-fried::before{content:"\f312"}.bi-egg::before{content:"\f313"}.bi-eject-fill::before{content:"\f314"}.bi-eject::before{content:"\f315"}.bi-emoji-angry-fill::before{content:"\f316"}.bi-emoji-angry::before{content:"\f317"}.bi-emoji-dizzy-fill::before{content:"\f318"}.bi-emoji-dizzy::before{content:"\f319"}.bi-emoji-expressionless-fill::before{content:"\f31a"}.bi-emoji-expressionless::before{content:"\f31b"}.bi-emoji-frown-fill::before{content:"\f31c"}.bi-emoji-frown::before{content:"\f31d"}.bi-emoji-heart-eyes-fill::before{content:"\f31e"}.bi-emoji-heart-eyes::before{content:"\f31f"}.bi-emoji-laughing-fill::before{content:"\f320"}.bi-emoji-laughing::before{content:"\f321"}.bi-emoji-neutral-fill::before{content:"\f322"}.bi-emoji-neutral::before{content:"\f323"}.bi-emoji-smile-fill::before{content:"\f324"}.bi-emoji-smile-upside-down-fill::before{content:"\f325"}.bi-emoji-smile-upside-down::before{content:"\f326"}.bi-emoji-smile::before{content:"\f327"}.bi-emoji-sunglasses-fill::before{content:"\f328"}.bi-emoji-sunglasses::before{content:"\f329"}.bi-emoji-wink-fill::before{content:"\f32a"}.bi-emoji-wink::before{content:"\f32b"}.bi-envelope-fill::before{content:"\f32c"}.bi-envelope-open-fill::before{content:"\f32d"}.bi-envelope-open::before{content:"\f32e"}.bi-envelope::before{content:"\f32f"}.bi-eraser-fill::before{content:"\f330"}.bi-eraser::before{content:"\f331"}.bi-exclamation-circle-fill::before{content:"\f332"}.bi-exclamation-circle::before{content:"\f333"}.bi-exclamation-diamond-fill::before{content:"\f334"}.bi-exclamation-diamond::before{content:"\f335"}.bi-exclamation-octagon-fill::before{content:"\f336"}.bi-exclamation-octagon::before{content:"\f337"}.bi-exclamation-square-fill::before{content:"\f338"}.bi-exclamation-square::before{content:"\f339"}.bi-exclamation-triangle-fill::before{content:"\f33a"}.bi-exclamation-triangle::before{content:"\f33b"}.bi-exclamation::before{content:"\f33c"}.bi-exclude::before{content:"\f33d"}.bi-eye-fill::before{content:"\f33e"}.bi-eye-slash-fill::before{content:"\f33f"}.bi-eye-slash::before{content:"\f340"}.bi-eye::before{content:"\f341"}.bi-eyedropper::before{content:"\f342"}.bi-eyeglasses::before{content:"\f343"}.bi-facebook::before{content:"\f344"}.bi-file-arrow-down-fill::before{content:"\f345"}.bi-file-arrow-down::before{content:"\f346"}.bi-file-arrow-up-fill::before{content:"\f347"}.bi-file-arrow-up::before{content:"\f348"}.bi-file-bar-graph-fill::before{content:"\f349"}.bi-file-bar-graph::before{content:"\f34a"}.bi-file-binary-fill::before{content:"\f34b"}.bi-file-binary::before{content:"\f34c"}.bi-file-break-fill::before{content:"\f34d"}.bi-file-break::before{content:"\f34e"}.bi-file-check-fill::before{content:"\f34f"}.bi-file-check::before{content:"\f350"}.bi-file-code-fill::before{content:"\f351"}.bi-file-code::before{content:"\f352"}.bi-file-diff-fill::before{content:"\f353"}.bi-file-diff::before{content:"\f354"}.bi-file-earmark-arrow-down-fill::before{content:"\f355"}.bi-file-earmark-arrow-down::before{content:"\f356"}.bi-file-earmark-arrow-up-fill::before{content:"\f357"}.bi-file-earmark-arrow-up::before{content:"\f358"}.bi-file-earmark-bar-graph-fill::before{content:"\f359"}.bi-file-earmark-bar-graph::before{content:"\f35a"}.bi-file-earmark-binary-fill::before{content:"\f35b"}.bi-file-earmark-binary::before{content:"\f35c"}.bi-file-earmark-break-fill::before{content:"\f35d"}.bi-file-earmark-break::before{content:"\f35e"}.bi-file-earmark-check-fill::before{content:"\f35f"}.bi-file-earmark-check::before{content:"\f360"}.bi-file-earmark-code-fill::before{content:"\f361"}.bi-file-earmark-code::before{content:"\f362"}.bi-file-earmark-diff-fill::before{content:"\f363"}.bi-file-earmark-diff::before{content:"\f364"}.bi-file-earmark-easel-fill::before{content:"\f365"}.bi-file-earmark-easel::before{content:"\f366"}.bi-file-earmark-excel-fill::before{content:"\f367"}.bi-file-earmark-excel::before{content:"\f368"}.bi-file-earmark-fill::before{content:"\f369"}.bi-file-earmark-font-fill::before{content:"\f36a"}.bi-file-earmark-font::before{content:"\f36b"}.bi-file-earmark-image-fill::before{content:"\f36c"}.bi-file-earmark-image::before{content:"\f36d"}.bi-file-earmark-lock-fill::before{content:"\f36e"}.bi-file-earmark-lock::before{content:"\f36f"}.bi-file-earmark-lock2-fill::before{content:"\f370"}.bi-file-earmark-lock2::before{content:"\f371"}.bi-file-earmark-medical-fill::before{content:"\f372"}.bi-file-earmark-medical::before{content:"\f373"}.bi-file-earmark-minus-fill::before{content:"\f374"}.bi-file-earmark-minus::before{content:"\f375"}.bi-file-earmark-music-fill::before{content:"\f376"}.bi-file-earmark-music::before{content:"\f377"}.bi-file-earmark-person-fill::before{content:"\f378"}.bi-file-earmark-person::before{content:"\f379"}.bi-file-earmark-play-fill::before{content:"\f37a"}.bi-file-earmark-play::before{content:"\f37b"}.bi-file-earmark-plus-fill::before{content:"\f37c"}.bi-file-earmark-plus::before{content:"\f37d"}.bi-file-earmark-post-fill::before{content:"\f37e"}.bi-file-earmark-post::before{content:"\f37f"}.bi-file-earmark-ppt-fill::before{content:"\f380"}.bi-file-earmark-ppt::before{content:"\f381"}.bi-file-earmark-richtext-fill::before{content:"\f382"}.bi-file-earmark-richtext::before{content:"\f383"}.bi-file-earmark-ruled-fill::before{content:"\f384"}.bi-file-earmark-ruled::before{content:"\f385"}.bi-file-earmark-slides-fill::before{content:"\f386"}.bi-file-earmark-slides::before{content:"\f387"}.bi-file-earmark-spreadsheet-fill::before{content:"\f388"}.bi-file-earmark-spreadsheet::before{content:"\f389"}.bi-file-earmark-text-fill::before{content:"\f38a"}.bi-file-earmark-text::before{content:"\f38b"}.bi-file-earmark-word-fill::before{content:"\f38c"}.bi-file-earmark-word::before{content:"\f38d"}.bi-file-earmark-x-fill::before{content:"\f38e"}.bi-file-earmark-x::before{content:"\f38f"}.bi-file-earmark-zip-fill::before{content:"\f390"}.bi-file-earmark-zip::before{content:"\f391"}.bi-file-earmark::before{content:"\f392"}.bi-file-easel-fill::before{content:"\f393"}.bi-file-easel::before{content:"\f394"}.bi-file-excel-fill::before{content:"\f395"}.bi-file-excel::before{content:"\f396"}.bi-file-fill::before{content:"\f397"}.bi-file-font-fill::before{content:"\f398"}.bi-file-font::before{content:"\f399"}.bi-file-image-fill::before{content:"\f39a"}.bi-file-image::before{content:"\f39b"}.bi-file-lock-fill::before{content:"\f39c"}.bi-file-lock::before{content:"\f39d"}.bi-file-lock2-fill::before{content:"\f39e"}.bi-file-lock2::before{content:"\f39f"}.bi-file-medical-fill::before{content:"\f3a0"}.bi-file-medical::before{content:"\f3a1"}.bi-file-minus-fill::before{content:"\f3a2"}.bi-file-minus::before{content:"\f3a3"}.bi-file-music-fill::before{content:"\f3a4"}.bi-file-music::before{content:"\f3a5"}.bi-file-person-fill::before{content:"\f3a6"}.bi-file-person::before{content:"\f3a7"}.bi-file-play-fill::before{content:"\f3a8"}.bi-file-play::before{content:"\f3a9"}.bi-file-plus-fill::before{content:"\f3aa"}.bi-file-plus::before{content:"\f3ab"}.bi-file-post-fill::before{content:"\f3ac"}.bi-file-post::before{content:"\f3ad"}.bi-file-ppt-fill::before{content:"\f3ae"}.bi-file-ppt::before{content:"\f3af"}.bi-file-richtext-fill::before{content:"\f3b0"}.bi-file-richtext::before{content:"\f3b1"}.bi-file-ruled-fill::before{content:"\f3b2"}.bi-file-ruled::before{content:"\f3b3"}.bi-file-slides-fill::before{content:"\f3b4"}.bi-file-slides::before{content:"\f3b5"}.bi-file-spreadsheet-fill::before{content:"\f3b6"}.bi-file-spreadsheet::before{content:"\f3b7"}.bi-file-text-fill::before{content:"\f3b8"}.bi-file-text::before{content:"\f3b9"}.bi-file-word-fill::before{content:"\f3ba"}.bi-file-word::before{content:"\f3bb"}.bi-file-x-fill::before{content:"\f3bc"}.bi-file-x::before{content:"\f3bd"}.bi-file-zip-fill::before{content:"\f3be"}.bi-file-zip::before{content:"\f3bf"}.bi-file::before{content:"\f3c0"}.bi-files-alt::before{content:"\f3c1"}.bi-files::before{content:"\f3c2"}.bi-film::before{content:"\f3c3"}.bi-filter-circle-fill::before{content:"\f3c4"}.bi-filter-circle::before{content:"\f3c5"}.bi-filter-left::before{content:"\f3c6"}.bi-filter-right::before{content:"\f3c7"}.bi-filter-square-fill::before{content:"\f3c8"}.bi-filter-square::before{content:"\f3c9"}.bi-filter::before{content:"\f3ca"}.bi-flag-fill::before{content:"\f3cb"}.bi-flag::before{content:"\f3cc"}.bi-flower1::before{content:"\f3cd"}.bi-flower2::before{content:"\f3ce"}.bi-flower3::before{content:"\f3cf"}.bi-folder-check::before{content:"\f3d0"}.bi-folder-fill::before{content:"\f3d1"}.bi-folder-minus::before{content:"\f3d2"}.bi-folder-plus::before{content:"\f3d3"}.bi-folder-symlink-fill::before{content:"\f3d4"}.bi-folder-symlink::before{content:"\f3d5"}.bi-folder-x::before{content:"\f3d6"}.bi-folder::before{content:"\f3d7"}.bi-folder2-open::before{content:"\f3d8"}.bi-folder2::before{content:"\f3d9"}.bi-fonts::before{content:"\f3da"}.bi-forward-fill::before{content:"\f3db"}.bi-forward::before{content:"\f3dc"}.bi-front::before{content:"\f3dd"}.bi-fullscreen-exit::before{content:"\f3de"}.bi-fullscreen::before{content:"\f3df"}.bi-funnel-fill::before{content:"\f3e0"}.bi-funnel::before{content:"\f3e1"}.bi-gear-fill::before{content:"\f3e2"}.bi-gear-wide-connected::before{content:"\f3e3"}.bi-gear-wide::before{content:"\f3e4"}.bi-gear::before{content:"\f3e5"}.bi-gem::before{content:"\f3e6"}.bi-geo-alt-fill::before{content:"\f3e7"}.bi-geo-alt::before{content:"\f3e8"}.bi-geo-fill::before{content:"\f3e9"}.bi-geo::before{content:"\f3ea"}.bi-gift-fill::before{content:"\f3eb"}.bi-gift::before{content:"\f3ec"}.bi-github::before{content:"\f3ed"}.bi-globe::before{content:"\f3ee"}.bi-globe2::before{content:"\f3ef"}.bi-google::before{content:"\f3f0"}.bi-graph-down::before{content:"\f3f1"}.bi-graph-up::before{content:"\f3f2"}.bi-grid-1x2-fill::before{content:"\f3f3"}.bi-grid-1x2::before{content:"\f3f4"}.bi-grid-3x2-gap-fill::before{content:"\f3f5"}.bi-grid-3x2-gap::before{content:"\f3f6"}.bi-grid-3x2::before{content:"\f3f7"}.bi-grid-3x3-gap-fill::before{content:"\f3f8"}.bi-grid-3x3-gap::before{content:"\f3f9"}.bi-grid-3x3::before{content:"\f3fa"}.bi-grid-fill::before{content:"\f3fb"}.bi-grid::before{content:"\f3fc"}.bi-grip-horizontal::before{content:"\f3fd"}.bi-grip-vertical::before{content:"\f3fe"}.bi-hammer::before{content:"\f3ff"}.bi-hand-index-fill::before{content:"\f400"}.bi-hand-index-thumb-fill::before{content:"\f401"}.bi-hand-index-thumb::before{content:"\f402"}.bi-hand-index::before{content:"\f403"}.bi-hand-thumbs-down-fill::before{content:"\f404"}.bi-hand-thumbs-down::before{content:"\f405"}.bi-hand-thumbs-up-fill::before{content:"\f406"}.bi-hand-thumbs-up::before{content:"\f407"}.bi-handbag-fill::before{content:"\f408"}.bi-handbag::before{content:"\f409"}.bi-hash::before{content:"\f40a"}.bi-hdd-fill::before{content:"\f40b"}.bi-hdd-network-fill::before{content:"\f40c"}.bi-hdd-network::before{content:"\f40d"}.bi-hdd-rack-fill::before{content:"\f40e"}.bi-hdd-rack::before{content:"\f40f"}.bi-hdd-stack-fill::before{content:"\f410"}.bi-hdd-stack::before{content:"\f411"}.bi-hdd::before{content:"\f412"}.bi-headphones::before{content:"\f413"}.bi-headset::before{content:"\f414"}.bi-heart-fill::before{content:"\f415"}.bi-heart-half::before{content:"\f416"}.bi-heart::before{content:"\f417"}.bi-heptagon-fill::before{content:"\f418"}.bi-heptagon-half::before{content:"\f419"}.bi-heptagon::before{content:"\f41a"}.bi-hexagon-fill::before{content:"\f41b"}.bi-hexagon-half::before{content:"\f41c"}.bi-hexagon::before{content:"\f41d"}.bi-hourglass-bottom::before{content:"\f41e"}.bi-hourglass-split::before{content:"\f41f"}.bi-hourglass-top::before{content:"\f420"}.bi-hourglass::before{content:"\f421"}.bi-house-door-fill::before{content:"\f422"}.bi-house-door::before{content:"\f423"}.bi-house-fill::before{content:"\f424"}.bi-house::before{content:"\f425"}.bi-hr::before{content:"\f426"}.bi-hurricane::before{content:"\f427"}.bi-image-alt::before{content:"\f428"}.bi-image-fill::before{content:"\f429"}.bi-image::before{content:"\f42a"}.bi-images::before{content:"\f42b"}.bi-inbox-fill::before{content:"\f42c"}.bi-inbox::before{content:"\f42d"}.bi-inboxes-fill::before{content:"\f42e"}.bi-inboxes::before{content:"\f42f"}.bi-info-circle-fill::before{content:"\f430"}.bi-info-circle::before{content:"\f431"}.bi-info-square-fill::before{content:"\f432"}.bi-info-square::before{content:"\f433"}.bi-info::before{content:"\f434"}.bi-input-cursor-text::before{content:"\f435"}.bi-input-cursor::before{content:"\f436"}.bi-instagram::before{content:"\f437"}.bi-intersect::before{content:"\f438"}.bi-journal-album::before{content:"\f439"}.bi-journal-arrow-down::before{content:"\f43a"}.bi-journal-arrow-up::before{content:"\f43b"}.bi-journal-bookmark-fill::before{content:"\f43c"}.bi-journal-bookmark::before{content:"\f43d"}.bi-journal-check::before{content:"\f43e"}.bi-journal-code::before{content:"\f43f"}.bi-journal-medical::before{content:"\f440"}.bi-journal-minus::before{content:"\f441"}.bi-journal-plus::before{content:"\f442"}.bi-journal-richtext::before{content:"\f443"}.bi-journal-text::before{content:"\f444"}.bi-journal-x::before{content:"\f445"}.bi-journal::before{content:"\f446"}.bi-journals::before{content:"\f447"}.bi-joystick::before{content:"\f448"}.bi-justify-left::before{content:"\f449"}.bi-justify-right::before{content:"\f44a"}.bi-justify::before{content:"\f44b"}.bi-kanban-fill::before{content:"\f44c"}.bi-kanban::before{content:"\f44d"}.bi-key-fill::before{content:"\f44e"}.bi-key::before{content:"\f44f"}.bi-keyboard-fill::before{content:"\f450"}.bi-keyboard::before{content:"\f451"}.bi-ladder::before{content:"\f452"}.bi-lamp-fill::before{content:"\f453"}.bi-lamp::before{content:"\f454"}.bi-laptop-fill::before{content:"\f455"}.bi-laptop::before{content:"\f456"}.bi-layer-backward::before{content:"\f457"}.bi-layer-forward::before{content:"\f458"}.bi-layers-fill::before{content:"\f459"}.bi-layers-half::before{content:"\f45a"}.bi-layers::before{content:"\f45b"}.bi-layout-sidebar-inset-reverse::before{content:"\f45c"}.bi-layout-sidebar-inset::before{content:"\f45d"}.bi-layout-sidebar-reverse::before{content:"\f45e"}.bi-layout-sidebar::before{content:"\f45f"}.bi-layout-split::before{content:"\f460"}.bi-layout-text-sidebar-reverse::before{content:"\f461"}.bi-layout-text-sidebar::before{content:"\f462"}.bi-layout-text-window-reverse::before{content:"\f463"}.bi-layout-text-window::before{content:"\f464"}.bi-layout-three-columns::before{content:"\f465"}.bi-layout-wtf::before{content:"\f466"}.bi-life-preserver::before{content:"\f467"}.bi-lightbulb-fill::before{content:"\f468"}.bi-lightbulb-off-fill::before{content:"\f469"}.bi-lightbulb-off::before{content:"\f46a"}.bi-lightbulb::before{content:"\f46b"}.bi-lightning-charge-fill::before{content:"\f46c"}.bi-lightning-charge::before{content:"\f46d"}.bi-lightning-fill::before{content:"\f46e"}.bi-lightning::before{content:"\f46f"}.bi-link-45deg::before{content:"\f470"}.bi-link::before{content:"\f471"}.bi-linkedin::before{content:"\f472"}.bi-list-check::before{content:"\f473"}.bi-list-nested::before{content:"\f474"}.bi-list-ol::before{content:"\f475"}.bi-list-stars::before{content:"\f476"}.bi-list-task::before{content:"\f477"}.bi-list-ul::before{content:"\f478"}.bi-list::before{content:"\f479"}.bi-lock-fill::before{content:"\f47a"}.bi-lock::before{content:"\f47b"}.bi-mailbox::before{content:"\f47c"}.bi-mailbox2::before{content:"\f47d"}.bi-map-fill::before{content:"\f47e"}.bi-map::before{content:"\f47f"}.bi-markdown-fill::before{content:"\f480"}.bi-markdown::before{content:"\f481"}.bi-mask::before{content:"\f482"}.bi-megaphone-fill::before{content:"\f483"}.bi-megaphone::before{content:"\f484"}.bi-menu-app-fill::before{content:"\f485"}.bi-menu-app::before{content:"\f486"}.bi-menu-button-fill::before{content:"\f487"}.bi-menu-button-wide-fill::before{content:"\f488"}.bi-menu-button-wide::before{content:"\f489"}.bi-menu-button::before{content:"\f48a"}.bi-menu-down::before{content:"\f48b"}.bi-menu-up::before{content:"\f48c"}.bi-mic-fill::before{content:"\f48d"}.bi-mic-mute-fill::before{content:"\f48e"}.bi-mic-mute::before{content:"\f48f"}.bi-mic::before{content:"\f490"}.bi-minecart-loaded::before{content:"\f491"}.bi-minecart::before{content:"\f492"}.bi-moisture::before{content:"\f493"}.bi-moon-fill::before{content:"\f494"}.bi-moon-stars-fill::before{content:"\f495"}.bi-moon-stars::before{content:"\f496"}.bi-moon::before{content:"\f497"}.bi-mouse-fill::before{content:"\f498"}.bi-mouse::before{content:"\f499"}.bi-mouse2-fill::before{content:"\f49a"}.bi-mouse2::before{content:"\f49b"}.bi-mouse3-fill::before{content:"\f49c"}.bi-mouse3::before{content:"\f49d"}.bi-music-note-beamed::before{content:"\f49e"}.bi-music-note-list::before{content:"\f49f"}.bi-music-note::before{content:"\f4a0"}.bi-music-player-fill::before{content:"\f4a1"}.bi-music-player::before{content:"\f4a2"}.bi-newspaper::before{content:"\f4a3"}.bi-node-minus-fill::before{content:"\f4a4"}.bi-node-minus::before{content:"\f4a5"}.bi-node-plus-fill::before{content:"\f4a6"}.bi-node-plus::before{content:"\f4a7"}.bi-nut-fill::before{content:"\f4a8"}.bi-nut::before{content:"\f4a9"}.bi-octagon-fill::before{content:"\f4aa"}.bi-octagon-half::before{content:"\f4ab"}.bi-octagon::before{content:"\f4ac"}.bi-option::before{content:"\f4ad"}.bi-outlet::before{content:"\f4ae"}.bi-paint-bucket::before{content:"\f4af"}.bi-palette-fill::before{content:"\f4b0"}.bi-palette::before{content:"\f4b1"}.bi-palette2::before{content:"\f4b2"}.bi-paperclip::before{content:"\f4b3"}.bi-paragraph::before{content:"\f4b4"}.bi-patch-check-fill::before{content:"\f4b5"}.bi-patch-check::before{content:"\f4b6"}.bi-patch-exclamation-fill::before{content:"\f4b7"}.bi-patch-exclamation::before{content:"\f4b8"}.bi-patch-minus-fill::before{content:"\f4b9"}.bi-patch-minus::before{content:"\f4ba"}.bi-patch-plus-fill::before{content:"\f4bb"}.bi-patch-plus::before{content:"\f4bc"}.bi-patch-question-fill::before{content:"\f4bd"}.bi-patch-question::before{content:"\f4be"}.bi-pause-btn-fill::before{content:"\f4bf"}.bi-pause-btn::before{content:"\f4c0"}.bi-pause-circle-fill::before{content:"\f4c1"}.bi-pause-circle::before{content:"\f4c2"}.bi-pause-fill::before{content:"\f4c3"}.bi-pause::before{content:"\f4c4"}.bi-peace-fill::before{content:"\f4c5"}.bi-peace::before{content:"\f4c6"}.bi-pen-fill::before{content:"\f4c7"}.bi-pen::before{content:"\f4c8"}.bi-pencil-fill::before{content:"\f4c9"}.bi-pencil-square::before{content:"\f4ca"}.bi-pencil::before{content:"\f4cb"}.bi-pentagon-fill::before{content:"\f4cc"}.bi-pentagon-half::before{content:"\f4cd"}.bi-pentagon::before{content:"\f4ce"}.bi-people-fill::before{content:"\f4cf"}.bi-people::before{content:"\f4d0"}.bi-percent::before{content:"\f4d1"}.bi-person-badge-fill::before{content:"\f4d2"}.bi-person-badge::before{content:"\f4d3"}.bi-person-bounding-box::before{content:"\f4d4"}.bi-person-check-fill::before{content:"\f4d5"}.bi-person-check::before{content:"\f4d6"}.bi-person-circle::before{content:"\f4d7"}.bi-person-dash-fill::before{content:"\f4d8"}.bi-person-dash::before{content:"\f4d9"}.bi-person-fill::before{content:"\f4da"}.bi-person-lines-fill::before{content:"\f4db"}.bi-person-plus-fill::before{content:"\f4dc"}.bi-person-plus::before{content:"\f4dd"}.bi-person-square::before{content:"\f4de"}.bi-person-x-fill::before{content:"\f4df"}.bi-person-x::before{content:"\f4e0"}.bi-person::before{content:"\f4e1"}.bi-phone-fill::before{content:"\f4e2"}.bi-phone-landscape-fill::before{content:"\f4e3"}.bi-phone-landscape::before{content:"\f4e4"}.bi-phone-vibrate-fill::before{content:"\f4e5"}.bi-phone-vibrate::before{content:"\f4e6"}.bi-phone::before{content:"\f4e7"}.bi-pie-chart-fill::before{content:"\f4e8"}.bi-pie-chart::before{content:"\f4e9"}.bi-pin-angle-fill::before{content:"\f4ea"}.bi-pin-angle::before{content:"\f4eb"}.bi-pin-fill::before{content:"\f4ec"}.bi-pin::before{content:"\f4ed"}.bi-pip-fill::before{content:"\f4ee"}.bi-pip::before{content:"\f4ef"}.bi-play-btn-fill::before{content:"\f4f0"}.bi-play-btn::before{content:"\f4f1"}.bi-play-circle-fill::before{content:"\f4f2"}.bi-play-circle::before{content:"\f4f3"}.bi-play-fill::before{content:"\f4f4"}.bi-play::before{content:"\f4f5"}.bi-plug-fill::before{content:"\f4f6"}.bi-plug::before{content:"\f4f7"}.bi-plus-circle-dotted::before{content:"\f4f8"}.bi-plus-circle-fill::before{content:"\f4f9"}.bi-plus-circle::before{content:"\f4fa"}.bi-plus-square-dotted::before{content:"\f4fb"}.bi-plus-square-fill::before{content:"\f4fc"}.bi-plus-square::before{content:"\f4fd"}.bi-plus::before{content:"\f4fe"}.bi-power::before{content:"\f4ff"}.bi-printer-fill::before{content:"\f500"}.bi-printer::before{content:"\f501"}.bi-puzzle-fill::before{content:"\f502"}.bi-puzzle::before{content:"\f503"}.bi-question-circle-fill::before{content:"\f504"}.bi-question-circle::before{content:"\f505"}.bi-question-diamond-fill::before{content:"\f506"}.bi-question-diamond::before{content:"\f507"}.bi-question-octagon-fill::before{content:"\f508"}.bi-question-octagon::before{content:"\f509"}.bi-question-square-fill::before{content:"\f50a"}.bi-question-square::before{content:"\f50b"}.bi-question::before{content:"\f50c"}.bi-rainbow::before{content:"\f50d"}.bi-receipt-cutoff::before{content:"\f50e"}.bi-receipt::before{content:"\f50f"}.bi-reception-0::before{content:"\f510"}.bi-reception-1::before{content:"\f511"}.bi-reception-2::before{content:"\f512"}.bi-reception-3::before{content:"\f513"}.bi-reception-4::before{content:"\f514"}.bi-record-btn-fill::before{content:"\f515"}.bi-record-btn::before{content:"\f516"}.bi-record-circle-fill::before{content:"\f517"}.bi-record-circle::before{content:"\f518"}.bi-record-fill::before{content:"\f519"}.bi-record::before{content:"\f51a"}.bi-record2-fill::before{content:"\f51b"}.bi-record2::before{content:"\f51c"}.bi-reply-all-fill::before{content:"\f51d"}.bi-reply-all::before{content:"\f51e"}.bi-reply-fill::before{content:"\f51f"}.bi-reply::before{content:"\f520"}.bi-rss-fill::before{content:"\f521"}.bi-rss::before{content:"\f522"}.bi-rulers::before{content:"\f523"}.bi-save-fill::before{content:"\f524"}.bi-save::before{content:"\f525"}.bi-save2-fill::before{content:"\f526"}.bi-save2::before{content:"\f527"}.bi-scissors::before{content:"\f528"}.bi-screwdriver::before{content:"\f529"}.bi-search::before{content:"\f52a"}.bi-segmented-nav::before{content:"\f52b"}.bi-server::before{content:"\f52c"}.bi-share-fill::before{content:"\f52d"}.bi-share::before{content:"\f52e"}.bi-shield-check::before{content:"\f52f"}.bi-shield-exclamation::before{content:"\f530"}.bi-shield-fill-check::before{content:"\f531"}.bi-shield-fill-exclamation::before{content:"\f532"}.bi-shield-fill-minus::before{content:"\f533"}.bi-shield-fill-plus::before{content:"\f534"}.bi-shield-fill-x::before{content:"\f535"}.bi-shield-fill::before{content:"\f536"}.bi-shield-lock-fill::before{content:"\f537"}.bi-shield-lock::before{content:"\f538"}.bi-shield-minus::before{content:"\f539"}.bi-shield-plus::before{content:"\f53a"}.bi-shield-shaded::before{content:"\f53b"}.bi-shield-slash-fill::before{content:"\f53c"}.bi-shield-slash::before{content:"\f53d"}.bi-shield-x::before{content:"\f53e"}.bi-shield::before{content:"\f53f"}.bi-shift-fill::before{content:"\f540"}.bi-shift::before{content:"\f541"}.bi-shop-window::before{content:"\f542"}.bi-shop::before{content:"\f543"}.bi-shuffle::before{content:"\f544"}.bi-signpost-2-fill::before{content:"\f545"}.bi-signpost-2::before{content:"\f546"}.bi-signpost-fill::before{content:"\f547"}.bi-signpost-split-fill::before{content:"\f548"}.bi-signpost-split::before{content:"\f549"}.bi-signpost::before{content:"\f54a"}.bi-sim-fill::before{content:"\f54b"}.bi-sim::before{content:"\f54c"}.bi-skip-backward-btn-fill::before{content:"\f54d"}.bi-skip-backward-btn::before{content:"\f54e"}.bi-skip-backward-circle-fill::before{content:"\f54f"}.bi-skip-backward-circle::before{content:"\f550"}.bi-skip-backward-fill::before{content:"\f551"}.bi-skip-backward::before{content:"\f552"}.bi-skip-end-btn-fill::before{content:"\f553"}.bi-skip-end-btn::before{content:"\f554"}.bi-skip-end-circle-fill::before{content:"\f555"}.bi-skip-end-circle::before{content:"\f556"}.bi-skip-end-fill::before{content:"\f557"}.bi-skip-end::before{content:"\f558"}.bi-skip-forward-btn-fill::before{content:"\f559"}.bi-skip-forward-btn::before{content:"\f55a"}.bi-skip-forward-circle-fill::before{content:"\f55b"}.bi-skip-forward-circle::before{content:"\f55c"}.bi-skip-forward-fill::before{content:"\f55d"}.bi-skip-forward::before{content:"\f55e"}.bi-skip-start-btn-fill::before{content:"\f55f"}.bi-skip-start-btn::before{content:"\f560"}.bi-skip-start-circle-fill::before{content:"\f561"}.bi-skip-start-circle::before{content:"\f562"}.bi-skip-start-fill::before{content:"\f563"}.bi-skip-start::before{content:"\f564"}.bi-slack::before{content:"\f565"}.bi-slash-circle-fill::before{content:"\f566"}.bi-slash-circle::before{content:"\f567"}.bi-slash-square-fill::before{content:"\f568"}.bi-slash-square::before{content:"\f569"}.bi-slash::before{content:"\f56a"}.bi-sliders::before{content:"\f56b"}.bi-smartwatch::before{content:"\f56c"}.bi-snow::before{content:"\f56d"}.bi-snow2::before{content:"\f56e"}.bi-snow3::before{content:"\f56f"}.bi-sort-alpha-down-alt::before{content:"\f570"}.bi-sort-alpha-down::before{content:"\f571"}.bi-sort-alpha-up-alt::before{content:"\f572"}.bi-sort-alpha-up::before{content:"\f573"}.bi-sort-down-alt::before{content:"\f574"}.bi-sort-down::before{content:"\f575"}.bi-sort-numeric-down-alt::before{content:"\f576"}.bi-sort-numeric-down::before{content:"\f577"}.bi-sort-numeric-up-alt::before{content:"\f578"}.bi-sort-numeric-up::before{content:"\f579"}.bi-sort-up-alt::before{content:"\f57a"}.bi-sort-up::before{content:"\f57b"}.bi-soundwave::before{content:"\f57c"}.bi-speaker-fill::before{content:"\f57d"}.bi-speaker::before{content:"\f57e"}.bi-speedometer::before{content:"\f57f"}.bi-speedometer2::before{content:"\f580"}.bi-spellcheck::before{content:"\f581"}.bi-square-fill::before{content:"\f582"}.bi-square-half::before{content:"\f583"}.bi-square::before{content:"\f584"}.bi-stack::before{content:"\f585"}.bi-star-fill::before{content:"\f586"}.bi-star-half::before{content:"\f587"}.bi-star::before{content:"\f588"}.bi-stars::before{content:"\f589"}.bi-stickies-fill::before{content:"\f58a"}.bi-stickies::before{content:"\f58b"}.bi-sticky-fill::before{content:"\f58c"}.bi-sticky::before{content:"\f58d"}.bi-stop-btn-fill::before{content:"\f58e"}.bi-stop-btn::before{content:"\f58f"}.bi-stop-circle-fill::before{content:"\f590"}.bi-stop-circle::before{content:"\f591"}.bi-stop-fill::before{content:"\f592"}.bi-stop::before{content:"\f593"}.bi-stoplights-fill::before{content:"\f594"}.bi-stoplights::before{content:"\f595"}.bi-stopwatch-fill::before{content:"\f596"}.bi-stopwatch::before{content:"\f597"}.bi-subtract::before{content:"\f598"}.bi-suit-club-fill::before{content:"\f599"}.bi-suit-club::before{content:"\f59a"}.bi-suit-diamond-fill::before{content:"\f59b"}.bi-suit-diamond::before{content:"\f59c"}.bi-suit-heart-fill::before{content:"\f59d"}.bi-suit-heart::before{content:"\f59e"}.bi-suit-spade-fill::before{content:"\f59f"}.bi-suit-spade::before{content:"\f5a0"}.bi-sun-fill::before{content:"\f5a1"}.bi-sun::before{content:"\f5a2"}.bi-sunglasses::before{content:"\f5a3"}.bi-sunrise-fill::before{content:"\f5a4"}.bi-sunrise::before{content:"\f5a5"}.bi-sunset-fill::before{content:"\f5a6"}.bi-sunset::before{content:"\f5a7"}.bi-symmetry-horizontal::before{content:"\f5a8"}.bi-symmetry-vertical::before{content:"\f5a9"}.bi-table::before{content:"\f5aa"}.bi-tablet-fill::before{content:"\f5ab"}.bi-tablet-landscape-fill::before{content:"\f5ac"}.bi-tablet-landscape::before{content:"\f5ad"}.bi-tablet::before{content:"\f5ae"}.bi-tag-fill::before{content:"\f5af"}.bi-tag::before{content:"\f5b0"}.bi-tags-fill::before{content:"\f5b1"}.bi-tags::before{content:"\f5b2"}.bi-telegram::before{content:"\f5b3"}.bi-telephone-fill::before{content:"\f5b4"}.bi-telephone-forward-fill::before{content:"\f5b5"}.bi-telephone-forward::before{content:"\f5b6"}.bi-telephone-inbound-fill::before{content:"\f5b7"}.bi-telephone-inbound::before{content:"\f5b8"}.bi-telephone-minus-fill::before{content:"\f5b9"}.bi-telephone-minus::before{content:"\f5ba"}.bi-telephone-outbound-fill::before{content:"\f5bb"}.bi-telephone-outbound::before{content:"\f5bc"}.bi-telephone-plus-fill::before{content:"\f5bd"}.bi-telephone-plus::before{content:"\f5be"}.bi-telephone-x-fill::before{content:"\f5bf"}.bi-telephone-x::before{content:"\f5c0"}.bi-telephone::before{content:"\f5c1"}.bi-terminal-fill::before{content:"\f5c2"}.bi-terminal::before{content:"\f5c3"}.bi-text-center::before{content:"\f5c4"}.bi-text-indent-left::before{content:"\f5c5"}.bi-text-indent-right::before{content:"\f5c6"}.bi-text-left::before{content:"\f5c7"}.bi-text-paragraph::before{content:"\f5c8"}.bi-text-right::before{content:"\f5c9"}.bi-textarea-resize::before{content:"\f5ca"}.bi-textarea-t::before{content:"\f5cb"}.bi-textarea::before{content:"\f5cc"}.bi-thermometer-half::before{content:"\f5cd"}.bi-thermometer-high::before{content:"\f5ce"}.bi-thermometer-low::before{content:"\f5cf"}.bi-thermometer-snow::before{content:"\f5d0"}.bi-thermometer-sun::before{content:"\f5d1"}.bi-thermometer::before{content:"\f5d2"}.bi-three-dots-vertical::before{content:"\f5d3"}.bi-three-dots::before{content:"\f5d4"}.bi-toggle-off::before{content:"\f5d5"}.bi-toggle-on::before{content:"\f5d6"}.bi-toggle2-off::before{content:"\f5d7"}.bi-toggle2-on::before{content:"\f5d8"}.bi-toggles::before{content:"\f5d9"}.bi-toggles2::before{content:"\f5da"}.bi-tools::before{content:"\f5db"}.bi-tornado::before{content:"\f5dc"}.bi-trash-fill::before{content:"\f5dd"}.bi-trash::before{content:"\f5de"}.bi-trash2-fill::before{content:"\f5df"}.bi-trash2::before{content:"\f5e0"}.bi-tree-fill::before{content:"\f5e1"}.bi-tree::before{content:"\f5e2"}.bi-triangle-fill::before{content:"\f5e3"}.bi-triangle-half::before{content:"\f5e4"}.bi-triangle::before{content:"\f5e5"}.bi-trophy-fill::before{content:"\f5e6"}.bi-trophy::before{content:"\f5e7"}.bi-tropical-storm::before{content:"\f5e8"}.bi-truck-flatbed::before{content:"\f5e9"}.bi-truck::before{content:"\f5ea"}.bi-tsunami::before{content:"\f5eb"}.bi-tv-fill::before{content:"\f5ec"}.bi-tv::before{content:"\f5ed"}.bi-twitch::before{content:"\f5ee"}.bi-twitter::before{content:"\f5ef"}.bi-type-bold::before{content:"\f5f0"}.bi-type-h1::before{content:"\f5f1"}.bi-type-h2::before{content:"\f5f2"}.bi-type-h3::before{content:"\f5f3"}.bi-type-italic::before{content:"\f5f4"}.bi-type-strikethrough::before{content:"\f5f5"}.bi-type-underline::before{content:"\f5f6"}.bi-type::before{content:"\f5f7"}.bi-ui-checks-grid::before{content:"\f5f8"}.bi-ui-checks::before{content:"\f5f9"}.bi-ui-radios-grid::before{content:"\f5fa"}.bi-ui-radios::before{content:"\f5fb"}.bi-umbrella-fill::before{content:"\f5fc"}.bi-umbrella::before{content:"\f5fd"}.bi-union::before{content:"\f5fe"}.bi-unlock-fill::before{content:"\f5ff"}.bi-unlock::before{content:"\f600"}.bi-upc-scan::before{content:"\f601"}.bi-upc::before{content:"\f602"}.bi-upload::before{content:"\f603"}.bi-vector-pen::before{content:"\f604"}.bi-view-list::before{content:"\f605"}.bi-view-stacked::before{content:"\f606"}.bi-vinyl-fill::before{content:"\f607"}.bi-vinyl::before{content:"\f608"}.bi-voicemail::before{content:"\f609"}.bi-volume-down-fill::before{content:"\f60a"}.bi-volume-down::before{content:"\f60b"}.bi-volume-mute-fill::before{content:"\f60c"}.bi-volume-mute::before{content:"\f60d"}.bi-volume-off-fill::before{content:"\f60e"}.bi-volume-off::before{content:"\f60f"}.bi-volume-up-fill::before{content:"\f610"}.bi-volume-up::before{content:"\f611"}.bi-vr::before{content:"\f612"}.bi-wallet-fill::before{content:"\f613"}.bi-wallet::before{content:"\f614"}.bi-wallet2::before{content:"\f615"}.bi-watch::before{content:"\f616"}.bi-water::before{content:"\f617"}.bi-whatsapp::before{content:"\f618"}.bi-wifi-1::before{content:"\f619"}.bi-wifi-2::before{content:"\f61a"}.bi-wifi-off::before{content:"\f61b"}.bi-wifi::before{content:"\f61c"}.bi-wind::before{content:"\f61d"}.bi-window-dock::before{content:"\f61e"}.bi-window-sidebar::before{content:"\f61f"}.bi-window::before{content:"\f620"}.bi-wrench::before{content:"\f621"}.bi-x-circle-fill::before{content:"\f622"}.bi-x-circle::before{content:"\f623"}.bi-x-diamond-fill::before{content:"\f624"}.bi-x-diamond::before{content:"\f625"}.bi-x-octagon-fill::before{content:"\f626"}.bi-x-octagon::before{content:"\f627"}.bi-x-square-fill::before{content:"\f628"}.bi-x-square::before{content:"\f629"}.bi-x::before{content:"\f62a"}.bi-youtube::before{content:"\f62b"}.bi-zoom-in::before{content:"\f62c"}.bi-zoom-out::before{content:"\f62d"}.bi-bank::before{content:"\f62e"}.bi-bank2::before{content:"\f62f"}.bi-bell-slash-fill::before{content:"\f630"}.bi-bell-slash::before{content:"\f631"}.bi-cash-coin::before{content:"\f632"}.bi-check-lg::before{content:"\f633"}.bi-coin::before{content:"\f634"}.bi-currency-bitcoin::before{content:"\f635"}.bi-currency-dollar::before{content:"\f636"}.bi-currency-euro::before{content:"\f637"}.bi-currency-exchange::before{content:"\f638"}.bi-currency-pound::before{content:"\f639"}.bi-currency-yen::before{content:"\f63a"}.bi-dash-lg::before{content:"\f63b"}.bi-exclamation-lg::before{content:"\f63c"}.bi-file-earmark-pdf-fill::before{content:"\f63d"}.bi-file-earmark-pdf::before{content:"\f63e"}.bi-file-pdf-fill::before{content:"\f63f"}.bi-file-pdf::before{content:"\f640"}.bi-gender-ambiguous::before{content:"\f641"}.bi-gender-female::before{content:"\f642"}.bi-gender-male::before{content:"\f643"}.bi-gender-trans::before{content:"\f644"}.bi-headset-vr::before{content:"\f645"}.bi-info-lg::before{content:"\f646"}.bi-mastodon::before{content:"\f647"}.bi-messenger::before{content:"\f648"}.bi-piggy-bank-fill::before{content:"\f649"}.bi-piggy-bank::before{content:"\f64a"}.bi-pin-map-fill::before{content:"\f64b"}.bi-pin-map::before{content:"\f64c"}.bi-plus-lg::before{content:"\f64d"}.bi-question-lg::before{content:"\f64e"}.bi-recycle::before{content:"\f64f"}.bi-reddit::before{content:"\f650"}.bi-safe-fill::before{content:"\f651"}.bi-safe2-fill::before{content:"\f652"}.bi-safe2::before{content:"\f653"}.bi-sd-card-fill::before{content:"\f654"}.bi-sd-card::before{content:"\f655"}.bi-skype::before{content:"\f656"}.bi-slash-lg::before{content:"\f657"}.bi-translate::before{content:"\f658"}.bi-x-lg::before{content:"\f659"}.bi-safe::before{content:"\f65a"}.bi-apple::before{content:"\f65b"}.bi-microsoft::before{content:"\f65d"}.bi-windows::before{content:"\f65e"}.bi-behance::before{content:"\f65c"}.bi-dribbble::before{content:"\f65f"}.bi-line::before{content:"\f660"}.bi-medium::before{content:"\f661"}.bi-paypal::before{content:"\f662"}.bi-pinterest::before{content:"\f663"}.bi-signal::before{content:"\f664"}.bi-snapchat::before{content:"\f665"}.bi-spotify::before{content:"\f666"}.bi-stack-overflow::before{content:"\f667"}.bi-strava::before{content:"\f668"}.bi-wordpress::before{content:"\f669"}.bi-vimeo::before{content:"\f66a"}.bi-activity::before{content:"\f66b"}.bi-easel2-fill::before{content:"\f66c"}.bi-easel2::before{content:"\f66d"}.bi-easel3-fill::before{content:"\f66e"}.bi-easel3::before{content:"\f66f"}.bi-fan::before{content:"\f670"}.bi-fingerprint::before{content:"\f671"}.bi-graph-down-arrow::before{content:"\f672"}.bi-graph-up-arrow::before{content:"\f673"}.bi-hypnotize::before{content:"\f674"}.bi-magic::before{content:"\f675"}.bi-person-rolodex::before{content:"\f676"}.bi-person-video::before{content:"\f677"}.bi-person-video2::before{content:"\f678"}.bi-person-video3::before{content:"\f679"}.bi-person-workspace::before{content:"\f67a"}.bi-radioactive::before{content:"\f67b"}.bi-webcam-fill::before{content:"\f67c"}.bi-webcam::before{content:"\f67d"}.bi-yin-yang::before{content:"\f67e"}.bi-bandaid-fill::before{content:"\f680"}.bi-bandaid::before{content:"\f681"}.bi-bluetooth::before{content:"\f682"}.bi-body-text::before{content:"\f683"}.bi-boombox::before{content:"\f684"}.bi-boxes::before{content:"\f685"}.bi-dpad-fill::before{content:"\f686"}.bi-dpad::before{content:"\f687"}.bi-ear-fill::before{content:"\f688"}.bi-ear::before{content:"\f689"}.bi-envelope-check-fill::before{content:"\f68b"}.bi-envelope-check::before{content:"\f68c"}.bi-envelope-dash-fill::before{content:"\f68e"}.bi-envelope-dash::before{content:"\f68f"}.bi-envelope-exclamation-fill::before{content:"\f691"}.bi-envelope-exclamation::before{content:"\f692"}.bi-envelope-plus-fill::before{content:"\f693"}.bi-envelope-plus::before{content:"\f694"}.bi-envelope-slash-fill::before{content:"\f696"}.bi-envelope-slash::before{content:"\f697"}.bi-envelope-x-fill::before{content:"\f699"}.bi-envelope-x::before{content:"\f69a"}.bi-explicit-fill::before{content:"\f69b"}.bi-explicit::before{content:"\f69c"}.bi-git::before{content:"\f69d"}.bi-infinity::before{content:"\f69e"}.bi-list-columns-reverse::before{content:"\f69f"}.bi-list-columns::before{content:"\f6a0"}.bi-meta::before{content:"\f6a1"}.bi-nintendo-switch::before{content:"\f6a4"}.bi-pc-display-horizontal::before{content:"\f6a5"}.bi-pc-display::before{content:"\f6a6"}.bi-pc-horizontal::before{content:"\f6a7"}.bi-pc::before{content:"\f6a8"}.bi-playstation::before{content:"\f6a9"}.bi-plus-slash-minus::before{content:"\f6aa"}.bi-projector-fill::before{content:"\f6ab"}.bi-projector::before{content:"\f6ac"}.bi-qr-code-scan::before{content:"\f6ad"}.bi-qr-code::before{content:"\f6ae"}.bi-quora::before{content:"\f6af"}.bi-quote::before{content:"\f6b0"}.bi-robot::before{content:"\f6b1"}.bi-send-check-fill::before{content:"\f6b2"}.bi-send-check::before{content:"\f6b3"}.bi-send-dash-fill::before{content:"\f6b4"}.bi-send-dash::before{content:"\f6b5"}.bi-send-exclamation-fill::before{content:"\f6b7"}.bi-send-exclamation::before{content:"\f6b8"}.bi-send-fill::before{content:"\f6b9"}.bi-send-plus-fill::before{content:"\f6ba"}.bi-send-plus::before{content:"\f6bb"}.bi-send-slash-fill::before{content:"\f6bc"}.bi-send-slash::before{content:"\f6bd"}.bi-send-x-fill::before{content:"\f6be"}.bi-send-x::before{content:"\f6bf"}.bi-send::before{content:"\f6c0"}.bi-steam::before{content:"\f6c1"}.bi-terminal-dash::before{content:"\f6c3"}.bi-terminal-plus::before{content:"\f6c4"}.bi-terminal-split::before{content:"\f6c5"}.bi-ticket-detailed-fill::before{content:"\f6c6"}.bi-ticket-detailed::before{content:"\f6c7"}.bi-ticket-fill::before{content:"\f6c8"}.bi-ticket-perforated-fill::before{content:"\f6c9"}.bi-ticket-perforated::before{content:"\f6ca"}.bi-ticket::before{content:"\f6cb"}.bi-tiktok::before{content:"\f6cc"}.bi-window-dash::before{content:"\f6cd"}.bi-window-desktop::before{content:"\f6ce"}.bi-window-fullscreen::before{content:"\f6cf"}.bi-window-plus::before{content:"\f6d0"}.bi-window-split::before{content:"\f6d1"}.bi-window-stack::before{content:"\f6d2"}.bi-window-x::before{content:"\f6d3"}.bi-xbox::before{content:"\f6d4"}.bi-ethernet::before{content:"\f6d5"}.bi-hdmi-fill::before{content:"\f6d6"}.bi-hdmi::before{content:"\f6d7"}.bi-usb-c-fill::before{content:"\f6d8"}.bi-usb-c::before{content:"\f6d9"}.bi-usb-fill::before{content:"\f6da"}.bi-usb-plug-fill::before{content:"\f6db"}.bi-usb-plug::before{content:"\f6dc"}.bi-usb-symbol::before{content:"\f6dd"}.bi-usb::before{content:"\f6de"}.bi-boombox-fill::before{content:"\f6df"}.bi-displayport::before{content:"\f6e1"}.bi-gpu-card::before{content:"\f6e2"}.bi-memory::before{content:"\f6e3"}.bi-modem-fill::before{content:"\f6e4"}.bi-modem::before{content:"\f6e5"}.bi-motherboard-fill::before{content:"\f6e6"}.bi-motherboard::before{content:"\f6e7"}.bi-optical-audio-fill::before{content:"\f6e8"}.bi-optical-audio::before{content:"\f6e9"}.bi-pci-card::before{content:"\f6ea"}.bi-router-fill::before{content:"\f6eb"}.bi-router::before{content:"\f6ec"}.bi-thunderbolt-fill::before{content:"\f6ef"}.bi-thunderbolt::before{content:"\f6f0"}.bi-usb-drive-fill::before{content:"\f6f1"}.bi-usb-drive::before{content:"\f6f2"}.bi-usb-micro-fill::before{content:"\f6f3"}.bi-usb-micro::before{content:"\f6f4"}.bi-usb-mini-fill::before{content:"\f6f5"}.bi-usb-mini::before{content:"\f6f6"}.bi-cloud-haze2::before{content:"\f6f7"}.bi-device-hdd-fill::before{content:"\f6f8"}.bi-device-hdd::before{content:"\f6f9"}.bi-device-ssd-fill::before{content:"\f6fa"}.bi-device-ssd::before{content:"\f6fb"}.bi-displayport-fill::before{content:"\f6fc"}.bi-mortarboard-fill::before{content:"\f6fd"}.bi-mortarboard::before{content:"\f6fe"}.bi-terminal-x::before{content:"\f6ff"}.bi-arrow-through-heart-fill::before{content:"\f700"}.bi-arrow-through-heart::before{content:"\f701"}.bi-badge-sd-fill::before{content:"\f702"}.bi-badge-sd::before{content:"\f703"}.bi-bag-heart-fill::before{content:"\f704"}.bi-bag-heart::before{content:"\f705"}.bi-balloon-fill::before{content:"\f706"}.bi-balloon-heart-fill::before{content:"\f707"}.bi-balloon-heart::before{content:"\f708"}.bi-balloon::before{content:"\f709"}.bi-box2-fill::before{content:"\f70a"}.bi-box2-heart-fill::before{content:"\f70b"}.bi-box2-heart::before{content:"\f70c"}.bi-box2::before{content:"\f70d"}.bi-braces-asterisk::before{content:"\f70e"}.bi-calendar-heart-fill::before{content:"\f70f"}.bi-calendar-heart::before{content:"\f710"}.bi-calendar2-heart-fill::before{content:"\f711"}.bi-calendar2-heart::before{content:"\f712"}.bi-chat-heart-fill::before{content:"\f713"}.bi-chat-heart::before{content:"\f714"}.bi-chat-left-heart-fill::before{content:"\f715"}.bi-chat-left-heart::before{content:"\f716"}.bi-chat-right-heart-fill::before{content:"\f717"}.bi-chat-right-heart::before{content:"\f718"}.bi-chat-square-heart-fill::before{content:"\f719"}.bi-chat-square-heart::before{content:"\f71a"}.bi-clipboard-check-fill::before{content:"\f71b"}.bi-clipboard-data-fill::before{content:"\f71c"}.bi-clipboard-fill::before{content:"\f71d"}.bi-clipboard-heart-fill::before{content:"\f71e"}.bi-clipboard-heart::before{content:"\f71f"}.bi-clipboard-minus-fill::before{content:"\f720"}.bi-clipboard-plus-fill::before{content:"\f721"}.bi-clipboard-pulse::before{content:"\f722"}.bi-clipboard-x-fill::before{content:"\f723"}.bi-clipboard2-check-fill::before{content:"\f724"}.bi-clipboard2-check::before{content:"\f725"}.bi-clipboard2-data-fill::before{content:"\f726"}.bi-clipboard2-data::before{content:"\f727"}.bi-clipboard2-fill::before{content:"\f728"}.bi-clipboard2-heart-fill::before{content:"\f729"}.bi-clipboard2-heart::before{content:"\f72a"}.bi-clipboard2-minus-fill::before{content:"\f72b"}.bi-clipboard2-minus::before{content:"\f72c"}.bi-clipboard2-plus-fill::before{content:"\f72d"}.bi-clipboard2-plus::before{content:"\f72e"}.bi-clipboard2-pulse-fill::before{content:"\f72f"}.bi-clipboard2-pulse::before{content:"\f730"}.bi-clipboard2-x-fill::before{content:"\f731"}.bi-clipboard2-x::before{content:"\f732"}.bi-clipboard2::before{content:"\f733"}.bi-emoji-kiss-fill::before{content:"\f734"}.bi-emoji-kiss::before{content:"\f735"}.bi-envelope-heart-fill::before{content:"\f736"}.bi-envelope-heart::before{content:"\f737"}.bi-envelope-open-heart-fill::before{content:"\f738"}.bi-envelope-open-heart::before{content:"\f739"}.bi-envelope-paper-fill::before{content:"\f73a"}.bi-envelope-paper-heart-fill::before{content:"\f73b"}.bi-envelope-paper-heart::before{content:"\f73c"}.bi-envelope-paper::before{content:"\f73d"}.bi-filetype-aac::before{content:"\f73e"}.bi-filetype-ai::before{content:"\f73f"}.bi-filetype-bmp::before{content:"\f740"}.bi-filetype-cs::before{content:"\f741"}.bi-filetype-css::before{content:"\f742"}.bi-filetype-csv::before{content:"\f743"}.bi-filetype-doc::before{content:"\f744"}.bi-filetype-docx::before{content:"\f745"}.bi-filetype-exe::before{content:"\f746"}.bi-filetype-gif::before{content:"\f747"}.bi-filetype-heic::before{content:"\f748"}.bi-filetype-html::before{content:"\f749"}.bi-filetype-java::before{content:"\f74a"}.bi-filetype-jpg::before{content:"\f74b"}.bi-filetype-js::before{content:"\f74c"}.bi-filetype-jsx::before{content:"\f74d"}.bi-filetype-key::before{content:"\f74e"}.bi-filetype-m4p::before{content:"\f74f"}.bi-filetype-md::before{content:"\f750"}.bi-filetype-mdx::before{content:"\f751"}.bi-filetype-mov::before{content:"\f752"}.bi-filetype-mp3::before{content:"\f753"}.bi-filetype-mp4::before{content:"\f754"}.bi-filetype-otf::before{content:"\f755"}.bi-filetype-pdf::before{content:"\f756"}.bi-filetype-php::before{content:"\f757"}.bi-filetype-png::before{content:"\f758"}.bi-filetype-ppt::before{content:"\f75a"}.bi-filetype-psd::before{content:"\f75b"}.bi-filetype-py::before{content:"\f75c"}.bi-filetype-raw::before{content:"\f75d"}.bi-filetype-rb::before{content:"\f75e"}.bi-filetype-sass::before{content:"\f75f"}.bi-filetype-scss::before{content:"\f760"}.bi-filetype-sh::before{content:"\f761"}.bi-filetype-svg::before{content:"\f762"}.bi-filetype-tiff::before{content:"\f763"}.bi-filetype-tsx::before{content:"\f764"}.bi-filetype-ttf::before{content:"\f765"}.bi-filetype-txt::before{content:"\f766"}.bi-filetype-wav::before{content:"\f767"}.bi-filetype-woff::before{content:"\f768"}.bi-filetype-xls::before{content:"\f76a"}.bi-filetype-xml::before{content:"\f76b"}.bi-filetype-yml::before{content:"\f76c"}.bi-heart-arrow::before{content:"\f76d"}.bi-heart-pulse-fill::before{content:"\f76e"}.bi-heart-pulse::before{content:"\f76f"}.bi-heartbreak-fill::before{content:"\f770"}.bi-heartbreak::before{content:"\f771"}.bi-hearts::before{content:"\f772"}.bi-hospital-fill::before{content:"\f773"}.bi-hospital::before{content:"\f774"}.bi-house-heart-fill::before{content:"\f775"}.bi-house-heart::before{content:"\f776"}.bi-incognito::before{content:"\f777"}.bi-magnet-fill::before{content:"\f778"}.bi-magnet::before{content:"\f779"}.bi-person-heart::before{content:"\f77a"}.bi-person-hearts::before{content:"\f77b"}.bi-phone-flip::before{content:"\f77c"}.bi-plugin::before{content:"\f77d"}.bi-postage-fill::before{content:"\f77e"}.bi-postage-heart-fill::before{content:"\f77f"}.bi-postage-heart::before{content:"\f780"}.bi-postage::before{content:"\f781"}.bi-postcard-fill::before{content:"\f782"}.bi-postcard-heart-fill::before{content:"\f783"}.bi-postcard-heart::before{content:"\f784"}.bi-postcard::before{content:"\f785"}.bi-search-heart-fill::before{content:"\f786"}.bi-search-heart::before{content:"\f787"}.bi-sliders2-vertical::before{content:"\f788"}.bi-sliders2::before{content:"\f789"}.bi-trash3-fill::before{content:"\f78a"}.bi-trash3::before{content:"\f78b"}.bi-valentine::before{content:"\f78c"}.bi-valentine2::before{content:"\f78d"}.bi-wrench-adjustable-circle-fill::before{content:"\f78e"}.bi-wrench-adjustable-circle::before{content:"\f78f"}.bi-wrench-adjustable::before{content:"\f790"}.bi-filetype-json::before{content:"\f791"}.bi-filetype-pptx::before{content:"\f792"}.bi-filetype-xlsx::before{content:"\f793"}.bi-1-circle-fill::before{content:"\f796"}.bi-1-circle::before{content:"\f797"}.bi-1-square-fill::before{content:"\f798"}.bi-1-square::before{content:"\f799"}.bi-2-circle-fill::before{content:"\f79c"}.bi-2-circle::before{content:"\f79d"}.bi-2-square-fill::before{content:"\f79e"}.bi-2-square::before{content:"\f79f"}.bi-3-circle-fill::before{content:"\f7a2"}.bi-3-circle::before{content:"\f7a3"}.bi-3-square-fill::before{content:"\f7a4"}.bi-3-square::before{content:"\f7a5"}.bi-4-circle-fill::before{content:"\f7a8"}.bi-4-circle::before{content:"\f7a9"}.bi-4-square-fill::before{content:"\f7aa"}.bi-4-square::before{content:"\f7ab"}.bi-5-circle-fill::before{content:"\f7ae"}.bi-5-circle::before{content:"\f7af"}.bi-5-square-fill::before{content:"\f7b0"}.bi-5-square::before{content:"\f7b1"}.bi-6-circle-fill::before{content:"\f7b4"}.bi-6-circle::before{content:"\f7b5"}.bi-6-square-fill::before{content:"\f7b6"}.bi-6-square::before{content:"\f7b7"}.bi-7-circle-fill::before{content:"\f7ba"}.bi-7-circle::before{content:"\f7bb"}.bi-7-square-fill::before{content:"\f7bc"}.bi-7-square::before{content:"\f7bd"}.bi-8-circle-fill::before{content:"\f7c0"}.bi-8-circle::before{content:"\f7c1"}.bi-8-square-fill::before{content:"\f7c2"}.bi-8-square::before{content:"\f7c3"}.bi-9-circle-fill::before{content:"\f7c6"}.bi-9-circle::before{content:"\f7c7"}.bi-9-square-fill::before{content:"\f7c8"}.bi-9-square::before{content:"\f7c9"}.bi-airplane-engines-fill::before{content:"\f7ca"}.bi-airplane-engines::before{content:"\f7cb"}.bi-airplane-fill::before{content:"\f7cc"}.bi-airplane::before{content:"\f7cd"}.bi-alexa::before{content:"\f7ce"}.bi-alipay::before{content:"\f7cf"}.bi-android::before{content:"\f7d0"}.bi-android2::before{content:"\f7d1"}.bi-box-fill::before{content:"\f7d2"}.bi-box-seam-fill::before{content:"\f7d3"}.bi-browser-chrome::before{content:"\f7d4"}.bi-browser-edge::before{content:"\f7d5"}.bi-browser-firefox::before{content:"\f7d6"}.bi-browser-safari::before{content:"\f7d7"}.bi-c-circle-fill::before{content:"\f7da"}.bi-c-circle::before{content:"\f7db"}.bi-c-square-fill::before{content:"\f7dc"}.bi-c-square::before{content:"\f7dd"}.bi-capsule-pill::before{content:"\f7de"}.bi-capsule::before{content:"\f7df"}.bi-car-front-fill::before{content:"\f7e0"}.bi-car-front::before{content:"\f7e1"}.bi-cassette-fill::before{content:"\f7e2"}.bi-cassette::before{content:"\f7e3"}.bi-cc-circle-fill::before{content:"\f7e6"}.bi-cc-circle::before{content:"\f7e7"}.bi-cc-square-fill::before{content:"\f7e8"}.bi-cc-square::before{content:"\f7e9"}.bi-cup-hot-fill::before{content:"\f7ea"}.bi-cup-hot::before{content:"\f7eb"}.bi-currency-rupee::before{content:"\f7ec"}.bi-dropbox::before{content:"\f7ed"}.bi-escape::before{content:"\f7ee"}.bi-fast-forward-btn-fill::before{content:"\f7ef"}.bi-fast-forward-btn::before{content:"\f7f0"}.bi-fast-forward-circle-fill::before{content:"\f7f1"}.bi-fast-forward-circle::before{content:"\f7f2"}.bi-fast-forward-fill::before{content:"\f7f3"}.bi-fast-forward::before{content:"\f7f4"}.bi-filetype-sql::before{content:"\f7f5"}.bi-fire::before{content:"\f7f6"}.bi-google-play::before{content:"\f7f7"}.bi-h-circle-fill::before{content:"\f7fa"}.bi-h-circle::before{content:"\f7fb"}.bi-h-square-fill::before{content:"\f7fc"}.bi-h-square::before{content:"\f7fd"}.bi-indent::before{content:"\f7fe"}.bi-lungs-fill::before{content:"\f7ff"}.bi-lungs::before{content:"\f800"}.bi-microsoft-teams::before{content:"\f801"}.bi-p-circle-fill::before{content:"\f804"}.bi-p-circle::before{content:"\f805"}.bi-p-square-fill::before{content:"\f806"}.bi-p-square::before{content:"\f807"}.bi-pass-fill::before{content:"\f808"}.bi-pass::before{content:"\f809"}.bi-prescription::before{content:"\f80a"}.bi-prescription2::before{content:"\f80b"}.bi-r-circle-fill::before{content:"\f80e"}.bi-r-circle::before{content:"\f80f"}.bi-r-square-fill::before{content:"\f810"}.bi-r-square::before{content:"\f811"}.bi-repeat-1::before{content:"\f812"}.bi-repeat::before{content:"\f813"}.bi-rewind-btn-fill::before{content:"\f814"}.bi-rewind-btn::before{content:"\f815"}.bi-rewind-circle-fill::before{content:"\f816"}.bi-rewind-circle::before{content:"\f817"}.bi-rewind-fill::before{content:"\f818"}.bi-rewind::before{content:"\f819"}.bi-train-freight-front-fill::before{content:"\f81a"}.bi-train-freight-front::before{content:"\f81b"}.bi-train-front-fill::before{content:"\f81c"}.bi-train-front::before{content:"\f81d"}.bi-train-lightrail-front-fill::before{content:"\f81e"}.bi-train-lightrail-front::before{content:"\f81f"}.bi-truck-front-fill::before{content:"\f820"}.bi-truck-front::before{content:"\f821"}.bi-ubuntu::before{content:"\f822"}.bi-unindent::before{content:"\f823"}.bi-unity::before{content:"\f824"}.bi-universal-access-circle::before{content:"\f825"}.bi-universal-access::before{content:"\f826"}.bi-virus::before{content:"\f827"}.bi-virus2::before{content:"\f828"}.bi-wechat::before{content:"\f829"}.bi-yelp::before{content:"\f82a"}.bi-sign-stop-fill::before{content:"\f82b"}.bi-sign-stop-lights-fill::before{content:"\f82c"}.bi-sign-stop-lights::before{content:"\f82d"}.bi-sign-stop::before{content:"\f82e"}.bi-sign-turn-left-fill::before{content:"\f82f"}.bi-sign-turn-left::before{content:"\f830"}.bi-sign-turn-right-fill::before{content:"\f831"}.bi-sign-turn-right::before{content:"\f832"}.bi-sign-turn-slight-left-fill::before{content:"\f833"}.bi-sign-turn-slight-left::before{content:"\f834"}.bi-sign-turn-slight-right-fill::before{content:"\f835"}.bi-sign-turn-slight-right::before{content:"\f836"}.bi-sign-yield-fill::before{content:"\f837"}.bi-sign-yield::before{content:"\f838"}.bi-ev-station-fill::before{content:"\f839"}.bi-ev-station::before{content:"\f83a"}.bi-fuel-pump-diesel-fill::before{content:"\f83b"}.bi-fuel-pump-diesel::before{content:"\f83c"}.bi-fuel-pump-fill::before{content:"\f83d"}.bi-fuel-pump::before{content:"\f83e"}.bi-0-circle-fill::before{content:"\f83f"}.bi-0-circle::before{content:"\f840"}.bi-0-square-fill::before{content:"\f841"}.bi-0-square::before{content:"\f842"}.bi-rocket-fill::before{content:"\f843"}.bi-rocket-takeoff-fill::before{content:"\f844"}.bi-rocket-takeoff::before{content:"\f845"}.bi-rocket::before{content:"\f846"}.bi-stripe::before{content:"\f847"}.bi-subscript::before{content:"\f848"}.bi-superscript::before{content:"\f849"}.bi-trello::before{content:"\f84a"}.bi-envelope-at-fill::before{content:"\f84b"}.bi-envelope-at::before{content:"\f84c"}.bi-regex::before{content:"\f84d"}.bi-text-wrap::before{content:"\f84e"}.bi-sign-dead-end-fill::before{content:"\f84f"}.bi-sign-dead-end::before{content:"\f850"}.bi-sign-do-not-enter-fill::before{content:"\f851"}.bi-sign-do-not-enter::before{content:"\f852"}.bi-sign-intersection-fill::before{content:"\f853"}.bi-sign-intersection-side-fill::before{content:"\f854"}.bi-sign-intersection-side::before{content:"\f855"}.bi-sign-intersection-t-fill::before{content:"\f856"}.bi-sign-intersection-t::before{content:"\f857"}.bi-sign-intersection-y-fill::before{content:"\f858"}.bi-sign-intersection-y::before{content:"\f859"}.bi-sign-intersection::before{content:"\f85a"}.bi-sign-merge-left-fill::before{content:"\f85b"}.bi-sign-merge-left::before{content:"\f85c"}.bi-sign-merge-right-fill::before{content:"\f85d"}.bi-sign-merge-right::before{content:"\f85e"}.bi-sign-no-left-turn-fill::before{content:"\f85f"}.bi-sign-no-left-turn::before{content:"\f860"}.bi-sign-no-parking-fill::before{content:"\f861"}.bi-sign-no-parking::before{content:"\f862"}.bi-sign-no-right-turn-fill::before{content:"\f863"}.bi-sign-no-right-turn::before{content:"\f864"}.bi-sign-railroad-fill::before{content:"\f865"}.bi-sign-railroad::before{content:"\f866"}.bi-building-add::before{content:"\f867"}.bi-building-check::before{content:"\f868"}.bi-building-dash::before{content:"\f869"}.bi-building-down::before{content:"\f86a"}.bi-building-exclamation::before{content:"\f86b"}.bi-building-fill-add::before{content:"\f86c"}.bi-building-fill-check::before{content:"\f86d"}.bi-building-fill-dash::before{content:"\f86e"}.bi-building-fill-down::before{content:"\f86f"}.bi-building-fill-exclamation::before{content:"\f870"}.bi-building-fill-gear::before{content:"\f871"}.bi-building-fill-lock::before{content:"\f872"}.bi-building-fill-slash::before{content:"\f873"}.bi-building-fill-up::before{content:"\f874"}.bi-building-fill-x::before{content:"\f875"}.bi-building-fill::before{content:"\f876"}.bi-building-gear::before{content:"\f877"}.bi-building-lock::before{content:"\f878"}.bi-building-slash::before{content:"\f879"}.bi-building-up::before{content:"\f87a"}.bi-building-x::before{content:"\f87b"}.bi-buildings-fill::before{content:"\f87c"}.bi-buildings::before{content:"\f87d"}.bi-bus-front-fill::before{content:"\f87e"}.bi-bus-front::before{content:"\f87f"}.bi-ev-front-fill::before{content:"\f880"}.bi-ev-front::before{content:"\f881"}.bi-globe-americas::before{content:"\f882"}.bi-globe-asia-australia::before{content:"\f883"}.bi-globe-central-south-asia::before{content:"\f884"}.bi-globe-europe-africa::before{content:"\f885"}.bi-house-add-fill::before{content:"\f886"}.bi-house-add::before{content:"\f887"}.bi-house-check-fill::before{content:"\f888"}.bi-house-check::before{content:"\f889"}.bi-house-dash-fill::before{content:"\f88a"}.bi-house-dash::before{content:"\f88b"}.bi-house-down-fill::before{content:"\f88c"}.bi-house-down::before{content:"\f88d"}.bi-house-exclamation-fill::before{content:"\f88e"}.bi-house-exclamation::before{content:"\f88f"}.bi-house-gear-fill::before{content:"\f890"}.bi-house-gear::before{content:"\f891"}.bi-house-lock-fill::before{content:"\f892"}.bi-house-lock::before{content:"\f893"}.bi-house-slash-fill::before{content:"\f894"}.bi-house-slash::before{content:"\f895"}.bi-house-up-fill::before{content:"\f896"}.bi-house-up::before{content:"\f897"}.bi-house-x-fill::before{content:"\f898"}.bi-house-x::before{content:"\f899"}.bi-person-add::before{content:"\f89a"}.bi-person-down::before{content:"\f89b"}.bi-person-exclamation::before{content:"\f89c"}.bi-person-fill-add::before{content:"\f89d"}.bi-person-fill-check::before{content:"\f89e"}.bi-person-fill-dash::before{content:"\f89f"}.bi-person-fill-down::before{content:"\f8a0"}.bi-person-fill-exclamation::before{content:"\f8a1"}.bi-person-fill-gear::before{content:"\f8a2"}.bi-person-fill-lock::before{content:"\f8a3"}.bi-person-fill-slash::before{content:"\f8a4"}.bi-person-fill-up::before{content:"\f8a5"}.bi-person-fill-x::before{content:"\f8a6"}.bi-person-gear::before{content:"\f8a7"}.bi-person-lock::before{content:"\f8a8"}.bi-person-slash::before{content:"\f8a9"}.bi-person-up::before{content:"\f8aa"}.bi-scooter::before{content:"\f8ab"}.bi-taxi-front-fill::before{content:"\f8ac"}.bi-taxi-front::before{content:"\f8ad"}.bi-amd::before{content:"\f8ae"}.bi-database-add::before{content:"\f8af"}.bi-database-check::before{content:"\f8b0"}.bi-database-dash::before{content:"\f8b1"}.bi-database-down::before{content:"\f8b2"}.bi-database-exclamation::before{content:"\f8b3"}.bi-database-fill-add::before{content:"\f8b4"}.bi-database-fill-check::before{content:"\f8b5"}.bi-database-fill-dash::before{content:"\f8b6"}.bi-database-fill-down::before{content:"\f8b7"}.bi-database-fill-exclamation::before{content:"\f8b8"}.bi-database-fill-gear::before{content:"\f8b9"}.bi-database-fill-lock::before{content:"\f8ba"}.bi-database-fill-slash::before{content:"\f8bb"}.bi-database-fill-up::before{content:"\f8bc"}.bi-database-fill-x::before{content:"\f8bd"}.bi-database-fill::before{content:"\f8be"}.bi-database-gear::before{content:"\f8bf"}.bi-database-lock::before{content:"\f8c0"}.bi-database-slash::before{content:"\f8c1"}.bi-database-up::before{content:"\f8c2"}.bi-database-x::before{content:"\f8c3"}.bi-database::before{content:"\f8c4"}.bi-houses-fill::before{content:"\f8c5"}.bi-houses::before{content:"\f8c6"}.bi-nvidia::before{content:"\f8c7"}.bi-person-vcard-fill::before{content:"\f8c8"}.bi-person-vcard::before{content:"\f8c9"}.bi-sina-weibo::before{content:"\f8ca"}.bi-tencent-qq::before{content:"\f8cb"}.bi-wikipedia::before{content:"\f8cc"} \ No newline at end of file diff --git a/app/static/css/bootstrap.min.css b/app/static/css/bootstrap.min.css new file mode 100644 index 0000000000000000000000000000000000000000..a89937ccb889c96d10c79cf2813ae07334178cb5 --- /dev/null +++ b/app/static/css/bootstrap.min.css @@ -0,0 +1,6 @@ +@charset "UTF-8";/*! + * Bootstrap v5.3.1 (https://getbootstrap.com/) + * Copyright 2011-2023 The Bootstrap Authors + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */:root,[data-bs-theme=light]{--bs-blue:#0d6efd;--bs-indigo:#6610f2;--bs-purple:#6f42c1;--bs-pink:#d63384;--bs-red:#dc3545;--bs-orange:#fd7e14;--bs-yellow:#ffc107;--bs-green:#198754;--bs-teal:#20c997;--bs-cyan:#0dcaf0;--bs-black:#000;--bs-white:#fff;--bs-gray:#6c757d;--bs-gray-dark:#343a40;--bs-gray-100:#f8f9fa;--bs-gray-200:#e9ecef;--bs-gray-300:#dee2e6;--bs-gray-400:#ced4da;--bs-gray-500:#adb5bd;--bs-gray-600:#6c757d;--bs-gray-700:#495057;--bs-gray-800:#343a40;--bs-gray-900:#212529;--bs-primary:#0d6efd;--bs-secondary:#6c757d;--bs-success:#198754;--bs-info:#0dcaf0;--bs-warning:#ffc107;--bs-danger:#dc3545;--bs-light:#f8f9fa;--bs-dark:#212529;--bs-primary-rgb:13,110,253;--bs-secondary-rgb:108,117,125;--bs-success-rgb:25,135,84;--bs-info-rgb:13,202,240;--bs-warning-rgb:255,193,7;--bs-danger-rgb:220,53,69;--bs-light-rgb:248,249,250;--bs-dark-rgb:33,37,41;--bs-primary-text-emphasis:#052c65;--bs-secondary-text-emphasis:#2b2f32;--bs-success-text-emphasis:#0a3622;--bs-info-text-emphasis:#055160;--bs-warning-text-emphasis:#664d03;--bs-danger-text-emphasis:#58151c;--bs-light-text-emphasis:#495057;--bs-dark-text-emphasis:#495057;--bs-primary-bg-subtle:#cfe2ff;--bs-secondary-bg-subtle:#e2e3e5;--bs-success-bg-subtle:#d1e7dd;--bs-info-bg-subtle:#cff4fc;--bs-warning-bg-subtle:#fff3cd;--bs-danger-bg-subtle:#f8d7da;--bs-light-bg-subtle:#fcfcfd;--bs-dark-bg-subtle:#ced4da;--bs-primary-border-subtle:#9ec5fe;--bs-secondary-border-subtle:#c4c8cb;--bs-success-border-subtle:#a3cfbb;--bs-info-border-subtle:#9eeaf9;--bs-warning-border-subtle:#ffe69c;--bs-danger-border-subtle:#f1aeb5;--bs-light-border-subtle:#e9ecef;--bs-dark-border-subtle:#adb5bd;--bs-white-rgb:255,255,255;--bs-black-rgb:0,0,0;--bs-font-sans-serif:system-ui,-apple-system,"Segoe UI",Roboto,"Helvetica Neue","Noto Sans","Liberation Sans",Arial,sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--bs-font-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;--bs-gradient:linear-gradient(180deg, rgba(255, 255, 255, 0.15), rgba(255, 255, 255, 0));--bs-body-font-family:var(--bs-font-sans-serif);--bs-body-font-size:1rem;--bs-body-font-weight:400;--bs-body-line-height:1.5;--bs-body-color:#212529;--bs-body-color-rgb:33,37,41;--bs-body-bg:#fff;--bs-body-bg-rgb:255,255,255;--bs-emphasis-color:#000;--bs-emphasis-color-rgb:0,0,0;--bs-secondary-color:rgba(33, 37, 41, 0.75);--bs-secondary-color-rgb:33,37,41;--bs-secondary-bg:#e9ecef;--bs-secondary-bg-rgb:233,236,239;--bs-tertiary-color:rgba(33, 37, 41, 0.5);--bs-tertiary-color-rgb:33,37,41;--bs-tertiary-bg:#f8f9fa;--bs-tertiary-bg-rgb:248,249,250;--bs-heading-color:inherit;--bs-link-color:#0d6efd;--bs-link-color-rgb:13,110,253;--bs-link-decoration:underline;--bs-link-hover-color:#0a58ca;--bs-link-hover-color-rgb:10,88,202;--bs-code-color:#d63384;--bs-highlight-bg:#fff3cd;--bs-border-width:1px;--bs-border-style:solid;--bs-border-color:#dee2e6;--bs-border-color-translucent:rgba(0, 0, 0, 0.175);--bs-border-radius:0.375rem;--bs-border-radius-sm:0.25rem;--bs-border-radius-lg:0.5rem;--bs-border-radius-xl:1rem;--bs-border-radius-xxl:2rem;--bs-border-radius-2xl:var(--bs-border-radius-xxl);--bs-border-radius-pill:50rem;--bs-box-shadow:0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-box-shadow-sm:0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-box-shadow-lg:0 1rem 3rem rgba(0, 0, 0, 0.175);--bs-box-shadow-inset:inset 0 1px 2px rgba(0, 0, 0, 0.075);--bs-focus-ring-width:0.25rem;--bs-focus-ring-opacity:0.25;--bs-focus-ring-color:rgba(13, 110, 253, 0.25);--bs-form-valid-color:#198754;--bs-form-valid-border-color:#198754;--bs-form-invalid-color:#dc3545;--bs-form-invalid-border-color:#dc3545}[data-bs-theme=dark]{color-scheme:dark;--bs-body-color:#dee2e6;--bs-body-color-rgb:222,226,230;--bs-body-bg:#212529;--bs-body-bg-rgb:33,37,41;--bs-emphasis-color:#fff;--bs-emphasis-color-rgb:255,255,255;--bs-secondary-color:rgba(222, 226, 230, 0.75);--bs-secondary-color-rgb:222,226,230;--bs-secondary-bg:#343a40;--bs-secondary-bg-rgb:52,58,64;--bs-tertiary-color:rgba(222, 226, 230, 0.5);--bs-tertiary-color-rgb:222,226,230;--bs-tertiary-bg:#2b3035;--bs-tertiary-bg-rgb:43,48,53;--bs-primary-text-emphasis:#6ea8fe;--bs-secondary-text-emphasis:#a7acb1;--bs-success-text-emphasis:#75b798;--bs-info-text-emphasis:#6edff6;--bs-warning-text-emphasis:#ffda6a;--bs-danger-text-emphasis:#ea868f;--bs-light-text-emphasis:#f8f9fa;--bs-dark-text-emphasis:#dee2e6;--bs-primary-bg-subtle:#031633;--bs-secondary-bg-subtle:#161719;--bs-success-bg-subtle:#051b11;--bs-info-bg-subtle:#032830;--bs-warning-bg-subtle:#332701;--bs-danger-bg-subtle:#2c0b0e;--bs-light-bg-subtle:#343a40;--bs-dark-bg-subtle:#1a1d20;--bs-primary-border-subtle:#084298;--bs-secondary-border-subtle:#41464b;--bs-success-border-subtle:#0f5132;--bs-info-border-subtle:#087990;--bs-warning-border-subtle:#997404;--bs-danger-border-subtle:#842029;--bs-light-border-subtle:#495057;--bs-dark-border-subtle:#343a40;--bs-heading-color:inherit;--bs-link-color:#6ea8fe;--bs-link-hover-color:#8bb9fe;--bs-link-color-rgb:110,168,254;--bs-link-hover-color-rgb:139,185,254;--bs-code-color:#e685b5;--bs-border-color:#495057;--bs-border-color-translucent:rgba(255, 255, 255, 0.15);--bs-form-valid-color:#75b798;--bs-form-valid-border-color:#75b798;--bs-form-invalid-color:#ea868f;--bs-form-invalid-border-color:#ea868f}*,::after,::before{box-sizing:border-box}@media (prefers-reduced-motion:no-preference){:root{scroll-behavior:smooth}}body{margin:0;font-family:var(--bs-body-font-family);font-size:var(--bs-body-font-size);font-weight:var(--bs-body-font-weight);line-height:var(--bs-body-line-height);color:var(--bs-body-color);text-align:var(--bs-body-text-align);background-color:var(--bs-body-bg);-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}hr{margin:1rem 0;color:inherit;border:0;border-top:var(--bs-border-width) solid;opacity:.25}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem;font-weight:500;line-height:1.2;color:var(--bs-heading-color)}.h1,h1{font-size:calc(1.375rem + 1.5vw)}@media (min-width:1200px){.h1,h1{font-size:2.5rem}}.h2,h2{font-size:calc(1.325rem + .9vw)}@media (min-width:1200px){.h2,h2{font-size:2rem}}.h3,h3{font-size:calc(1.3rem + .6vw)}@media (min-width:1200px){.h3,h3{font-size:1.75rem}}.h4,h4{font-size:calc(1.275rem + .3vw)}@media (min-width:1200px){.h4,h4{font-size:1.5rem}}.h5,h5{font-size:1.25rem}.h6,h6{font-size:1rem}p{margin-top:0;margin-bottom:1rem}abbr[title]{-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}ol,ul{padding-left:2rem}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}.small,small{font-size:.875em}.mark,mark{padding:.1875em;background-color:var(--bs-highlight-bg)}sub,sup{position:relative;font-size:.75em;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:rgba(var(--bs-link-color-rgb),var(--bs-link-opacity,1));text-decoration:underline}a:hover{--bs-link-color-rgb:var(--bs-link-hover-color-rgb)}a:not([href]):not([class]),a:not([href]):not([class]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:var(--bs-font-monospace);font-size:1em}pre{display:block;margin-top:0;margin-bottom:1rem;overflow:auto;font-size:.875em}pre code{font-size:inherit;color:inherit;word-break:normal}code{font-size:.875em;color:var(--bs-code-color);word-wrap:break-word}a>code{color:inherit}kbd{padding:.1875rem .375rem;font-size:.875em;color:var(--bs-body-bg);background-color:var(--bs-body-color);border-radius:.25rem}kbd kbd{padding:0;font-size:1em}figure{margin:0 0 1rem}img,svg{vertical-align:middle}table{caption-side:bottom;border-collapse:collapse}caption{padding-top:.5rem;padding-bottom:.5rem;color:var(--bs-secondary-color);text-align:left}th{text-align:inherit;text-align:-webkit-match-parent}tbody,td,tfoot,th,thead,tr{border-color:inherit;border-style:solid;border-width:0}label{display:inline-block}button{border-radius:0}button:focus:not(:focus-visible){outline:0}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}select:disabled{opacity:1}[list]:not([type=date]):not([type=datetime-local]):not([type=month]):not([type=week]):not([type=time])::-webkit-calendar-picker-indicator{display:none!important}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}::-moz-focus-inner{padding:0;border-style:none}textarea{resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{float:left;width:100%;padding:0;margin-bottom:.5rem;font-size:calc(1.275rem + .3vw);line-height:inherit}@media (min-width:1200px){legend{font-size:1.5rem}}legend+*{clear:left}::-webkit-datetime-edit-day-field,::-webkit-datetime-edit-fields-wrapper,::-webkit-datetime-edit-hour-field,::-webkit-datetime-edit-minute,::-webkit-datetime-edit-month-field,::-webkit-datetime-edit-text,::-webkit-datetime-edit-year-field{padding:0}::-webkit-inner-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}::-webkit-search-decoration{-webkit-appearance:none}::-webkit-color-swatch-wrapper{padding:0}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}::file-selector-button{font:inherit;-webkit-appearance:button}output{display:inline-block}iframe{border:0}summary{display:list-item;cursor:pointer}progress{vertical-align:baseline}[hidden]{display:none!important}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:calc(1.625rem + 4.5vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-1{font-size:5rem}}.display-2{font-size:calc(1.575rem + 3.9vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-2{font-size:4.5rem}}.display-3{font-size:calc(1.525rem + 3.3vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-3{font-size:4rem}}.display-4{font-size:calc(1.475rem + 2.7vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-4{font-size:3.5rem}}.display-5{font-size:calc(1.425rem + 2.1vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-5{font-size:3rem}}.display-6{font-size:calc(1.375rem + 1.5vw);font-weight:300;line-height:1.2}@media (min-width:1200px){.display-6{font-size:2.5rem}}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:.875em;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote>:last-child{margin-bottom:0}.blockquote-footer{margin-top:-1rem;margin-bottom:1rem;font-size:.875em;color:#6c757d}.blockquote-footer::before{content:"— "}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:var(--bs-body-bg);border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius);max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:.875em;color:var(--bs-secondary-color)}.container,.container-fluid,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{--bs-gutter-x:1.5rem;--bs-gutter-y:0;width:100%;padding-right:calc(var(--bs-gutter-x) * .5);padding-left:calc(var(--bs-gutter-x) * .5);margin-right:auto;margin-left:auto}@media (min-width:576px){.container,.container-sm{max-width:540px}}@media (min-width:768px){.container,.container-md,.container-sm{max-width:720px}}@media (min-width:992px){.container,.container-lg,.container-md,.container-sm{max-width:960px}}@media (min-width:1200px){.container,.container-lg,.container-md,.container-sm,.container-xl{max-width:1140px}}@media (min-width:1400px){.container,.container-lg,.container-md,.container-sm,.container-xl,.container-xxl{max-width:1320px}}:root{--bs-breakpoint-xs:0;--bs-breakpoint-sm:576px;--bs-breakpoint-md:768px;--bs-breakpoint-lg:992px;--bs-breakpoint-xl:1200px;--bs-breakpoint-xxl:1400px}.row{--bs-gutter-x:1.5rem;--bs-gutter-y:0;display:flex;flex-wrap:wrap;margin-top:calc(-1 * var(--bs-gutter-y));margin-right:calc(-.5 * var(--bs-gutter-x));margin-left:calc(-.5 * var(--bs-gutter-x))}.row>*{flex-shrink:0;width:100%;max-width:100%;padding-right:calc(var(--bs-gutter-x) * .5);padding-left:calc(var(--bs-gutter-x) * .5);margin-top:var(--bs-gutter-y)}.col{flex:1 0 0%}.row-cols-auto>*{flex:0 0 auto;width:auto}.row-cols-1>*{flex:0 0 auto;width:100%}.row-cols-2>*{flex:0 0 auto;width:50%}.row-cols-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-4>*{flex:0 0 auto;width:25%}.row-cols-5>*{flex:0 0 auto;width:20%}.row-cols-6>*{flex:0 0 auto;width:16.6666666667%}.col-auto{flex:0 0 auto;width:auto}.col-1{flex:0 0 auto;width:8.33333333%}.col-2{flex:0 0 auto;width:16.66666667%}.col-3{flex:0 0 auto;width:25%}.col-4{flex:0 0 auto;width:33.33333333%}.col-5{flex:0 0 auto;width:41.66666667%}.col-6{flex:0 0 auto;width:50%}.col-7{flex:0 0 auto;width:58.33333333%}.col-8{flex:0 0 auto;width:66.66666667%}.col-9{flex:0 0 auto;width:75%}.col-10{flex:0 0 auto;width:83.33333333%}.col-11{flex:0 0 auto;width:91.66666667%}.col-12{flex:0 0 auto;width:100%}.offset-1{margin-left:8.33333333%}.offset-2{margin-left:16.66666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.33333333%}.offset-5{margin-left:41.66666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.33333333%}.offset-8{margin-left:66.66666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.33333333%}.offset-11{margin-left:91.66666667%}.g-0,.gx-0{--bs-gutter-x:0}.g-0,.gy-0{--bs-gutter-y:0}.g-1,.gx-1{--bs-gutter-x:0.25rem}.g-1,.gy-1{--bs-gutter-y:0.25rem}.g-2,.gx-2{--bs-gutter-x:0.5rem}.g-2,.gy-2{--bs-gutter-y:0.5rem}.g-3,.gx-3{--bs-gutter-x:1rem}.g-3,.gy-3{--bs-gutter-y:1rem}.g-4,.gx-4{--bs-gutter-x:1.5rem}.g-4,.gy-4{--bs-gutter-y:1.5rem}.g-5,.gx-5{--bs-gutter-x:3rem}.g-5,.gy-5{--bs-gutter-y:3rem}@media (min-width:576px){.col-sm{flex:1 0 0%}.row-cols-sm-auto>*{flex:0 0 auto;width:auto}.row-cols-sm-1>*{flex:0 0 auto;width:100%}.row-cols-sm-2>*{flex:0 0 auto;width:50%}.row-cols-sm-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-sm-4>*{flex:0 0 auto;width:25%}.row-cols-sm-5>*{flex:0 0 auto;width:20%}.row-cols-sm-6>*{flex:0 0 auto;width:16.6666666667%}.col-sm-auto{flex:0 0 auto;width:auto}.col-sm-1{flex:0 0 auto;width:8.33333333%}.col-sm-2{flex:0 0 auto;width:16.66666667%}.col-sm-3{flex:0 0 auto;width:25%}.col-sm-4{flex:0 0 auto;width:33.33333333%}.col-sm-5{flex:0 0 auto;width:41.66666667%}.col-sm-6{flex:0 0 auto;width:50%}.col-sm-7{flex:0 0 auto;width:58.33333333%}.col-sm-8{flex:0 0 auto;width:66.66666667%}.col-sm-9{flex:0 0 auto;width:75%}.col-sm-10{flex:0 0 auto;width:83.33333333%}.col-sm-11{flex:0 0 auto;width:91.66666667%}.col-sm-12{flex:0 0 auto;width:100%}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.33333333%}.offset-sm-2{margin-left:16.66666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.33333333%}.offset-sm-5{margin-left:41.66666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.33333333%}.offset-sm-8{margin-left:66.66666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.33333333%}.offset-sm-11{margin-left:91.66666667%}.g-sm-0,.gx-sm-0{--bs-gutter-x:0}.g-sm-0,.gy-sm-0{--bs-gutter-y:0}.g-sm-1,.gx-sm-1{--bs-gutter-x:0.25rem}.g-sm-1,.gy-sm-1{--bs-gutter-y:0.25rem}.g-sm-2,.gx-sm-2{--bs-gutter-x:0.5rem}.g-sm-2,.gy-sm-2{--bs-gutter-y:0.5rem}.g-sm-3,.gx-sm-3{--bs-gutter-x:1rem}.g-sm-3,.gy-sm-3{--bs-gutter-y:1rem}.g-sm-4,.gx-sm-4{--bs-gutter-x:1.5rem}.g-sm-4,.gy-sm-4{--bs-gutter-y:1.5rem}.g-sm-5,.gx-sm-5{--bs-gutter-x:3rem}.g-sm-5,.gy-sm-5{--bs-gutter-y:3rem}}@media (min-width:768px){.col-md{flex:1 0 0%}.row-cols-md-auto>*{flex:0 0 auto;width:auto}.row-cols-md-1>*{flex:0 0 auto;width:100%}.row-cols-md-2>*{flex:0 0 auto;width:50%}.row-cols-md-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-md-4>*{flex:0 0 auto;width:25%}.row-cols-md-5>*{flex:0 0 auto;width:20%}.row-cols-md-6>*{flex:0 0 auto;width:16.6666666667%}.col-md-auto{flex:0 0 auto;width:auto}.col-md-1{flex:0 0 auto;width:8.33333333%}.col-md-2{flex:0 0 auto;width:16.66666667%}.col-md-3{flex:0 0 auto;width:25%}.col-md-4{flex:0 0 auto;width:33.33333333%}.col-md-5{flex:0 0 auto;width:41.66666667%}.col-md-6{flex:0 0 auto;width:50%}.col-md-7{flex:0 0 auto;width:58.33333333%}.col-md-8{flex:0 0 auto;width:66.66666667%}.col-md-9{flex:0 0 auto;width:75%}.col-md-10{flex:0 0 auto;width:83.33333333%}.col-md-11{flex:0 0 auto;width:91.66666667%}.col-md-12{flex:0 0 auto;width:100%}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.33333333%}.offset-md-2{margin-left:16.66666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.33333333%}.offset-md-5{margin-left:41.66666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.33333333%}.offset-md-8{margin-left:66.66666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.33333333%}.offset-md-11{margin-left:91.66666667%}.g-md-0,.gx-md-0{--bs-gutter-x:0}.g-md-0,.gy-md-0{--bs-gutter-y:0}.g-md-1,.gx-md-1{--bs-gutter-x:0.25rem}.g-md-1,.gy-md-1{--bs-gutter-y:0.25rem}.g-md-2,.gx-md-2{--bs-gutter-x:0.5rem}.g-md-2,.gy-md-2{--bs-gutter-y:0.5rem}.g-md-3,.gx-md-3{--bs-gutter-x:1rem}.g-md-3,.gy-md-3{--bs-gutter-y:1rem}.g-md-4,.gx-md-4{--bs-gutter-x:1.5rem}.g-md-4,.gy-md-4{--bs-gutter-y:1.5rem}.g-md-5,.gx-md-5{--bs-gutter-x:3rem}.g-md-5,.gy-md-5{--bs-gutter-y:3rem}}@media (min-width:992px){.col-lg{flex:1 0 0%}.row-cols-lg-auto>*{flex:0 0 auto;width:auto}.row-cols-lg-1>*{flex:0 0 auto;width:100%}.row-cols-lg-2>*{flex:0 0 auto;width:50%}.row-cols-lg-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-lg-4>*{flex:0 0 auto;width:25%}.row-cols-lg-5>*{flex:0 0 auto;width:20%}.row-cols-lg-6>*{flex:0 0 auto;width:16.6666666667%}.col-lg-auto{flex:0 0 auto;width:auto}.col-lg-1{flex:0 0 auto;width:8.33333333%}.col-lg-2{flex:0 0 auto;width:16.66666667%}.col-lg-3{flex:0 0 auto;width:25%}.col-lg-4{flex:0 0 auto;width:33.33333333%}.col-lg-5{flex:0 0 auto;width:41.66666667%}.col-lg-6{flex:0 0 auto;width:50%}.col-lg-7{flex:0 0 auto;width:58.33333333%}.col-lg-8{flex:0 0 auto;width:66.66666667%}.col-lg-9{flex:0 0 auto;width:75%}.col-lg-10{flex:0 0 auto;width:83.33333333%}.col-lg-11{flex:0 0 auto;width:91.66666667%}.col-lg-12{flex:0 0 auto;width:100%}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.33333333%}.offset-lg-2{margin-left:16.66666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.33333333%}.offset-lg-5{margin-left:41.66666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.33333333%}.offset-lg-8{margin-left:66.66666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.33333333%}.offset-lg-11{margin-left:91.66666667%}.g-lg-0,.gx-lg-0{--bs-gutter-x:0}.g-lg-0,.gy-lg-0{--bs-gutter-y:0}.g-lg-1,.gx-lg-1{--bs-gutter-x:0.25rem}.g-lg-1,.gy-lg-1{--bs-gutter-y:0.25rem}.g-lg-2,.gx-lg-2{--bs-gutter-x:0.5rem}.g-lg-2,.gy-lg-2{--bs-gutter-y:0.5rem}.g-lg-3,.gx-lg-3{--bs-gutter-x:1rem}.g-lg-3,.gy-lg-3{--bs-gutter-y:1rem}.g-lg-4,.gx-lg-4{--bs-gutter-x:1.5rem}.g-lg-4,.gy-lg-4{--bs-gutter-y:1.5rem}.g-lg-5,.gx-lg-5{--bs-gutter-x:3rem}.g-lg-5,.gy-lg-5{--bs-gutter-y:3rem}}@media (min-width:1200px){.col-xl{flex:1 0 0%}.row-cols-xl-auto>*{flex:0 0 auto;width:auto}.row-cols-xl-1>*{flex:0 0 auto;width:100%}.row-cols-xl-2>*{flex:0 0 auto;width:50%}.row-cols-xl-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-xl-4>*{flex:0 0 auto;width:25%}.row-cols-xl-5>*{flex:0 0 auto;width:20%}.row-cols-xl-6>*{flex:0 0 auto;width:16.6666666667%}.col-xl-auto{flex:0 0 auto;width:auto}.col-xl-1{flex:0 0 auto;width:8.33333333%}.col-xl-2{flex:0 0 auto;width:16.66666667%}.col-xl-3{flex:0 0 auto;width:25%}.col-xl-4{flex:0 0 auto;width:33.33333333%}.col-xl-5{flex:0 0 auto;width:41.66666667%}.col-xl-6{flex:0 0 auto;width:50%}.col-xl-7{flex:0 0 auto;width:58.33333333%}.col-xl-8{flex:0 0 auto;width:66.66666667%}.col-xl-9{flex:0 0 auto;width:75%}.col-xl-10{flex:0 0 auto;width:83.33333333%}.col-xl-11{flex:0 0 auto;width:91.66666667%}.col-xl-12{flex:0 0 auto;width:100%}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.33333333%}.offset-xl-2{margin-left:16.66666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.33333333%}.offset-xl-5{margin-left:41.66666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.33333333%}.offset-xl-8{margin-left:66.66666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.33333333%}.offset-xl-11{margin-left:91.66666667%}.g-xl-0,.gx-xl-0{--bs-gutter-x:0}.g-xl-0,.gy-xl-0{--bs-gutter-y:0}.g-xl-1,.gx-xl-1{--bs-gutter-x:0.25rem}.g-xl-1,.gy-xl-1{--bs-gutter-y:0.25rem}.g-xl-2,.gx-xl-2{--bs-gutter-x:0.5rem}.g-xl-2,.gy-xl-2{--bs-gutter-y:0.5rem}.g-xl-3,.gx-xl-3{--bs-gutter-x:1rem}.g-xl-3,.gy-xl-3{--bs-gutter-y:1rem}.g-xl-4,.gx-xl-4{--bs-gutter-x:1.5rem}.g-xl-4,.gy-xl-4{--bs-gutter-y:1.5rem}.g-xl-5,.gx-xl-5{--bs-gutter-x:3rem}.g-xl-5,.gy-xl-5{--bs-gutter-y:3rem}}@media (min-width:1400px){.col-xxl{flex:1 0 0%}.row-cols-xxl-auto>*{flex:0 0 auto;width:auto}.row-cols-xxl-1>*{flex:0 0 auto;width:100%}.row-cols-xxl-2>*{flex:0 0 auto;width:50%}.row-cols-xxl-3>*{flex:0 0 auto;width:33.3333333333%}.row-cols-xxl-4>*{flex:0 0 auto;width:25%}.row-cols-xxl-5>*{flex:0 0 auto;width:20%}.row-cols-xxl-6>*{flex:0 0 auto;width:16.6666666667%}.col-xxl-auto{flex:0 0 auto;width:auto}.col-xxl-1{flex:0 0 auto;width:8.33333333%}.col-xxl-2{flex:0 0 auto;width:16.66666667%}.col-xxl-3{flex:0 0 auto;width:25%}.col-xxl-4{flex:0 0 auto;width:33.33333333%}.col-xxl-5{flex:0 0 auto;width:41.66666667%}.col-xxl-6{flex:0 0 auto;width:50%}.col-xxl-7{flex:0 0 auto;width:58.33333333%}.col-xxl-8{flex:0 0 auto;width:66.66666667%}.col-xxl-9{flex:0 0 auto;width:75%}.col-xxl-10{flex:0 0 auto;width:83.33333333%}.col-xxl-11{flex:0 0 auto;width:91.66666667%}.col-xxl-12{flex:0 0 auto;width:100%}.offset-xxl-0{margin-left:0}.offset-xxl-1{margin-left:8.33333333%}.offset-xxl-2{margin-left:16.66666667%}.offset-xxl-3{margin-left:25%}.offset-xxl-4{margin-left:33.33333333%}.offset-xxl-5{margin-left:41.66666667%}.offset-xxl-6{margin-left:50%}.offset-xxl-7{margin-left:58.33333333%}.offset-xxl-8{margin-left:66.66666667%}.offset-xxl-9{margin-left:75%}.offset-xxl-10{margin-left:83.33333333%}.offset-xxl-11{margin-left:91.66666667%}.g-xxl-0,.gx-xxl-0{--bs-gutter-x:0}.g-xxl-0,.gy-xxl-0{--bs-gutter-y:0}.g-xxl-1,.gx-xxl-1{--bs-gutter-x:0.25rem}.g-xxl-1,.gy-xxl-1{--bs-gutter-y:0.25rem}.g-xxl-2,.gx-xxl-2{--bs-gutter-x:0.5rem}.g-xxl-2,.gy-xxl-2{--bs-gutter-y:0.5rem}.g-xxl-3,.gx-xxl-3{--bs-gutter-x:1rem}.g-xxl-3,.gy-xxl-3{--bs-gutter-y:1rem}.g-xxl-4,.gx-xxl-4{--bs-gutter-x:1.5rem}.g-xxl-4,.gy-xxl-4{--bs-gutter-y:1.5rem}.g-xxl-5,.gx-xxl-5{--bs-gutter-x:3rem}.g-xxl-5,.gy-xxl-5{--bs-gutter-y:3rem}}.table{--bs-table-color-type:initial;--bs-table-bg-type:initial;--bs-table-color-state:initial;--bs-table-bg-state:initial;--bs-table-color:var(--bs-body-color);--bs-table-bg:var(--bs-body-bg);--bs-table-border-color:var(--bs-border-color);--bs-table-accent-bg:transparent;--bs-table-striped-color:var(--bs-body-color);--bs-table-striped-bg:rgba(0, 0, 0, 0.05);--bs-table-active-color:var(--bs-body-color);--bs-table-active-bg:rgba(0, 0, 0, 0.1);--bs-table-hover-color:var(--bs-body-color);--bs-table-hover-bg:rgba(0, 0, 0, 0.075);width:100%;margin-bottom:1rem;vertical-align:top;border-color:var(--bs-table-border-color)}.table>:not(caption)>*>*{padding:.5rem .5rem;color:var(--bs-table-color-state,var(--bs-table-color-type,var(--bs-table-color)));background-color:var(--bs-table-bg);border-bottom-width:var(--bs-border-width);box-shadow:inset 0 0 0 9999px var(--bs-table-bg-state,var(--bs-table-bg-type,var(--bs-table-accent-bg)))}.table>tbody{vertical-align:inherit}.table>thead{vertical-align:bottom}.table-group-divider{border-top:calc(var(--bs-border-width) * 2) solid currentcolor}.caption-top{caption-side:top}.table-sm>:not(caption)>*>*{padding:.25rem .25rem}.table-bordered>:not(caption)>*{border-width:var(--bs-border-width) 0}.table-bordered>:not(caption)>*>*{border-width:0 var(--bs-border-width)}.table-borderless>:not(caption)>*>*{border-bottom-width:0}.table-borderless>:not(:first-child){border-top-width:0}.table-striped>tbody>tr:nth-of-type(odd)>*{--bs-table-color-type:var(--bs-table-striped-color);--bs-table-bg-type:var(--bs-table-striped-bg)}.table-striped-columns>:not(caption)>tr>:nth-child(2n){--bs-table-color-type:var(--bs-table-striped-color);--bs-table-bg-type:var(--bs-table-striped-bg)}.table-active{--bs-table-color-state:var(--bs-table-active-color);--bs-table-bg-state:var(--bs-table-active-bg)}.table-hover>tbody>tr:hover>*{--bs-table-color-state:var(--bs-table-hover-color);--bs-table-bg-state:var(--bs-table-hover-bg)}.table-primary{--bs-table-color:#000;--bs-table-bg:#cfe2ff;--bs-table-border-color:#bacbe6;--bs-table-striped-bg:#c5d7f2;--bs-table-striped-color:#000;--bs-table-active-bg:#bacbe6;--bs-table-active-color:#000;--bs-table-hover-bg:#bfd1ec;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-secondary{--bs-table-color:#000;--bs-table-bg:#e2e3e5;--bs-table-border-color:#cbccce;--bs-table-striped-bg:#d7d8da;--bs-table-striped-color:#000;--bs-table-active-bg:#cbccce;--bs-table-active-color:#000;--bs-table-hover-bg:#d1d2d4;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-success{--bs-table-color:#000;--bs-table-bg:#d1e7dd;--bs-table-border-color:#bcd0c7;--bs-table-striped-bg:#c7dbd2;--bs-table-striped-color:#000;--bs-table-active-bg:#bcd0c7;--bs-table-active-color:#000;--bs-table-hover-bg:#c1d6cc;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-info{--bs-table-color:#000;--bs-table-bg:#cff4fc;--bs-table-border-color:#badce3;--bs-table-striped-bg:#c5e8ef;--bs-table-striped-color:#000;--bs-table-active-bg:#badce3;--bs-table-active-color:#000;--bs-table-hover-bg:#bfe2e9;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-warning{--bs-table-color:#000;--bs-table-bg:#fff3cd;--bs-table-border-color:#e6dbb9;--bs-table-striped-bg:#f2e7c3;--bs-table-striped-color:#000;--bs-table-active-bg:#e6dbb9;--bs-table-active-color:#000;--bs-table-hover-bg:#ece1be;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-danger{--bs-table-color:#000;--bs-table-bg:#f8d7da;--bs-table-border-color:#dfc2c4;--bs-table-striped-bg:#eccccf;--bs-table-striped-color:#000;--bs-table-active-bg:#dfc2c4;--bs-table-active-color:#000;--bs-table-hover-bg:#e5c7ca;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-light{--bs-table-color:#000;--bs-table-bg:#f8f9fa;--bs-table-border-color:#dfe0e1;--bs-table-striped-bg:#ecedee;--bs-table-striped-color:#000;--bs-table-active-bg:#dfe0e1;--bs-table-active-color:#000;--bs-table-hover-bg:#e5e6e7;--bs-table-hover-color:#000;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-dark{--bs-table-color:#fff;--bs-table-bg:#212529;--bs-table-border-color:#373b3e;--bs-table-striped-bg:#2c3034;--bs-table-striped-color:#fff;--bs-table-active-bg:#373b3e;--bs-table-active-color:#fff;--bs-table-hover-bg:#323539;--bs-table-hover-color:#fff;color:var(--bs-table-color);border-color:var(--bs-table-border-color)}.table-responsive{overflow-x:auto;-webkit-overflow-scrolling:touch}@media (max-width:575.98px){.table-responsive-sm{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:767.98px){.table-responsive-md{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:991.98px){.table-responsive-lg{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:1199.98px){.table-responsive-xl{overflow-x:auto;-webkit-overflow-scrolling:touch}}@media (max-width:1399.98px){.table-responsive-xxl{overflow-x:auto;-webkit-overflow-scrolling:touch}}.form-label{margin-bottom:.5rem}.col-form-label{padding-top:calc(.375rem + var(--bs-border-width));padding-bottom:calc(.375rem + var(--bs-border-width));margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + var(--bs-border-width));padding-bottom:calc(.5rem + var(--bs-border-width));font-size:1.25rem}.col-form-label-sm{padding-top:calc(.25rem + var(--bs-border-width));padding-bottom:calc(.25rem + var(--bs-border-width));font-size:.875rem}.form-text{margin-top:.25rem;font-size:.875em;color:var(--bs-secondary-color)}.form-control{display:block;width:100%;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:var(--bs-body-color);-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:var(--bs-body-bg);background-clip:padding-box;border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius);transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control{transition:none}}.form-control[type=file]{overflow:hidden}.form-control[type=file]:not(:disabled):not([readonly]){cursor:pointer}.form-control:focus{color:var(--bs-body-color);background-color:var(--bs-body-bg);border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-control::-webkit-date-and-time-value{min-width:85px;height:1.5em;margin:0}.form-control::-webkit-datetime-edit{display:block;padding:0}.form-control::-moz-placeholder{color:var(--bs-secondary-color);opacity:1}.form-control::placeholder{color:var(--bs-secondary-color);opacity:1}.form-control:disabled{background-color:var(--bs-secondary-bg);opacity:1}.form-control::-webkit-file-upload-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:var(--bs-body-color);background-color:var(--bs-tertiary-bg);pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:var(--bs-border-width);border-radius:0;-webkit-transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}.form-control::file-selector-button{padding:.375rem .75rem;margin:-.375rem -.75rem;-webkit-margin-end:.75rem;margin-inline-end:.75rem;color:var(--bs-body-color);background-color:var(--bs-tertiary-bg);pointer-events:none;border-color:inherit;border-style:solid;border-width:0;border-inline-end-width:var(--bs-border-width);border-radius:0;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control::-webkit-file-upload-button{-webkit-transition:none;transition:none}.form-control::file-selector-button{transition:none}}.form-control:hover:not(:disabled):not([readonly])::-webkit-file-upload-button{background-color:var(--bs-secondary-bg)}.form-control:hover:not(:disabled):not([readonly])::file-selector-button{background-color:var(--bs-secondary-bg)}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;line-height:1.5;color:var(--bs-body-color);background-color:transparent;border:solid transparent;border-width:var(--bs-border-width) 0}.form-control-plaintext:focus{outline:0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm{padding-right:0;padding-left:0}.form-control-sm{min-height:calc(1.5em + .5rem + calc(var(--bs-border-width) * 2));padding:.25rem .5rem;font-size:.875rem;border-radius:var(--bs-border-radius-sm)}.form-control-sm::-webkit-file-upload-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-sm::file-selector-button{padding:.25rem .5rem;margin:-.25rem -.5rem;-webkit-margin-end:.5rem;margin-inline-end:.5rem}.form-control-lg{min-height:calc(1.5em + 1rem + calc(var(--bs-border-width) * 2));padding:.5rem 1rem;font-size:1.25rem;border-radius:var(--bs-border-radius-lg)}.form-control-lg::-webkit-file-upload-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}.form-control-lg::file-selector-button{padding:.5rem 1rem;margin:-.5rem -1rem;-webkit-margin-end:1rem;margin-inline-end:1rem}textarea.form-control{min-height:calc(1.5em + .75rem + calc(var(--bs-border-width) * 2))}textarea.form-control-sm{min-height:calc(1.5em + .5rem + calc(var(--bs-border-width) * 2))}textarea.form-control-lg{min-height:calc(1.5em + 1rem + calc(var(--bs-border-width) * 2))}.form-control-color{width:3rem;height:calc(1.5em + .75rem + calc(var(--bs-border-width) * 2));padding:.375rem}.form-control-color:not(:disabled):not([readonly]){cursor:pointer}.form-control-color::-moz-color-swatch{border:0!important;border-radius:var(--bs-border-radius)}.form-control-color::-webkit-color-swatch{border:0!important;border-radius:var(--bs-border-radius)}.form-control-color.form-control-sm{height:calc(1.5em + .5rem + calc(var(--bs-border-width) * 2))}.form-control-color.form-control-lg{height:calc(1.5em + 1rem + calc(var(--bs-border-width) * 2))}.form-select{--bs-form-select-bg-img:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23343a40' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e");display:block;width:100%;padding:.375rem 2.25rem .375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:var(--bs-body-color);-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:var(--bs-body-bg);background-image:var(--bs-form-select-bg-img),var(--bs-form-select-bg-icon,none);background-repeat:no-repeat;background-position:right .75rem center;background-size:16px 12px;border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius);transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-select{transition:none}}.form-select:focus{border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-select[multiple],.form-select[size]:not([size="1"]){padding-right:.75rem;background-image:none}.form-select:disabled{background-color:var(--bs-secondary-bg)}.form-select:-moz-focusring{color:transparent;text-shadow:0 0 0 var(--bs-body-color)}.form-select-sm{padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem;border-radius:var(--bs-border-radius-sm)}.form-select-lg{padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem;border-radius:var(--bs-border-radius-lg)}[data-bs-theme=dark] .form-select{--bs-form-select-bg-img:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16'%3e%3cpath fill='none' stroke='%23dee2e6' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' d='m2 5 6 6 6-6'/%3e%3c/svg%3e")}.form-check{display:block;min-height:1.5rem;padding-left:1.5em;margin-bottom:.125rem}.form-check .form-check-input{float:left;margin-left:-1.5em}.form-check-reverse{padding-right:1.5em;padding-left:0;text-align:right}.form-check-reverse .form-check-input{float:right;margin-right:-1.5em;margin-left:0}.form-check-input{--bs-form-check-bg:var(--bs-body-bg);width:1em;height:1em;margin-top:.25em;vertical-align:top;-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:var(--bs-form-check-bg);background-image:var(--bs-form-check-bg-image);background-repeat:no-repeat;background-position:center;background-size:contain;border:var(--bs-border-width) solid var(--bs-border-color);-webkit-print-color-adjust:exact;color-adjust:exact;print-color-adjust:exact}.form-check-input[type=checkbox]{border-radius:.25em}.form-check-input[type=radio]{border-radius:50%}.form-check-input:active{filter:brightness(90%)}.form-check-input:focus{border-color:#86b7fe;outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.form-check-input:checked{background-color:#0d6efd;border-color:#0d6efd}.form-check-input:checked[type=checkbox]{--bs-form-check-bg-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='m6 10 3 3 6-6'/%3e%3c/svg%3e")}.form-check-input:checked[type=radio]{--bs-form-check-bg-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='2' fill='%23fff'/%3e%3c/svg%3e")}.form-check-input[type=checkbox]:indeterminate{background-color:#0d6efd;border-color:#0d6efd;--bs-form-check-bg-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20'%3e%3cpath fill='none' stroke='%23fff' stroke-linecap='round' stroke-linejoin='round' stroke-width='3' d='M6 10h8'/%3e%3c/svg%3e")}.form-check-input:disabled{pointer-events:none;filter:none;opacity:.5}.form-check-input:disabled~.form-check-label,.form-check-input[disabled]~.form-check-label{cursor:default;opacity:.5}.form-switch{padding-left:2.5em}.form-switch .form-check-input{--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba%280, 0, 0, 0.25%29'/%3e%3c/svg%3e");width:2em;margin-left:-2.5em;background-image:var(--bs-form-switch-bg);background-position:left center;border-radius:2em;transition:background-position .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-switch .form-check-input{transition:none}}.form-switch .form-check-input:focus{--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%2386b7fe'/%3e%3c/svg%3e")}.form-switch .form-check-input:checked{background-position:right center;--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e")}.form-switch.form-check-reverse{padding-right:2.5em;padding-left:0}.form-switch.form-check-reverse .form-check-input{margin-right:-2.5em;margin-left:0}.form-check-inline{display:inline-block;margin-right:1rem}.btn-check{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.btn-check:disabled+.btn,.btn-check[disabled]+.btn{pointer-events:none;filter:none;opacity:.65}[data-bs-theme=dark] .form-switch .form-check-input:not(:checked):not(:focus){--bs-form-switch-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='rgba%28255, 255, 255, 0.25%29'/%3e%3c/svg%3e")}.form-range{width:100%;height:1.5rem;padding:0;-webkit-appearance:none;-moz-appearance:none;appearance:none;background-color:transparent}.form-range:focus{outline:0}.form-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(13,110,253,.25)}.form-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .25rem rgba(13,110,253,.25)}.form-range::-moz-focus-outer{border:0}.form-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;-webkit-appearance:none;appearance:none;background-color:#0d6efd;border:0;border-radius:1rem;-webkit-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-range::-webkit-slider-thumb{-webkit-transition:none;transition:none}}.form-range::-webkit-slider-thumb:active{background-color:#b6d4fe}.form-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:var(--bs-tertiary-bg);border-color:transparent;border-radius:1rem}.form-range::-moz-range-thumb{width:1rem;height:1rem;-moz-appearance:none;appearance:none;background-color:#0d6efd;border:0;border-radius:1rem;-moz-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-range::-moz-range-thumb{-moz-transition:none;transition:none}}.form-range::-moz-range-thumb:active{background-color:#b6d4fe}.form-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:var(--bs-tertiary-bg);border-color:transparent;border-radius:1rem}.form-range:disabled{pointer-events:none}.form-range:disabled::-webkit-slider-thumb{background-color:var(--bs-secondary-color)}.form-range:disabled::-moz-range-thumb{background-color:var(--bs-secondary-color)}.form-floating{position:relative}.form-floating>.form-control,.form-floating>.form-control-plaintext,.form-floating>.form-select{height:calc(3.5rem + calc(var(--bs-border-width) * 2));min-height:calc(3.5rem + calc(var(--bs-border-width) * 2));line-height:1.25}.form-floating>label{position:absolute;top:0;left:0;z-index:2;height:100%;padding:1rem .75rem;overflow:hidden;text-align:start;text-overflow:ellipsis;white-space:nowrap;pointer-events:none;border:var(--bs-border-width) solid transparent;transform-origin:0 0;transition:opacity .1s ease-in-out,transform .1s ease-in-out}@media (prefers-reduced-motion:reduce){.form-floating>label{transition:none}}.form-floating>.form-control,.form-floating>.form-control-plaintext{padding:1rem .75rem}.form-floating>.form-control-plaintext::-moz-placeholder,.form-floating>.form-control::-moz-placeholder{color:transparent}.form-floating>.form-control-plaintext::placeholder,.form-floating>.form-control::placeholder{color:transparent}.form-floating>.form-control-plaintext:not(:-moz-placeholder-shown),.form-floating>.form-control:not(:-moz-placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control-plaintext:focus,.form-floating>.form-control-plaintext:not(:placeholder-shown),.form-floating>.form-control:focus,.form-floating>.form-control:not(:placeholder-shown){padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control-plaintext:-webkit-autofill,.form-floating>.form-control:-webkit-autofill{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-select{padding-top:1.625rem;padding-bottom:.625rem}.form-floating>.form-control:not(:-moz-placeholder-shown)~label{color:rgba(var(--bs-body-color-rgb),.65);transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control-plaintext~label,.form-floating>.form-control:focus~label,.form-floating>.form-control:not(:placeholder-shown)~label,.form-floating>.form-select~label{color:rgba(var(--bs-body-color-rgb),.65);transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control:not(:-moz-placeholder-shown)~label::after{position:absolute;inset:1rem 0.375rem;z-index:-1;height:1.5em;content:"";background-color:var(--bs-body-bg);border-radius:var(--bs-border-radius)}.form-floating>.form-control-plaintext~label::after,.form-floating>.form-control:focus~label::after,.form-floating>.form-control:not(:placeholder-shown)~label::after,.form-floating>.form-select~label::after{position:absolute;inset:1rem 0.375rem;z-index:-1;height:1.5em;content:"";background-color:var(--bs-body-bg);border-radius:var(--bs-border-radius)}.form-floating>.form-control:-webkit-autofill~label{color:rgba(var(--bs-body-color-rgb),.65);transform:scale(.85) translateY(-.5rem) translateX(.15rem)}.form-floating>.form-control-plaintext~label{border-width:var(--bs-border-width) 0}.form-floating>.form-control:disabled~label,.form-floating>:disabled~label{color:#6c757d}.form-floating>.form-control:disabled~label::after,.form-floating>:disabled~label::after{background-color:var(--bs-secondary-bg)}.input-group{position:relative;display:flex;flex-wrap:wrap;align-items:stretch;width:100%}.input-group>.form-control,.input-group>.form-floating,.input-group>.form-select{position:relative;flex:1 1 auto;width:1%;min-width:0}.input-group>.form-control:focus,.input-group>.form-floating:focus-within,.input-group>.form-select:focus{z-index:5}.input-group .btn{position:relative;z-index:2}.input-group .btn:focus{z-index:5}.input-group-text{display:flex;align-items:center;padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:var(--bs-body-color);text-align:center;white-space:nowrap;background-color:var(--bs-tertiary-bg);border:var(--bs-border-width) solid var(--bs-border-color);border-radius:var(--bs-border-radius)}.input-group-lg>.btn,.input-group-lg>.form-control,.input-group-lg>.form-select,.input-group-lg>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;border-radius:var(--bs-border-radius-lg)}.input-group-sm>.btn,.input-group-sm>.form-control,.input-group-sm>.form-select,.input-group-sm>.input-group-text{padding:.25rem .5rem;font-size:.875rem;border-radius:var(--bs-border-radius-sm)}.input-group-lg>.form-select,.input-group-sm>.form-select{padding-right:3rem}.input-group:not(.has-validation)>.dropdown-toggle:nth-last-child(n+3),.input-group:not(.has-validation)>.form-floating:not(:last-child)>.form-control,.input-group:not(.has-validation)>.form-floating:not(:last-child)>.form-select,.input-group:not(.has-validation)>:not(:last-child):not(.dropdown-toggle):not(.dropdown-menu):not(.form-floating){border-top-right-radius:0;border-bottom-right-radius:0}.input-group.has-validation>.dropdown-toggle:nth-last-child(n+4),.input-group.has-validation>.form-floating:nth-last-child(n+3)>.form-control,.input-group.has-validation>.form-floating:nth-last-child(n+3)>.form-select,.input-group.has-validation>:nth-last-child(n+3):not(.dropdown-toggle):not(.dropdown-menu):not(.form-floating){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>:not(:first-child):not(.dropdown-menu):not(.valid-tooltip):not(.valid-feedback):not(.invalid-tooltip):not(.invalid-feedback){margin-left:calc(var(--bs-border-width) * -1);border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.form-floating:not(:first-child)>.form-control,.input-group>.form-floating:not(:first-child)>.form-select{border-top-left-radius:0;border-bottom-left-radius:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:var(--bs-form-valid-color)}.valid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;color:#fff;background-color:var(--bs-success);border-radius:var(--bs-border-radius)}.is-valid~.valid-feedback,.is-valid~.valid-tooltip,.was-validated :valid~.valid-feedback,.was-validated :valid~.valid-tooltip{display:block}.form-control.is-valid,.was-validated .form-control:valid{border-color:var(--bs-form-valid-border-color);padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23198754' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-valid:focus,.was-validated .form-control:valid:focus{border-color:var(--bs-form-valid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-success-rgb),.25)}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.form-select.is-valid,.was-validated .form-select:valid{border-color:var(--bs-form-valid-border-color)}.form-select.is-valid:not([multiple]):not([size]),.form-select.is-valid:not([multiple])[size="1"],.was-validated .form-select:valid:not([multiple]):not([size]),.was-validated .form-select:valid:not([multiple])[size="1"]{--bs-form-select-bg-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 8 8'%3e%3cpath fill='%23198754' d='M2.3 6.73.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");padding-right:4.125rem;background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(.75em + .375rem) calc(.75em + .375rem)}.form-select.is-valid:focus,.was-validated .form-select:valid:focus{border-color:var(--bs-form-valid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-success-rgb),.25)}.form-control-color.is-valid,.was-validated .form-control-color:valid{width:calc(3rem + calc(1.5em + .75rem))}.form-check-input.is-valid,.was-validated .form-check-input:valid{border-color:var(--bs-form-valid-border-color)}.form-check-input.is-valid:checked,.was-validated .form-check-input:valid:checked{background-color:var(--bs-form-valid-color)}.form-check-input.is-valid:focus,.was-validated .form-check-input:valid:focus{box-shadow:0 0 0 .25rem rgba(var(--bs-success-rgb),.25)}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:var(--bs-form-valid-color)}.form-check-inline .form-check-input~.valid-feedback{margin-left:.5em}.input-group>.form-control:not(:focus).is-valid,.input-group>.form-floating:not(:focus-within).is-valid,.input-group>.form-select:not(:focus).is-valid,.was-validated .input-group>.form-control:not(:focus):valid,.was-validated .input-group>.form-floating:not(:focus-within):valid,.was-validated .input-group>.form-select:not(:focus):valid{z-index:3}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:.875em;color:var(--bs-form-invalid-color)}.invalid-tooltip{position:absolute;top:100%;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;color:#fff;background-color:var(--bs-danger);border-radius:var(--bs-border-radius)}.is-invalid~.invalid-feedback,.is-invalid~.invalid-tooltip,.was-validated :invalid~.invalid-feedback,.was-validated :invalid~.invalid-tooltip{display:block}.form-control.is-invalid,.was-validated .form-control:invalid{border-color:var(--bs-form-invalid-border-color);padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23dc3545'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-invalid:focus,.was-validated .form-control:invalid:focus{border-color:var(--bs-form-invalid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-danger-rgb),.25)}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.form-select.is-invalid,.was-validated .form-select:invalid{border-color:var(--bs-form-invalid-border-color)}.form-select.is-invalid:not([multiple]):not([size]),.form-select.is-invalid:not([multiple])[size="1"],.was-validated .form-select:invalid:not([multiple]):not([size]),.was-validated .form-select:invalid:not([multiple])[size="1"]{--bs-form-select-bg-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 12 12' width='12' height='12' fill='none' stroke='%23dc3545'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e");padding-right:4.125rem;background-position:right .75rem center,center right 2.25rem;background-size:16px 12px,calc(.75em + .375rem) calc(.75em + .375rem)}.form-select.is-invalid:focus,.was-validated .form-select:invalid:focus{border-color:var(--bs-form-invalid-border-color);box-shadow:0 0 0 .25rem rgba(var(--bs-danger-rgb),.25)}.form-control-color.is-invalid,.was-validated .form-control-color:invalid{width:calc(3rem + calc(1.5em + .75rem))}.form-check-input.is-invalid,.was-validated .form-check-input:invalid{border-color:var(--bs-form-invalid-border-color)}.form-check-input.is-invalid:checked,.was-validated .form-check-input:invalid:checked{background-color:var(--bs-form-invalid-color)}.form-check-input.is-invalid:focus,.was-validated .form-check-input:invalid:focus{box-shadow:0 0 0 .25rem rgba(var(--bs-danger-rgb),.25)}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:var(--bs-form-invalid-color)}.form-check-inline .form-check-input~.invalid-feedback{margin-left:.5em}.input-group>.form-control:not(:focus).is-invalid,.input-group>.form-floating:not(:focus-within).is-invalid,.input-group>.form-select:not(:focus).is-invalid,.was-validated .input-group>.form-control:not(:focus):invalid,.was-validated .input-group>.form-floating:not(:focus-within):invalid,.was-validated .input-group>.form-select:not(:focus):invalid{z-index:4}.btn{--bs-btn-padding-x:0.75rem;--bs-btn-padding-y:0.375rem;--bs-btn-font-family: ;--bs-btn-font-size:1rem;--bs-btn-font-weight:400;--bs-btn-line-height:1.5;--bs-btn-color:var(--bs-body-color);--bs-btn-bg:transparent;--bs-btn-border-width:var(--bs-border-width);--bs-btn-border-color:transparent;--bs-btn-border-radius:var(--bs-border-radius);--bs-btn-hover-border-color:transparent;--bs-btn-box-shadow:inset 0 1px 0 rgba(255, 255, 255, 0.15),0 1px 1px rgba(0, 0, 0, 0.075);--bs-btn-disabled-opacity:0.65;--bs-btn-focus-box-shadow:0 0 0 0.25rem rgba(var(--bs-btn-focus-shadow-rgb), .5);display:inline-block;padding:var(--bs-btn-padding-y) var(--bs-btn-padding-x);font-family:var(--bs-btn-font-family);font-size:var(--bs-btn-font-size);font-weight:var(--bs-btn-font-weight);line-height:var(--bs-btn-line-height);color:var(--bs-btn-color);text-align:center;text-decoration:none;vertical-align:middle;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;user-select:none;border:var(--bs-btn-border-width) solid var(--bs-btn-border-color);border-radius:var(--bs-btn-border-radius);background-color:var(--bs-btn-bg);transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.btn{transition:none}}.btn:hover{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color)}.btn-check+.btn:hover{color:var(--bs-btn-color);background-color:var(--bs-btn-bg);border-color:var(--bs-btn-border-color)}.btn:focus-visible{color:var(--bs-btn-hover-color);background-color:var(--bs-btn-hover-bg);border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:focus-visible+.btn{border-color:var(--bs-btn-hover-border-color);outline:0;box-shadow:var(--bs-btn-focus-box-shadow)}.btn-check:checked+.btn,.btn.active,.btn.show,.btn:first-child:active,:not(.btn-check)+.btn:active{color:var(--bs-btn-active-color);background-color:var(--bs-btn-active-bg);border-color:var(--bs-btn-active-border-color)}.btn-check:checked+.btn:focus-visible,.btn.active:focus-visible,.btn.show:focus-visible,.btn:first-child:active:focus-visible,:not(.btn-check)+.btn:active:focus-visible{box-shadow:var(--bs-btn-focus-box-shadow)}.btn.disabled,.btn:disabled,fieldset:disabled .btn{color:var(--bs-btn-disabled-color);pointer-events:none;background-color:var(--bs-btn-disabled-bg);border-color:var(--bs-btn-disabled-border-color);opacity:var(--bs-btn-disabled-opacity)}.btn-primary{--bs-btn-color:#fff;--bs-btn-bg:#0d6efd;--bs-btn-border-color:#0d6efd;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#0b5ed7;--bs-btn-hover-border-color:#0a58ca;--bs-btn-focus-shadow-rgb:49,132,253;--bs-btn-active-color:#fff;--bs-btn-active-bg:#0a58ca;--bs-btn-active-border-color:#0a53be;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#0d6efd;--bs-btn-disabled-border-color:#0d6efd}.btn-secondary{--bs-btn-color:#fff;--bs-btn-bg:#6c757d;--bs-btn-border-color:#6c757d;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#5c636a;--bs-btn-hover-border-color:#565e64;--bs-btn-focus-shadow-rgb:130,138,145;--bs-btn-active-color:#fff;--bs-btn-active-bg:#565e64;--bs-btn-active-border-color:#51585e;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#6c757d;--bs-btn-disabled-border-color:#6c757d}.btn-success{--bs-btn-color:#fff;--bs-btn-bg:#198754;--bs-btn-border-color:#198754;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#157347;--bs-btn-hover-border-color:#146c43;--bs-btn-focus-shadow-rgb:60,153,110;--bs-btn-active-color:#fff;--bs-btn-active-bg:#146c43;--bs-btn-active-border-color:#13653f;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#198754;--bs-btn-disabled-border-color:#198754}.btn-info{--bs-btn-color:#000;--bs-btn-bg:#0dcaf0;--bs-btn-border-color:#0dcaf0;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#31d2f2;--bs-btn-hover-border-color:#25cff2;--bs-btn-focus-shadow-rgb:11,172,204;--bs-btn-active-color:#000;--bs-btn-active-bg:#3dd5f3;--bs-btn-active-border-color:#25cff2;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#000;--bs-btn-disabled-bg:#0dcaf0;--bs-btn-disabled-border-color:#0dcaf0}.btn-warning{--bs-btn-color:#000;--bs-btn-bg:#ffc107;--bs-btn-border-color:#ffc107;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#ffca2c;--bs-btn-hover-border-color:#ffc720;--bs-btn-focus-shadow-rgb:217,164,6;--bs-btn-active-color:#000;--bs-btn-active-bg:#ffcd39;--bs-btn-active-border-color:#ffc720;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#000;--bs-btn-disabled-bg:#ffc107;--bs-btn-disabled-border-color:#ffc107}.btn-danger{--bs-btn-color:#fff;--bs-btn-bg:#dc3545;--bs-btn-border-color:#dc3545;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#bb2d3b;--bs-btn-hover-border-color:#b02a37;--bs-btn-focus-shadow-rgb:225,83,97;--bs-btn-active-color:#fff;--bs-btn-active-bg:#b02a37;--bs-btn-active-border-color:#a52834;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#dc3545;--bs-btn-disabled-border-color:#dc3545}.btn-light{--bs-btn-color:#000;--bs-btn-bg:#f8f9fa;--bs-btn-border-color:#f8f9fa;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#d3d4d5;--bs-btn-hover-border-color:#c6c7c8;--bs-btn-focus-shadow-rgb:211,212,213;--bs-btn-active-color:#000;--bs-btn-active-bg:#c6c7c8;--bs-btn-active-border-color:#babbbc;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#000;--bs-btn-disabled-bg:#f8f9fa;--bs-btn-disabled-border-color:#f8f9fa}.btn-dark{--bs-btn-color:#fff;--bs-btn-bg:#212529;--bs-btn-border-color:#212529;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#424649;--bs-btn-hover-border-color:#373b3e;--bs-btn-focus-shadow-rgb:66,70,73;--bs-btn-active-color:#fff;--bs-btn-active-bg:#4d5154;--bs-btn-active-border-color:#373b3e;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#fff;--bs-btn-disabled-bg:#212529;--bs-btn-disabled-border-color:#212529}.btn-outline-primary{--bs-btn-color:#0d6efd;--bs-btn-border-color:#0d6efd;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#0d6efd;--bs-btn-hover-border-color:#0d6efd;--bs-btn-focus-shadow-rgb:13,110,253;--bs-btn-active-color:#fff;--bs-btn-active-bg:#0d6efd;--bs-btn-active-border-color:#0d6efd;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#0d6efd;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#0d6efd;--bs-gradient:none}.btn-outline-secondary{--bs-btn-color:#6c757d;--bs-btn-border-color:#6c757d;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#6c757d;--bs-btn-hover-border-color:#6c757d;--bs-btn-focus-shadow-rgb:108,117,125;--bs-btn-active-color:#fff;--bs-btn-active-bg:#6c757d;--bs-btn-active-border-color:#6c757d;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#6c757d;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#6c757d;--bs-gradient:none}.btn-outline-success{--bs-btn-color:#198754;--bs-btn-border-color:#198754;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#198754;--bs-btn-hover-border-color:#198754;--bs-btn-focus-shadow-rgb:25,135,84;--bs-btn-active-color:#fff;--bs-btn-active-bg:#198754;--bs-btn-active-border-color:#198754;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#198754;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#198754;--bs-gradient:none}.btn-outline-info{--bs-btn-color:#0dcaf0;--bs-btn-border-color:#0dcaf0;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#0dcaf0;--bs-btn-hover-border-color:#0dcaf0;--bs-btn-focus-shadow-rgb:13,202,240;--bs-btn-active-color:#000;--bs-btn-active-bg:#0dcaf0;--bs-btn-active-border-color:#0dcaf0;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#0dcaf0;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#0dcaf0;--bs-gradient:none}.btn-outline-warning{--bs-btn-color:#ffc107;--bs-btn-border-color:#ffc107;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#ffc107;--bs-btn-hover-border-color:#ffc107;--bs-btn-focus-shadow-rgb:255,193,7;--bs-btn-active-color:#000;--bs-btn-active-bg:#ffc107;--bs-btn-active-border-color:#ffc107;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#ffc107;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#ffc107;--bs-gradient:none}.btn-outline-danger{--bs-btn-color:#dc3545;--bs-btn-border-color:#dc3545;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#dc3545;--bs-btn-hover-border-color:#dc3545;--bs-btn-focus-shadow-rgb:220,53,69;--bs-btn-active-color:#fff;--bs-btn-active-bg:#dc3545;--bs-btn-active-border-color:#dc3545;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#dc3545;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#dc3545;--bs-gradient:none}.btn-outline-light{--bs-btn-color:#f8f9fa;--bs-btn-border-color:#f8f9fa;--bs-btn-hover-color:#000;--bs-btn-hover-bg:#f8f9fa;--bs-btn-hover-border-color:#f8f9fa;--bs-btn-focus-shadow-rgb:248,249,250;--bs-btn-active-color:#000;--bs-btn-active-bg:#f8f9fa;--bs-btn-active-border-color:#f8f9fa;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#f8f9fa;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#f8f9fa;--bs-gradient:none}.btn-outline-dark{--bs-btn-color:#212529;--bs-btn-border-color:#212529;--bs-btn-hover-color:#fff;--bs-btn-hover-bg:#212529;--bs-btn-hover-border-color:#212529;--bs-btn-focus-shadow-rgb:33,37,41;--bs-btn-active-color:#fff;--bs-btn-active-bg:#212529;--bs-btn-active-border-color:#212529;--bs-btn-active-shadow:inset 0 3px 5px rgba(0, 0, 0, 0.125);--bs-btn-disabled-color:#212529;--bs-btn-disabled-bg:transparent;--bs-btn-disabled-border-color:#212529;--bs-gradient:none}.btn-link{--bs-btn-font-weight:400;--bs-btn-color:var(--bs-link-color);--bs-btn-bg:transparent;--bs-btn-border-color:transparent;--bs-btn-hover-color:var(--bs-link-hover-color);--bs-btn-hover-border-color:transparent;--bs-btn-active-color:var(--bs-link-hover-color);--bs-btn-active-border-color:transparent;--bs-btn-disabled-color:#6c757d;--bs-btn-disabled-border-color:transparent;--bs-btn-box-shadow:0 0 0 #000;--bs-btn-focus-shadow-rgb:49,132,253;text-decoration:underline}.btn-link:focus-visible{color:var(--bs-btn-color)}.btn-link:hover{color:var(--bs-btn-hover-color)}.btn-group-lg>.btn,.btn-lg{--bs-btn-padding-y:0.5rem;--bs-btn-padding-x:1rem;--bs-btn-font-size:1.25rem;--bs-btn-border-radius:var(--bs-border-radius-lg)}.btn-group-sm>.btn,.btn-sm{--bs-btn-padding-y:0.25rem;--bs-btn-padding-x:0.5rem;--bs-btn-font-size:0.875rem;--bs-btn-border-radius:var(--bs-border-radius-sm)}.fade{transition:opacity .15s linear}@media (prefers-reduced-motion:reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{height:0;overflow:hidden;transition:height .35s ease}@media (prefers-reduced-motion:reduce){.collapsing{transition:none}}.collapsing.collapse-horizontal{width:0;height:auto;transition:width .35s ease}@media (prefers-reduced-motion:reduce){.collapsing.collapse-horizontal{transition:none}}.dropdown,.dropdown-center,.dropend,.dropstart,.dropup,.dropup-center{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{--bs-dropdown-zindex:1000;--bs-dropdown-min-width:10rem;--bs-dropdown-padding-x:0;--bs-dropdown-padding-y:0.5rem;--bs-dropdown-spacer:0.125rem;--bs-dropdown-font-size:1rem;--bs-dropdown-color:var(--bs-body-color);--bs-dropdown-bg:var(--bs-body-bg);--bs-dropdown-border-color:var(--bs-border-color-translucent);--bs-dropdown-border-radius:var(--bs-border-radius);--bs-dropdown-border-width:var(--bs-border-width);--bs-dropdown-inner-border-radius:calc(var(--bs-border-radius) - var(--bs-border-width));--bs-dropdown-divider-bg:var(--bs-border-color-translucent);--bs-dropdown-divider-margin-y:0.5rem;--bs-dropdown-box-shadow:0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-dropdown-link-color:var(--bs-body-color);--bs-dropdown-link-hover-color:var(--bs-body-color);--bs-dropdown-link-hover-bg:var(--bs-tertiary-bg);--bs-dropdown-link-active-color:#fff;--bs-dropdown-link-active-bg:#0d6efd;--bs-dropdown-link-disabled-color:var(--bs-tertiary-color);--bs-dropdown-item-padding-x:1rem;--bs-dropdown-item-padding-y:0.25rem;--bs-dropdown-header-color:#6c757d;--bs-dropdown-header-padding-x:1rem;--bs-dropdown-header-padding-y:0.5rem;position:absolute;z-index:var(--bs-dropdown-zindex);display:none;min-width:var(--bs-dropdown-min-width);padding:var(--bs-dropdown-padding-y) var(--bs-dropdown-padding-x);margin:0;font-size:var(--bs-dropdown-font-size);color:var(--bs-dropdown-color);text-align:left;list-style:none;background-color:var(--bs-dropdown-bg);background-clip:padding-box;border:var(--bs-dropdown-border-width) solid var(--bs-dropdown-border-color);border-radius:var(--bs-dropdown-border-radius)}.dropdown-menu[data-bs-popper]{top:100%;left:0;margin-top:var(--bs-dropdown-spacer)}.dropdown-menu-start{--bs-position:start}.dropdown-menu-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-end{--bs-position:end}.dropdown-menu-end[data-bs-popper]{right:0;left:auto}@media (min-width:576px){.dropdown-menu-sm-start{--bs-position:start}.dropdown-menu-sm-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-sm-end{--bs-position:end}.dropdown-menu-sm-end[data-bs-popper]{right:0;left:auto}}@media (min-width:768px){.dropdown-menu-md-start{--bs-position:start}.dropdown-menu-md-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-md-end{--bs-position:end}.dropdown-menu-md-end[data-bs-popper]{right:0;left:auto}}@media (min-width:992px){.dropdown-menu-lg-start{--bs-position:start}.dropdown-menu-lg-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-lg-end{--bs-position:end}.dropdown-menu-lg-end[data-bs-popper]{right:0;left:auto}}@media (min-width:1200px){.dropdown-menu-xl-start{--bs-position:start}.dropdown-menu-xl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xl-end{--bs-position:end}.dropdown-menu-xl-end[data-bs-popper]{right:0;left:auto}}@media (min-width:1400px){.dropdown-menu-xxl-start{--bs-position:start}.dropdown-menu-xxl-start[data-bs-popper]{right:auto;left:0}.dropdown-menu-xxl-end{--bs-position:end}.dropdown-menu-xxl-end[data-bs-popper]{right:0;left:auto}}.dropup .dropdown-menu[data-bs-popper]{top:auto;bottom:100%;margin-top:0;margin-bottom:var(--bs-dropdown-spacer)}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-menu[data-bs-popper]{top:0;right:auto;left:100%;margin-top:0;margin-left:var(--bs-dropdown-spacer)}.dropend .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropend .dropdown-toggle:empty::after{margin-left:0}.dropend .dropdown-toggle::after{vertical-align:0}.dropstart .dropdown-menu[data-bs-popper]{top:0;right:100%;left:auto;margin-top:0;margin-right:var(--bs-dropdown-spacer)}.dropstart .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropstart .dropdown-toggle::after{display:none}.dropstart .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropstart .dropdown-toggle:empty::after{margin-left:0}.dropstart .dropdown-toggle::before{vertical-align:0}.dropdown-divider{height:0;margin:var(--bs-dropdown-divider-margin-y) 0;overflow:hidden;border-top:1px solid var(--bs-dropdown-divider-bg);opacity:1}.dropdown-item{display:block;width:100%;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);clear:both;font-weight:400;color:var(--bs-dropdown-link-color);text-align:inherit;text-decoration:none;white-space:nowrap;background-color:transparent;border:0;border-radius:var(--bs-dropdown-item-border-radius,0)}.dropdown-item:focus,.dropdown-item:hover{color:var(--bs-dropdown-link-hover-color);background-color:var(--bs-dropdown-link-hover-bg)}.dropdown-item.active,.dropdown-item:active{color:var(--bs-dropdown-link-active-color);text-decoration:none;background-color:var(--bs-dropdown-link-active-bg)}.dropdown-item.disabled,.dropdown-item:disabled{color:var(--bs-dropdown-link-disabled-color);pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:var(--bs-dropdown-header-padding-y) var(--bs-dropdown-header-padding-x);margin-bottom:0;font-size:.875rem;color:var(--bs-dropdown-header-color);white-space:nowrap}.dropdown-item-text{display:block;padding:var(--bs-dropdown-item-padding-y) var(--bs-dropdown-item-padding-x);color:var(--bs-dropdown-link-color)}.dropdown-menu-dark{--bs-dropdown-color:#dee2e6;--bs-dropdown-bg:#343a40;--bs-dropdown-border-color:var(--bs-border-color-translucent);--bs-dropdown-box-shadow: ;--bs-dropdown-link-color:#dee2e6;--bs-dropdown-link-hover-color:#fff;--bs-dropdown-divider-bg:var(--bs-border-color-translucent);--bs-dropdown-link-hover-bg:rgba(255, 255, 255, 0.15);--bs-dropdown-link-active-color:#fff;--bs-dropdown-link-active-bg:#0d6efd;--bs-dropdown-link-disabled-color:#adb5bd;--bs-dropdown-header-color:#adb5bd}.btn-group,.btn-group-vertical{position:relative;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;flex:1 1 auto}.btn-group-vertical>.btn-check:checked+.btn,.btn-group-vertical>.btn-check:focus+.btn,.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:hover,.btn-group>.btn-check:checked+.btn,.btn-group>.btn-check:focus+.btn,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus,.btn-group>.btn:hover{z-index:1}.btn-toolbar{display:flex;flex-wrap:wrap;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group{border-radius:var(--bs-border-radius)}.btn-group>.btn-group:not(:first-child),.btn-group>:not(.btn-check:first-child)+.btn{margin-left:calc(var(--bs-border-width) * -1)}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn.dropdown-toggle-split:first-child,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:nth-child(n+3),.btn-group>:not(.btn-check)+.btn{border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropend .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropstart .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{flex-direction:column;align-items:flex-start;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn-group:not(:first-child),.btn-group-vertical>.btn:not(:first-child){margin-top:calc(var(--bs-border-width) * -1)}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn~.btn{border-top-left-radius:0;border-top-right-radius:0}.nav{--bs-nav-link-padding-x:1rem;--bs-nav-link-padding-y:0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color:var(--bs-link-color);--bs-nav-link-hover-color:var(--bs-link-hover-color);--bs-nav-link-disabled-color:var(--bs-secondary-color);display:flex;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:var(--bs-nav-link-padding-y) var(--bs-nav-link-padding-x);font-size:var(--bs-nav-link-font-size);font-weight:var(--bs-nav-link-font-weight);color:var(--bs-nav-link-color);text-decoration:none;background:0 0;border:0;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out}@media (prefers-reduced-motion:reduce){.nav-link{transition:none}}.nav-link:focus,.nav-link:hover{color:var(--bs-nav-link-hover-color)}.nav-link:focus-visible{outline:0;box-shadow:0 0 0 .25rem rgba(13,110,253,.25)}.nav-link.disabled,.nav-link:disabled{color:var(--bs-nav-link-disabled-color);pointer-events:none;cursor:default}.nav-tabs{--bs-nav-tabs-border-width:var(--bs-border-width);--bs-nav-tabs-border-color:var(--bs-border-color);--bs-nav-tabs-border-radius:var(--bs-border-radius);--bs-nav-tabs-link-hover-border-color:var(--bs-secondary-bg) var(--bs-secondary-bg) var(--bs-border-color);--bs-nav-tabs-link-active-color:var(--bs-emphasis-color);--bs-nav-tabs-link-active-bg:var(--bs-body-bg);--bs-nav-tabs-link-active-border-color:var(--bs-border-color) var(--bs-border-color) var(--bs-body-bg);border-bottom:var(--bs-nav-tabs-border-width) solid var(--bs-nav-tabs-border-color)}.nav-tabs .nav-link{margin-bottom:calc(-1 * var(--bs-nav-tabs-border-width));border:var(--bs-nav-tabs-border-width) solid transparent;border-top-left-radius:var(--bs-nav-tabs-border-radius);border-top-right-radius:var(--bs-nav-tabs-border-radius)}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{isolation:isolate;border-color:var(--bs-nav-tabs-link-hover-border-color)}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:var(--bs-nav-tabs-link-active-color);background-color:var(--bs-nav-tabs-link-active-bg);border-color:var(--bs-nav-tabs-link-active-border-color)}.nav-tabs .dropdown-menu{margin-top:calc(-1 * var(--bs-nav-tabs-border-width));border-top-left-radius:0;border-top-right-radius:0}.nav-pills{--bs-nav-pills-border-radius:var(--bs-border-radius);--bs-nav-pills-link-active-color:#fff;--bs-nav-pills-link-active-bg:#0d6efd}.nav-pills .nav-link{border-radius:var(--bs-nav-pills-border-radius)}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:var(--bs-nav-pills-link-active-color);background-color:var(--bs-nav-pills-link-active-bg)}.nav-underline{--bs-nav-underline-gap:1rem;--bs-nav-underline-border-width:0.125rem;--bs-nav-underline-link-active-color:var(--bs-emphasis-color);gap:var(--bs-nav-underline-gap)}.nav-underline .nav-link{padding-right:0;padding-left:0;border-bottom:var(--bs-nav-underline-border-width) solid transparent}.nav-underline .nav-link:focus,.nav-underline .nav-link:hover{border-bottom-color:currentcolor}.nav-underline .nav-link.active,.nav-underline .show>.nav-link{font-weight:700;color:var(--bs-nav-underline-link-active-color);border-bottom-color:currentcolor}.nav-fill .nav-item,.nav-fill>.nav-link{flex:1 1 auto;text-align:center}.nav-justified .nav-item,.nav-justified>.nav-link{flex-basis:0;flex-grow:1;text-align:center}.nav-fill .nav-item .nav-link,.nav-justified .nav-item .nav-link{width:100%}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{--bs-navbar-padding-x:0;--bs-navbar-padding-y:0.5rem;--bs-navbar-color:rgba(var(--bs-emphasis-color-rgb), 0.65);--bs-navbar-hover-color:rgba(var(--bs-emphasis-color-rgb), 0.8);--bs-navbar-disabled-color:rgba(var(--bs-emphasis-color-rgb), 0.3);--bs-navbar-active-color:rgba(var(--bs-emphasis-color-rgb), 1);--bs-navbar-brand-padding-y:0.3125rem;--bs-navbar-brand-margin-end:1rem;--bs-navbar-brand-font-size:1.25rem;--bs-navbar-brand-color:rgba(var(--bs-emphasis-color-rgb), 1);--bs-navbar-brand-hover-color:rgba(var(--bs-emphasis-color-rgb), 1);--bs-navbar-nav-link-padding-x:0.5rem;--bs-navbar-toggler-padding-y:0.25rem;--bs-navbar-toggler-padding-x:0.75rem;--bs-navbar-toggler-font-size:1.25rem;--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%2833, 37, 41, 0.75%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e");--bs-navbar-toggler-border-color:rgba(var(--bs-emphasis-color-rgb), 0.15);--bs-navbar-toggler-border-radius:var(--bs-border-radius);--bs-navbar-toggler-focus-width:0.25rem;--bs-navbar-toggler-transition:box-shadow 0.15s ease-in-out;position:relative;display:flex;flex-wrap:wrap;align-items:center;justify-content:space-between;padding:var(--bs-navbar-padding-y) var(--bs-navbar-padding-x)}.navbar>.container,.navbar>.container-fluid,.navbar>.container-lg,.navbar>.container-md,.navbar>.container-sm,.navbar>.container-xl,.navbar>.container-xxl{display:flex;flex-wrap:inherit;align-items:center;justify-content:space-between}.navbar-brand{padding-top:var(--bs-navbar-brand-padding-y);padding-bottom:var(--bs-navbar-brand-padding-y);margin-right:var(--bs-navbar-brand-margin-end);font-size:var(--bs-navbar-brand-font-size);color:var(--bs-navbar-brand-color);text-decoration:none;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{color:var(--bs-navbar-brand-hover-color)}.navbar-nav{--bs-nav-link-padding-x:0;--bs-nav-link-padding-y:0.5rem;--bs-nav-link-font-weight: ;--bs-nav-link-color:var(--bs-navbar-color);--bs-nav-link-hover-color:var(--bs-navbar-hover-color);--bs-nav-link-disabled-color:var(--bs-navbar-disabled-color);display:flex;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link.active,.navbar-nav .nav-link.show{color:var(--bs-navbar-active-color)}.navbar-nav .dropdown-menu{position:static}.navbar-text{padding-top:.5rem;padding-bottom:.5rem;color:var(--bs-navbar-color)}.navbar-text a,.navbar-text a:focus,.navbar-text a:hover{color:var(--bs-navbar-active-color)}.navbar-collapse{flex-basis:100%;flex-grow:1;align-items:center}.navbar-toggler{padding:var(--bs-navbar-toggler-padding-y) var(--bs-navbar-toggler-padding-x);font-size:var(--bs-navbar-toggler-font-size);line-height:1;color:var(--bs-navbar-color);background-color:transparent;border:var(--bs-border-width) solid var(--bs-navbar-toggler-border-color);border-radius:var(--bs-navbar-toggler-border-radius);transition:var(--bs-navbar-toggler-transition)}@media (prefers-reduced-motion:reduce){.navbar-toggler{transition:none}}.navbar-toggler:hover{text-decoration:none}.navbar-toggler:focus{text-decoration:none;outline:0;box-shadow:0 0 0 var(--bs-navbar-toggler-focus-width)}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;background-image:var(--bs-navbar-toggler-icon-bg);background-repeat:no-repeat;background-position:center;background-size:100%}.navbar-nav-scroll{max-height:var(--bs-scroll-height,75vh);overflow-y:auto}@media (min-width:576px){.navbar-expand-sm{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-sm .navbar-nav{flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-sm .navbar-nav-scroll{overflow:visible}.navbar-expand-sm .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}.navbar-expand-sm .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-sm .offcanvas .offcanvas-header{display:none}.navbar-expand-sm .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:768px){.navbar-expand-md{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-md .navbar-nav{flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-md .navbar-nav-scroll{overflow:visible}.navbar-expand-md .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}.navbar-expand-md .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-md .offcanvas .offcanvas-header{display:none}.navbar-expand-md .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:992px){.navbar-expand-lg{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-lg .navbar-nav{flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-lg .navbar-nav-scroll{overflow:visible}.navbar-expand-lg .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}.navbar-expand-lg .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-lg .offcanvas .offcanvas-header{display:none}.navbar-expand-lg .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:1200px){.navbar-expand-xl{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-xl .navbar-nav{flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xl .navbar-nav-scroll{overflow:visible}.navbar-expand-xl .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}.navbar-expand-xl .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-xl .offcanvas .offcanvas-header{display:none}.navbar-expand-xl .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}@media (min-width:1400px){.navbar-expand-xxl{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand-xxl .navbar-nav{flex-direction:row}.navbar-expand-xxl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xxl .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand-xxl .navbar-nav-scroll{overflow:visible}.navbar-expand-xxl .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand-xxl .navbar-toggler{display:none}.navbar-expand-xxl .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand-xxl .offcanvas .offcanvas-header{display:none}.navbar-expand-xxl .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}}.navbar-expand{flex-wrap:nowrap;justify-content:flex-start}.navbar-expand .navbar-nav{flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:var(--bs-navbar-nav-link-padding-x);padding-left:var(--bs-navbar-nav-link-padding-x)}.navbar-expand .navbar-nav-scroll{overflow:visible}.navbar-expand .navbar-collapse{display:flex!important;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-expand .offcanvas{position:static;z-index:auto;flex-grow:1;width:auto!important;height:auto!important;visibility:visible!important;background-color:transparent!important;border:0!important;transform:none!important;transition:none}.navbar-expand .offcanvas .offcanvas-header{display:none}.navbar-expand .offcanvas .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible}.navbar-dark,.navbar[data-bs-theme=dark]{--bs-navbar-color:rgba(255, 255, 255, 0.55);--bs-navbar-hover-color:rgba(255, 255, 255, 0.75);--bs-navbar-disabled-color:rgba(255, 255, 255, 0.25);--bs-navbar-active-color:#fff;--bs-navbar-brand-color:#fff;--bs-navbar-brand-hover-color:#fff;--bs-navbar-toggler-border-color:rgba(255, 255, 255, 0.1);--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}[data-bs-theme=dark] .navbar-toggler-icon{--bs-navbar-toggler-icon-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.card{--bs-card-spacer-y:1rem;--bs-card-spacer-x:1rem;--bs-card-title-spacer-y:0.5rem;--bs-card-title-color: ;--bs-card-subtitle-color: ;--bs-card-border-width:var(--bs-border-width);--bs-card-border-color:var(--bs-border-color-translucent);--bs-card-border-radius:var(--bs-border-radius);--bs-card-box-shadow: ;--bs-card-inner-border-radius:calc(var(--bs-border-radius) - (var(--bs-border-width)));--bs-card-cap-padding-y:0.5rem;--bs-card-cap-padding-x:1rem;--bs-card-cap-bg:rgba(var(--bs-body-color-rgb), 0.03);--bs-card-cap-color: ;--bs-card-height: ;--bs-card-color: ;--bs-card-bg:var(--bs-body-bg);--bs-card-img-overlay-padding:1rem;--bs-card-group-margin:0.75rem;position:relative;display:flex;flex-direction:column;min-width:0;height:var(--bs-card-height);color:var(--bs-body-color);word-wrap:break-word;background-color:var(--bs-card-bg);background-clip:border-box;border:var(--bs-card-border-width) solid var(--bs-card-border-color);border-radius:var(--bs-card-border-radius)}.card>hr{margin-right:0;margin-left:0}.card>.list-group{border-top:inherit;border-bottom:inherit}.card>.list-group:first-child{border-top-width:0;border-top-left-radius:var(--bs-card-inner-border-radius);border-top-right-radius:var(--bs-card-inner-border-radius)}.card>.list-group:last-child{border-bottom-width:0;border-bottom-right-radius:var(--bs-card-inner-border-radius);border-bottom-left-radius:var(--bs-card-inner-border-radius)}.card>.card-header+.list-group,.card>.list-group+.card-footer{border-top:0}.card-body{flex:1 1 auto;padding:var(--bs-card-spacer-y) var(--bs-card-spacer-x);color:var(--bs-card-color)}.card-title{margin-bottom:var(--bs-card-title-spacer-y);color:var(--bs-card-title-color)}.card-subtitle{margin-top:calc(-.5 * var(--bs-card-title-spacer-y));margin-bottom:0;color:var(--bs-card-subtitle-color)}.card-text:last-child{margin-bottom:0}.card-link+.card-link{margin-left:var(--bs-card-spacer-x)}.card-header{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);margin-bottom:0;color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-bottom:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-header:first-child{border-radius:var(--bs-card-inner-border-radius) var(--bs-card-inner-border-radius) 0 0}.card-footer{padding:var(--bs-card-cap-padding-y) var(--bs-card-cap-padding-x);color:var(--bs-card-cap-color);background-color:var(--bs-card-cap-bg);border-top:var(--bs-card-border-width) solid var(--bs-card-border-color)}.card-footer:last-child{border-radius:0 0 var(--bs-card-inner-border-radius) var(--bs-card-inner-border-radius)}.card-header-tabs{margin-right:calc(-.5 * var(--bs-card-cap-padding-x));margin-bottom:calc(-1 * var(--bs-card-cap-padding-y));margin-left:calc(-.5 * var(--bs-card-cap-padding-x));border-bottom:0}.card-header-tabs .nav-link.active{background-color:var(--bs-card-bg);border-bottom-color:var(--bs-card-bg)}.card-header-pills{margin-right:calc(-.5 * var(--bs-card-cap-padding-x));margin-left:calc(-.5 * var(--bs-card-cap-padding-x))}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:var(--bs-card-img-overlay-padding);border-radius:var(--bs-card-inner-border-radius)}.card-img,.card-img-bottom,.card-img-top{width:100%}.card-img,.card-img-top{border-top-left-radius:var(--bs-card-inner-border-radius);border-top-right-radius:var(--bs-card-inner-border-radius)}.card-img,.card-img-bottom{border-bottom-right-radius:var(--bs-card-inner-border-radius);border-bottom-left-radius:var(--bs-card-inner-border-radius)}.card-group>.card{margin-bottom:var(--bs-card-group-margin)}@media (min-width:576px){.card-group{display:flex;flex-flow:row wrap}.card-group>.card{flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:not(:last-child) .card-header,.card-group>.card:not(:last-child) .card-img-top{border-top-right-radius:0}.card-group>.card:not(:last-child) .card-footer,.card-group>.card:not(:last-child) .card-img-bottom{border-bottom-right-radius:0}.card-group>.card:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:not(:first-child) .card-header,.card-group>.card:not(:first-child) .card-img-top{border-top-left-radius:0}.card-group>.card:not(:first-child) .card-footer,.card-group>.card:not(:first-child) .card-img-bottom{border-bottom-left-radius:0}}.accordion{--bs-accordion-color:var(--bs-body-color);--bs-accordion-bg:var(--bs-body-bg);--bs-accordion-transition:color 0.15s ease-in-out,background-color 0.15s ease-in-out,border-color 0.15s ease-in-out,box-shadow 0.15s ease-in-out,border-radius 0.15s ease;--bs-accordion-border-color:var(--bs-border-color);--bs-accordion-border-width:var(--bs-border-width);--bs-accordion-border-radius:var(--bs-border-radius);--bs-accordion-inner-border-radius:calc(var(--bs-border-radius) - (var(--bs-border-width)));--bs-accordion-btn-padding-x:1.25rem;--bs-accordion-btn-padding-y:1rem;--bs-accordion-btn-color:var(--bs-body-color);--bs-accordion-btn-bg:var(--bs-accordion-bg);--bs-accordion-btn-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23212529'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-icon-width:1.25rem;--bs-accordion-btn-icon-transform:rotate(-180deg);--bs-accordion-btn-icon-transition:transform 0.2s ease-in-out;--bs-accordion-btn-active-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23052c65'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-focus-border-color:#86b7fe;--bs-accordion-btn-focus-box-shadow:0 0 0 0.25rem rgba(13, 110, 253, 0.25);--bs-accordion-body-padding-x:1.25rem;--bs-accordion-body-padding-y:1rem;--bs-accordion-active-color:var(--bs-primary-text-emphasis);--bs-accordion-active-bg:var(--bs-primary-bg-subtle)}.accordion-button{position:relative;display:flex;align-items:center;width:100%;padding:var(--bs-accordion-btn-padding-y) var(--bs-accordion-btn-padding-x);font-size:1rem;color:var(--bs-accordion-btn-color);text-align:left;background-color:var(--bs-accordion-btn-bg);border:0;border-radius:0;overflow-anchor:none;transition:var(--bs-accordion-transition)}@media (prefers-reduced-motion:reduce){.accordion-button{transition:none}}.accordion-button:not(.collapsed){color:var(--bs-accordion-active-color);background-color:var(--bs-accordion-active-bg);box-shadow:inset 0 calc(-1 * var(--bs-accordion-border-width)) 0 var(--bs-accordion-border-color)}.accordion-button:not(.collapsed)::after{background-image:var(--bs-accordion-btn-active-icon);transform:var(--bs-accordion-btn-icon-transform)}.accordion-button::after{flex-shrink:0;width:var(--bs-accordion-btn-icon-width);height:var(--bs-accordion-btn-icon-width);margin-left:auto;content:"";background-image:var(--bs-accordion-btn-icon);background-repeat:no-repeat;background-size:var(--bs-accordion-btn-icon-width);transition:var(--bs-accordion-btn-icon-transition)}@media (prefers-reduced-motion:reduce){.accordion-button::after{transition:none}}.accordion-button:hover{z-index:2}.accordion-button:focus{z-index:3;border-color:var(--bs-accordion-btn-focus-border-color);outline:0;box-shadow:var(--bs-accordion-btn-focus-box-shadow)}.accordion-header{margin-bottom:0}.accordion-item{color:var(--bs-accordion-color);background-color:var(--bs-accordion-bg);border:var(--bs-accordion-border-width) solid var(--bs-accordion-border-color)}.accordion-item:first-of-type{border-top-left-radius:var(--bs-accordion-border-radius);border-top-right-radius:var(--bs-accordion-border-radius)}.accordion-item:first-of-type .accordion-button{border-top-left-radius:var(--bs-accordion-inner-border-radius);border-top-right-radius:var(--bs-accordion-inner-border-radius)}.accordion-item:not(:first-of-type){border-top:0}.accordion-item:last-of-type{border-bottom-right-radius:var(--bs-accordion-border-radius);border-bottom-left-radius:var(--bs-accordion-border-radius)}.accordion-item:last-of-type .accordion-button.collapsed{border-bottom-right-radius:var(--bs-accordion-inner-border-radius);border-bottom-left-radius:var(--bs-accordion-inner-border-radius)}.accordion-item:last-of-type .accordion-collapse{border-bottom-right-radius:var(--bs-accordion-border-radius);border-bottom-left-radius:var(--bs-accordion-border-radius)}.accordion-body{padding:var(--bs-accordion-body-padding-y) var(--bs-accordion-body-padding-x)}.accordion-flush .accordion-collapse{border-width:0}.accordion-flush .accordion-item{border-right:0;border-left:0;border-radius:0}.accordion-flush .accordion-item:first-child{border-top:0}.accordion-flush .accordion-item:last-child{border-bottom:0}.accordion-flush .accordion-item .accordion-button,.accordion-flush .accordion-item .accordion-button.collapsed{border-radius:0}[data-bs-theme=dark] .accordion-button::after{--bs-accordion-btn-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%236ea8fe'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e");--bs-accordion-btn-active-icon:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%236ea8fe'%3e%3cpath fill-rule='evenodd' d='M1.646 4.646a.5.5 0 0 1 .708 0L8 10.293l5.646-5.647a.5.5 0 0 1 .708.708l-6 6a.5.5 0 0 1-.708 0l-6-6a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e")}.breadcrumb{--bs-breadcrumb-padding-x:0;--bs-breadcrumb-padding-y:0;--bs-breadcrumb-margin-bottom:1rem;--bs-breadcrumb-bg: ;--bs-breadcrumb-border-radius: ;--bs-breadcrumb-divider-color:var(--bs-secondary-color);--bs-breadcrumb-item-padding-x:0.5rem;--bs-breadcrumb-item-active-color:var(--bs-secondary-color);display:flex;flex-wrap:wrap;padding:var(--bs-breadcrumb-padding-y) var(--bs-breadcrumb-padding-x);margin-bottom:var(--bs-breadcrumb-margin-bottom);font-size:var(--bs-breadcrumb-font-size);list-style:none;background-color:var(--bs-breadcrumb-bg);border-radius:var(--bs-breadcrumb-border-radius)}.breadcrumb-item+.breadcrumb-item{padding-left:var(--bs-breadcrumb-item-padding-x)}.breadcrumb-item+.breadcrumb-item::before{float:left;padding-right:var(--bs-breadcrumb-item-padding-x);color:var(--bs-breadcrumb-divider-color);content:var(--bs-breadcrumb-divider, "/")}.breadcrumb-item.active{color:var(--bs-breadcrumb-item-active-color)}.pagination{--bs-pagination-padding-x:0.75rem;--bs-pagination-padding-y:0.375rem;--bs-pagination-font-size:1rem;--bs-pagination-color:var(--bs-link-color);--bs-pagination-bg:var(--bs-body-bg);--bs-pagination-border-width:var(--bs-border-width);--bs-pagination-border-color:var(--bs-border-color);--bs-pagination-border-radius:var(--bs-border-radius);--bs-pagination-hover-color:var(--bs-link-hover-color);--bs-pagination-hover-bg:var(--bs-tertiary-bg);--bs-pagination-hover-border-color:var(--bs-border-color);--bs-pagination-focus-color:var(--bs-link-hover-color);--bs-pagination-focus-bg:var(--bs-secondary-bg);--bs-pagination-focus-box-shadow:0 0 0 0.25rem rgba(13, 110, 253, 0.25);--bs-pagination-active-color:#fff;--bs-pagination-active-bg:#0d6efd;--bs-pagination-active-border-color:#0d6efd;--bs-pagination-disabled-color:var(--bs-secondary-color);--bs-pagination-disabled-bg:var(--bs-secondary-bg);--bs-pagination-disabled-border-color:var(--bs-border-color);display:flex;padding-left:0;list-style:none}.page-link{position:relative;display:block;padding:var(--bs-pagination-padding-y) var(--bs-pagination-padding-x);font-size:var(--bs-pagination-font-size);color:var(--bs-pagination-color);text-decoration:none;background-color:var(--bs-pagination-bg);border:var(--bs-pagination-border-width) solid var(--bs-pagination-border-color);transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.page-link{transition:none}}.page-link:hover{z-index:2;color:var(--bs-pagination-hover-color);background-color:var(--bs-pagination-hover-bg);border-color:var(--bs-pagination-hover-border-color)}.page-link:focus{z-index:3;color:var(--bs-pagination-focus-color);background-color:var(--bs-pagination-focus-bg);outline:0;box-shadow:var(--bs-pagination-focus-box-shadow)}.active>.page-link,.page-link.active{z-index:3;color:var(--bs-pagination-active-color);background-color:var(--bs-pagination-active-bg);border-color:var(--bs-pagination-active-border-color)}.disabled>.page-link,.page-link.disabled{color:var(--bs-pagination-disabled-color);pointer-events:none;background-color:var(--bs-pagination-disabled-bg);border-color:var(--bs-pagination-disabled-border-color)}.page-item:not(:first-child) .page-link{margin-left:calc(var(--bs-border-width) * -1)}.page-item:first-child .page-link{border-top-left-radius:var(--bs-pagination-border-radius);border-bottom-left-radius:var(--bs-pagination-border-radius)}.page-item:last-child .page-link{border-top-right-radius:var(--bs-pagination-border-radius);border-bottom-right-radius:var(--bs-pagination-border-radius)}.pagination-lg{--bs-pagination-padding-x:1.5rem;--bs-pagination-padding-y:0.75rem;--bs-pagination-font-size:1.25rem;--bs-pagination-border-radius:var(--bs-border-radius-lg)}.pagination-sm{--bs-pagination-padding-x:0.5rem;--bs-pagination-padding-y:0.25rem;--bs-pagination-font-size:0.875rem;--bs-pagination-border-radius:var(--bs-border-radius-sm)}.badge{--bs-badge-padding-x:0.65em;--bs-badge-padding-y:0.35em;--bs-badge-font-size:0.75em;--bs-badge-font-weight:700;--bs-badge-color:#fff;--bs-badge-border-radius:var(--bs-border-radius);display:inline-block;padding:var(--bs-badge-padding-y) var(--bs-badge-padding-x);font-size:var(--bs-badge-font-size);font-weight:var(--bs-badge-font-weight);line-height:1;color:var(--bs-badge-color);text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:var(--bs-badge-border-radius)}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.alert{--bs-alert-bg:transparent;--bs-alert-padding-x:1rem;--bs-alert-padding-y:1rem;--bs-alert-margin-bottom:1rem;--bs-alert-color:inherit;--bs-alert-border-color:transparent;--bs-alert-border:var(--bs-border-width) solid var(--bs-alert-border-color);--bs-alert-border-radius:var(--bs-border-radius);--bs-alert-link-color:inherit;position:relative;padding:var(--bs-alert-padding-y) var(--bs-alert-padding-x);margin-bottom:var(--bs-alert-margin-bottom);color:var(--bs-alert-color);background-color:var(--bs-alert-bg);border:var(--bs-alert-border);border-radius:var(--bs-alert-border-radius)}.alert-heading{color:inherit}.alert-link{font-weight:700;color:var(--bs-alert-link-color)}.alert-dismissible{padding-right:3rem}.alert-dismissible .btn-close{position:absolute;top:0;right:0;z-index:2;padding:1.25rem 1rem}.alert-primary{--bs-alert-color:var(--bs-primary-text-emphasis);--bs-alert-bg:var(--bs-primary-bg-subtle);--bs-alert-border-color:var(--bs-primary-border-subtle);--bs-alert-link-color:var(--bs-primary-text-emphasis)}.alert-secondary{--bs-alert-color:var(--bs-secondary-text-emphasis);--bs-alert-bg:var(--bs-secondary-bg-subtle);--bs-alert-border-color:var(--bs-secondary-border-subtle);--bs-alert-link-color:var(--bs-secondary-text-emphasis)}.alert-success{--bs-alert-color:var(--bs-success-text-emphasis);--bs-alert-bg:var(--bs-success-bg-subtle);--bs-alert-border-color:var(--bs-success-border-subtle);--bs-alert-link-color:var(--bs-success-text-emphasis)}.alert-info{--bs-alert-color:var(--bs-info-text-emphasis);--bs-alert-bg:var(--bs-info-bg-subtle);--bs-alert-border-color:var(--bs-info-border-subtle);--bs-alert-link-color:var(--bs-info-text-emphasis)}.alert-warning{--bs-alert-color:var(--bs-warning-text-emphasis);--bs-alert-bg:var(--bs-warning-bg-subtle);--bs-alert-border-color:var(--bs-warning-border-subtle);--bs-alert-link-color:var(--bs-warning-text-emphasis)}.alert-danger{--bs-alert-color:var(--bs-danger-text-emphasis);--bs-alert-bg:var(--bs-danger-bg-subtle);--bs-alert-border-color:var(--bs-danger-border-subtle);--bs-alert-link-color:var(--bs-danger-text-emphasis)}.alert-light{--bs-alert-color:var(--bs-light-text-emphasis);--bs-alert-bg:var(--bs-light-bg-subtle);--bs-alert-border-color:var(--bs-light-border-subtle);--bs-alert-link-color:var(--bs-light-text-emphasis)}.alert-dark{--bs-alert-color:var(--bs-dark-text-emphasis);--bs-alert-bg:var(--bs-dark-bg-subtle);--bs-alert-border-color:var(--bs-dark-border-subtle);--bs-alert-link-color:var(--bs-dark-text-emphasis)}@keyframes progress-bar-stripes{0%{background-position-x:1rem}}.progress,.progress-stacked{--bs-progress-height:1rem;--bs-progress-font-size:0.75rem;--bs-progress-bg:var(--bs-secondary-bg);--bs-progress-border-radius:var(--bs-border-radius);--bs-progress-box-shadow:var(--bs-box-shadow-inset);--bs-progress-bar-color:#fff;--bs-progress-bar-bg:#0d6efd;--bs-progress-bar-transition:width 0.6s ease;display:flex;height:var(--bs-progress-height);overflow:hidden;font-size:var(--bs-progress-font-size);background-color:var(--bs-progress-bg);border-radius:var(--bs-progress-border-radius)}.progress-bar{display:flex;flex-direction:column;justify-content:center;overflow:hidden;color:var(--bs-progress-bar-color);text-align:center;white-space:nowrap;background-color:var(--bs-progress-bar-bg);transition:var(--bs-progress-bar-transition)}@media (prefers-reduced-motion:reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:var(--bs-progress-height) var(--bs-progress-height)}.progress-stacked>.progress{overflow:visible}.progress-stacked>.progress>.progress-bar{width:100%}.progress-bar-animated{animation:1s linear infinite progress-bar-stripes}@media (prefers-reduced-motion:reduce){.progress-bar-animated{animation:none}}.list-group{--bs-list-group-color:var(--bs-body-color);--bs-list-group-bg:var(--bs-body-bg);--bs-list-group-border-color:var(--bs-border-color);--bs-list-group-border-width:var(--bs-border-width);--bs-list-group-border-radius:var(--bs-border-radius);--bs-list-group-item-padding-x:1rem;--bs-list-group-item-padding-y:0.5rem;--bs-list-group-action-color:var(--bs-secondary-color);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-tertiary-bg);--bs-list-group-action-active-color:var(--bs-body-color);--bs-list-group-action-active-bg:var(--bs-secondary-bg);--bs-list-group-disabled-color:var(--bs-secondary-color);--bs-list-group-disabled-bg:var(--bs-body-bg);--bs-list-group-active-color:#fff;--bs-list-group-active-bg:#0d6efd;--bs-list-group-active-border-color:#0d6efd;display:flex;flex-direction:column;padding-left:0;margin-bottom:0;border-radius:var(--bs-list-group-border-radius)}.list-group-numbered{list-style-type:none;counter-reset:section}.list-group-numbered>.list-group-item::before{content:counters(section, ".") ". ";counter-increment:section}.list-group-item-action{width:100%;color:var(--bs-list-group-action-color);text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{z-index:1;color:var(--bs-list-group-action-hover-color);text-decoration:none;background-color:var(--bs-list-group-action-hover-bg)}.list-group-item-action:active{color:var(--bs-list-group-action-active-color);background-color:var(--bs-list-group-action-active-bg)}.list-group-item{position:relative;display:block;padding:var(--bs-list-group-item-padding-y) var(--bs-list-group-item-padding-x);color:var(--bs-list-group-color);text-decoration:none;background-color:var(--bs-list-group-bg);border:var(--bs-list-group-border-width) solid var(--bs-list-group-border-color)}.list-group-item:first-child{border-top-left-radius:inherit;border-top-right-radius:inherit}.list-group-item:last-child{border-bottom-right-radius:inherit;border-bottom-left-radius:inherit}.list-group-item.disabled,.list-group-item:disabled{color:var(--bs-list-group-disabled-color);pointer-events:none;background-color:var(--bs-list-group-disabled-bg)}.list-group-item.active{z-index:2;color:var(--bs-list-group-active-color);background-color:var(--bs-list-group-active-bg);border-color:var(--bs-list-group-active-border-color)}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:calc(-1 * var(--bs-list-group-border-width));border-top-width:var(--bs-list-group-border-width)}.list-group-horizontal{flex-direction:row}.list-group-horizontal>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal>.list-group-item.active{margin-top:0}.list-group-horizontal>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}@media (min-width:576px){.list-group-horizontal-sm{flex-direction:row}.list-group-horizontal-sm>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-sm>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-sm>.list-group-item.active{margin-top:0}.list-group-horizontal-sm>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-sm>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:768px){.list-group-horizontal-md{flex-direction:row}.list-group-horizontal-md>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-md>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-md>.list-group-item.active{margin-top:0}.list-group-horizontal-md>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-md>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:992px){.list-group-horizontal-lg{flex-direction:row}.list-group-horizontal-lg>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-lg>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-lg>.list-group-item.active{margin-top:0}.list-group-horizontal-lg>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-lg>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:1200px){.list-group-horizontal-xl{flex-direction:row}.list-group-horizontal-xl>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-xl>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-xl>.list-group-item.active{margin-top:0}.list-group-horizontal-xl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xl>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}@media (min-width:1400px){.list-group-horizontal-xxl{flex-direction:row}.list-group-horizontal-xxl>.list-group-item:first-child:not(:last-child){border-bottom-left-radius:var(--bs-list-group-border-radius);border-top-right-radius:0}.list-group-horizontal-xxl>.list-group-item:last-child:not(:first-child){border-top-right-radius:var(--bs-list-group-border-radius);border-bottom-left-radius:0}.list-group-horizontal-xxl>.list-group-item.active{margin-top:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item{border-top-width:var(--bs-list-group-border-width);border-left-width:0}.list-group-horizontal-xxl>.list-group-item+.list-group-item.active{margin-left:calc(-1 * var(--bs-list-group-border-width));border-left-width:var(--bs-list-group-border-width)}}.list-group-flush{border-radius:0}.list-group-flush>.list-group-item{border-width:0 0 var(--bs-list-group-border-width)}.list-group-flush>.list-group-item:last-child{border-bottom-width:0}.list-group-item-primary{--bs-list-group-color:var(--bs-primary-text-emphasis);--bs-list-group-bg:var(--bs-primary-bg-subtle);--bs-list-group-border-color:var(--bs-primary-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-primary-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-primary-border-subtle);--bs-list-group-active-color:var(--bs-primary-bg-subtle);--bs-list-group-active-bg:var(--bs-primary-text-emphasis);--bs-list-group-active-border-color:var(--bs-primary-text-emphasis)}.list-group-item-secondary{--bs-list-group-color:var(--bs-secondary-text-emphasis);--bs-list-group-bg:var(--bs-secondary-bg-subtle);--bs-list-group-border-color:var(--bs-secondary-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-secondary-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-secondary-border-subtle);--bs-list-group-active-color:var(--bs-secondary-bg-subtle);--bs-list-group-active-bg:var(--bs-secondary-text-emphasis);--bs-list-group-active-border-color:var(--bs-secondary-text-emphasis)}.list-group-item-success{--bs-list-group-color:var(--bs-success-text-emphasis);--bs-list-group-bg:var(--bs-success-bg-subtle);--bs-list-group-border-color:var(--bs-success-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-success-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-success-border-subtle);--bs-list-group-active-color:var(--bs-success-bg-subtle);--bs-list-group-active-bg:var(--bs-success-text-emphasis);--bs-list-group-active-border-color:var(--bs-success-text-emphasis)}.list-group-item-info{--bs-list-group-color:var(--bs-info-text-emphasis);--bs-list-group-bg:var(--bs-info-bg-subtle);--bs-list-group-border-color:var(--bs-info-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-info-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-info-border-subtle);--bs-list-group-active-color:var(--bs-info-bg-subtle);--bs-list-group-active-bg:var(--bs-info-text-emphasis);--bs-list-group-active-border-color:var(--bs-info-text-emphasis)}.list-group-item-warning{--bs-list-group-color:var(--bs-warning-text-emphasis);--bs-list-group-bg:var(--bs-warning-bg-subtle);--bs-list-group-border-color:var(--bs-warning-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-warning-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-warning-border-subtle);--bs-list-group-active-color:var(--bs-warning-bg-subtle);--bs-list-group-active-bg:var(--bs-warning-text-emphasis);--bs-list-group-active-border-color:var(--bs-warning-text-emphasis)}.list-group-item-danger{--bs-list-group-color:var(--bs-danger-text-emphasis);--bs-list-group-bg:var(--bs-danger-bg-subtle);--bs-list-group-border-color:var(--bs-danger-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-danger-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-danger-border-subtle);--bs-list-group-active-color:var(--bs-danger-bg-subtle);--bs-list-group-active-bg:var(--bs-danger-text-emphasis);--bs-list-group-active-border-color:var(--bs-danger-text-emphasis)}.list-group-item-light{--bs-list-group-color:var(--bs-light-text-emphasis);--bs-list-group-bg:var(--bs-light-bg-subtle);--bs-list-group-border-color:var(--bs-light-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-light-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-light-border-subtle);--bs-list-group-active-color:var(--bs-light-bg-subtle);--bs-list-group-active-bg:var(--bs-light-text-emphasis);--bs-list-group-active-border-color:var(--bs-light-text-emphasis)}.list-group-item-dark{--bs-list-group-color:var(--bs-dark-text-emphasis);--bs-list-group-bg:var(--bs-dark-bg-subtle);--bs-list-group-border-color:var(--bs-dark-border-subtle);--bs-list-group-action-hover-color:var(--bs-emphasis-color);--bs-list-group-action-hover-bg:var(--bs-dark-border-subtle);--bs-list-group-action-active-color:var(--bs-emphasis-color);--bs-list-group-action-active-bg:var(--bs-dark-border-subtle);--bs-list-group-active-color:var(--bs-dark-bg-subtle);--bs-list-group-active-bg:var(--bs-dark-text-emphasis);--bs-list-group-active-border-color:var(--bs-dark-text-emphasis)}.btn-close{--bs-btn-close-color:#000;--bs-btn-close-bg:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23000'%3e%3cpath d='M.293.293a1 1 0 0 1 1.414 0L8 6.586 14.293.293a1 1 0 1 1 1.414 1.414L9.414 8l6.293 6.293a1 1 0 0 1-1.414 1.414L8 9.414l-6.293 6.293a1 1 0 0 1-1.414-1.414L6.586 8 .293 1.707a1 1 0 0 1 0-1.414z'/%3e%3c/svg%3e");--bs-btn-close-opacity:0.5;--bs-btn-close-hover-opacity:0.75;--bs-btn-close-focus-shadow:0 0 0 0.25rem rgba(13, 110, 253, 0.25);--bs-btn-close-focus-opacity:1;--bs-btn-close-disabled-opacity:0.25;--bs-btn-close-white-filter:invert(1) grayscale(100%) brightness(200%);box-sizing:content-box;width:1em;height:1em;padding:.25em .25em;color:var(--bs-btn-close-color);background:transparent var(--bs-btn-close-bg) center/1em auto no-repeat;border:0;border-radius:.375rem;opacity:var(--bs-btn-close-opacity)}.btn-close:hover{color:var(--bs-btn-close-color);text-decoration:none;opacity:var(--bs-btn-close-hover-opacity)}.btn-close:focus{outline:0;box-shadow:var(--bs-btn-close-focus-shadow);opacity:var(--bs-btn-close-focus-opacity)}.btn-close.disabled,.btn-close:disabled{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none;opacity:var(--bs-btn-close-disabled-opacity)}.btn-close-white{filter:var(--bs-btn-close-white-filter)}[data-bs-theme=dark] .btn-close{filter:var(--bs-btn-close-white-filter)}.toast{--bs-toast-zindex:1090;--bs-toast-padding-x:0.75rem;--bs-toast-padding-y:0.5rem;--bs-toast-spacing:1.5rem;--bs-toast-max-width:350px;--bs-toast-font-size:0.875rem;--bs-toast-color: ;--bs-toast-bg:rgba(var(--bs-body-bg-rgb), 0.85);--bs-toast-border-width:var(--bs-border-width);--bs-toast-border-color:var(--bs-border-color-translucent);--bs-toast-border-radius:var(--bs-border-radius);--bs-toast-box-shadow:var(--bs-box-shadow);--bs-toast-header-color:var(--bs-secondary-color);--bs-toast-header-bg:rgba(var(--bs-body-bg-rgb), 0.85);--bs-toast-header-border-color:var(--bs-border-color-translucent);width:var(--bs-toast-max-width);max-width:100%;font-size:var(--bs-toast-font-size);color:var(--bs-toast-color);pointer-events:auto;background-color:var(--bs-toast-bg);background-clip:padding-box;border:var(--bs-toast-border-width) solid var(--bs-toast-border-color);box-shadow:var(--bs-toast-box-shadow);border-radius:var(--bs-toast-border-radius)}.toast.showing{opacity:0}.toast:not(.show){display:none}.toast-container{--bs-toast-zindex:1090;position:absolute;z-index:var(--bs-toast-zindex);width:-webkit-max-content;width:-moz-max-content;width:max-content;max-width:100%;pointer-events:none}.toast-container>:not(:last-child){margin-bottom:var(--bs-toast-spacing)}.toast-header{display:flex;align-items:center;padding:var(--bs-toast-padding-y) var(--bs-toast-padding-x);color:var(--bs-toast-header-color);background-color:var(--bs-toast-header-bg);background-clip:padding-box;border-bottom:var(--bs-toast-border-width) solid var(--bs-toast-header-border-color);border-top-left-radius:calc(var(--bs-toast-border-radius) - var(--bs-toast-border-width));border-top-right-radius:calc(var(--bs-toast-border-radius) - var(--bs-toast-border-width))}.toast-header .btn-close{margin-right:calc(-.5 * var(--bs-toast-padding-x));margin-left:var(--bs-toast-padding-x)}.toast-body{padding:var(--bs-toast-padding-x);word-wrap:break-word}.modal{--bs-modal-zindex:1055;--bs-modal-width:500px;--bs-modal-padding:1rem;--bs-modal-margin:0.5rem;--bs-modal-color: ;--bs-modal-bg:var(--bs-body-bg);--bs-modal-border-color:var(--bs-border-color-translucent);--bs-modal-border-width:var(--bs-border-width);--bs-modal-border-radius:var(--bs-border-radius-lg);--bs-modal-box-shadow:0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-modal-inner-border-radius:calc(var(--bs-border-radius-lg) - (var(--bs-border-width)));--bs-modal-header-padding-x:1rem;--bs-modal-header-padding-y:1rem;--bs-modal-header-padding:1rem 1rem;--bs-modal-header-border-color:var(--bs-border-color);--bs-modal-header-border-width:var(--bs-border-width);--bs-modal-title-line-height:1.5;--bs-modal-footer-gap:0.5rem;--bs-modal-footer-bg: ;--bs-modal-footer-border-color:var(--bs-border-color);--bs-modal-footer-border-width:var(--bs-border-width);position:fixed;top:0;left:0;z-index:var(--bs-modal-zindex);display:none;width:100%;height:100%;overflow-x:hidden;overflow-y:auto;outline:0}.modal-dialog{position:relative;width:auto;margin:var(--bs-modal-margin);pointer-events:none}.modal.fade .modal-dialog{transition:transform .3s ease-out;transform:translate(0,-50px)}@media (prefers-reduced-motion:reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{transform:none}.modal.modal-static .modal-dialog{transform:scale(1.02)}.modal-dialog-scrollable{height:calc(100% - var(--bs-modal-margin) * 2)}.modal-dialog-scrollable .modal-content{max-height:100%;overflow:hidden}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:flex;align-items:center;min-height:calc(100% - var(--bs-modal-margin) * 2)}.modal-content{position:relative;display:flex;flex-direction:column;width:100%;color:var(--bs-modal-color);pointer-events:auto;background-color:var(--bs-modal-bg);background-clip:padding-box;border:var(--bs-modal-border-width) solid var(--bs-modal-border-color);border-radius:var(--bs-modal-border-radius);outline:0}.modal-backdrop{--bs-backdrop-zindex:1050;--bs-backdrop-bg:#000;--bs-backdrop-opacity:0.5;position:fixed;top:0;left:0;z-index:var(--bs-backdrop-zindex);width:100vw;height:100vh;background-color:var(--bs-backdrop-bg)}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:var(--bs-backdrop-opacity)}.modal-header{display:flex;flex-shrink:0;align-items:center;justify-content:space-between;padding:var(--bs-modal-header-padding);border-bottom:var(--bs-modal-header-border-width) solid var(--bs-modal-header-border-color);border-top-left-radius:var(--bs-modal-inner-border-radius);border-top-right-radius:var(--bs-modal-inner-border-radius)}.modal-header .btn-close{padding:calc(var(--bs-modal-header-padding-y) * .5) calc(var(--bs-modal-header-padding-x) * .5);margin:calc(-.5 * var(--bs-modal-header-padding-y)) calc(-.5 * var(--bs-modal-header-padding-x)) calc(-.5 * var(--bs-modal-header-padding-y)) auto}.modal-title{margin-bottom:0;line-height:var(--bs-modal-title-line-height)}.modal-body{position:relative;flex:1 1 auto;padding:var(--bs-modal-padding)}.modal-footer{display:flex;flex-shrink:0;flex-wrap:wrap;align-items:center;justify-content:flex-end;padding:calc(var(--bs-modal-padding) - var(--bs-modal-footer-gap) * .5);background-color:var(--bs-modal-footer-bg);border-top:var(--bs-modal-footer-border-width) solid var(--bs-modal-footer-border-color);border-bottom-right-radius:var(--bs-modal-inner-border-radius);border-bottom-left-radius:var(--bs-modal-inner-border-radius)}.modal-footer>*{margin:calc(var(--bs-modal-footer-gap) * .5)}@media (min-width:576px){.modal{--bs-modal-margin:1.75rem;--bs-modal-box-shadow:0 0.5rem 1rem rgba(0, 0, 0, 0.15)}.modal-dialog{max-width:var(--bs-modal-width);margin-right:auto;margin-left:auto}.modal-sm{--bs-modal-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{--bs-modal-width:800px}}@media (min-width:1200px){.modal-xl{--bs-modal-width:1140px}}.modal-fullscreen{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen .modal-footer,.modal-fullscreen .modal-header{border-radius:0}.modal-fullscreen .modal-body{overflow-y:auto}@media (max-width:575.98px){.modal-fullscreen-sm-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-sm-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-sm-down .modal-footer,.modal-fullscreen-sm-down .modal-header{border-radius:0}.modal-fullscreen-sm-down .modal-body{overflow-y:auto}}@media (max-width:767.98px){.modal-fullscreen-md-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-md-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-md-down .modal-footer,.modal-fullscreen-md-down .modal-header{border-radius:0}.modal-fullscreen-md-down .modal-body{overflow-y:auto}}@media (max-width:991.98px){.modal-fullscreen-lg-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-lg-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-lg-down .modal-footer,.modal-fullscreen-lg-down .modal-header{border-radius:0}.modal-fullscreen-lg-down .modal-body{overflow-y:auto}}@media (max-width:1199.98px){.modal-fullscreen-xl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xl-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-xl-down .modal-footer,.modal-fullscreen-xl-down .modal-header{border-radius:0}.modal-fullscreen-xl-down .modal-body{overflow-y:auto}}@media (max-width:1399.98px){.modal-fullscreen-xxl-down{width:100vw;max-width:none;height:100%;margin:0}.modal-fullscreen-xxl-down .modal-content{height:100%;border:0;border-radius:0}.modal-fullscreen-xxl-down .modal-footer,.modal-fullscreen-xxl-down .modal-header{border-radius:0}.modal-fullscreen-xxl-down .modal-body{overflow-y:auto}}.tooltip{--bs-tooltip-zindex:1080;--bs-tooltip-max-width:200px;--bs-tooltip-padding-x:0.5rem;--bs-tooltip-padding-y:0.25rem;--bs-tooltip-margin: ;--bs-tooltip-font-size:0.875rem;--bs-tooltip-color:var(--bs-body-bg);--bs-tooltip-bg:var(--bs-emphasis-color);--bs-tooltip-border-radius:var(--bs-border-radius);--bs-tooltip-opacity:0.9;--bs-tooltip-arrow-width:0.8rem;--bs-tooltip-arrow-height:0.4rem;z-index:var(--bs-tooltip-zindex);display:block;margin:var(--bs-tooltip-margin);font-family:var(--bs-font-sans-serif);font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-tooltip-font-size);word-wrap:break-word;opacity:0}.tooltip.show{opacity:var(--bs-tooltip-opacity)}.tooltip .tooltip-arrow{display:block;width:var(--bs-tooltip-arrow-width);height:var(--bs-tooltip-arrow-height)}.tooltip .tooltip-arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow,.bs-tooltip-top .tooltip-arrow{bottom:calc(-1 * var(--bs-tooltip-arrow-height))}.bs-tooltip-auto[data-popper-placement^=top] .tooltip-arrow::before,.bs-tooltip-top .tooltip-arrow::before{top:-1px;border-width:var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width) * .5) 0;border-top-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow,.bs-tooltip-end .tooltip-arrow{left:calc(-1 * var(--bs-tooltip-arrow-height));width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-auto[data-popper-placement^=right] .tooltip-arrow::before,.bs-tooltip-end .tooltip-arrow::before{right:-1px;border-width:calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height) calc(var(--bs-tooltip-arrow-width) * .5) 0;border-right-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow,.bs-tooltip-bottom .tooltip-arrow{top:calc(-1 * var(--bs-tooltip-arrow-height))}.bs-tooltip-auto[data-popper-placement^=bottom] .tooltip-arrow::before,.bs-tooltip-bottom .tooltip-arrow::before{bottom:-1px;border-width:0 calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height);border-bottom-color:var(--bs-tooltip-bg)}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow,.bs-tooltip-start .tooltip-arrow{right:calc(-1 * var(--bs-tooltip-arrow-height));width:var(--bs-tooltip-arrow-height);height:var(--bs-tooltip-arrow-width)}.bs-tooltip-auto[data-popper-placement^=left] .tooltip-arrow::before,.bs-tooltip-start .tooltip-arrow::before{left:-1px;border-width:calc(var(--bs-tooltip-arrow-width) * .5) 0 calc(var(--bs-tooltip-arrow-width) * .5) var(--bs-tooltip-arrow-height);border-left-color:var(--bs-tooltip-bg)}.tooltip-inner{max-width:var(--bs-tooltip-max-width);padding:var(--bs-tooltip-padding-y) var(--bs-tooltip-padding-x);color:var(--bs-tooltip-color);text-align:center;background-color:var(--bs-tooltip-bg);border-radius:var(--bs-tooltip-border-radius)}.popover{--bs-popover-zindex:1070;--bs-popover-max-width:276px;--bs-popover-font-size:0.875rem;--bs-popover-bg:var(--bs-body-bg);--bs-popover-border-width:var(--bs-border-width);--bs-popover-border-color:var(--bs-border-color-translucent);--bs-popover-border-radius:var(--bs-border-radius-lg);--bs-popover-inner-border-radius:calc(var(--bs-border-radius-lg) - var(--bs-border-width));--bs-popover-box-shadow:0 0.5rem 1rem rgba(0, 0, 0, 0.15);--bs-popover-header-padding-x:1rem;--bs-popover-header-padding-y:0.5rem;--bs-popover-header-font-size:1rem;--bs-popover-header-color:inherit;--bs-popover-header-bg:var(--bs-secondary-bg);--bs-popover-body-padding-x:1rem;--bs-popover-body-padding-y:1rem;--bs-popover-body-color:var(--bs-body-color);--bs-popover-arrow-width:1rem;--bs-popover-arrow-height:0.5rem;--bs-popover-arrow-border:var(--bs-popover-border-color);z-index:var(--bs-popover-zindex);display:block;max-width:var(--bs-popover-max-width);font-family:var(--bs-font-sans-serif);font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;white-space:normal;word-spacing:normal;line-break:auto;font-size:var(--bs-popover-font-size);word-wrap:break-word;background-color:var(--bs-popover-bg);background-clip:padding-box;border:var(--bs-popover-border-width) solid var(--bs-popover-border-color);border-radius:var(--bs-popover-border-radius)}.popover .popover-arrow{display:block;width:var(--bs-popover-arrow-width);height:var(--bs-popover-arrow-height)}.popover .popover-arrow::after,.popover .popover-arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid;border-width:0}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow,.bs-popover-top>.popover-arrow{bottom:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::after,.bs-popover-top>.popover-arrow::before{border-width:var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width) * .5) 0}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::before,.bs-popover-top>.popover-arrow::before{bottom:0;border-top-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=top]>.popover-arrow::after,.bs-popover-top>.popover-arrow::after{bottom:var(--bs-popover-border-width);border-top-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow,.bs-popover-end>.popover-arrow{left:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::after,.bs-popover-end>.popover-arrow::before{border-width:calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height) calc(var(--bs-popover-arrow-width) * .5) 0}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::before,.bs-popover-end>.popover-arrow::before{left:0;border-right-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=right]>.popover-arrow::after,.bs-popover-end>.popover-arrow::after{left:var(--bs-popover-border-width);border-right-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow,.bs-popover-bottom>.popover-arrow{top:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width))}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::before{border-width:0 calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::before,.bs-popover-bottom>.popover-arrow::before{top:0;border-bottom-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=bottom]>.popover-arrow::after,.bs-popover-bottom>.popover-arrow::after{top:var(--bs-popover-border-width);border-bottom-color:var(--bs-popover-bg)}.bs-popover-auto[data-popper-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:var(--bs-popover-arrow-width);margin-left:calc(-.5 * var(--bs-popover-arrow-width));content:"";border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-header-bg)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow,.bs-popover-start>.popover-arrow{right:calc(-1 * (var(--bs-popover-arrow-height)) - var(--bs-popover-border-width));width:var(--bs-popover-arrow-height);height:var(--bs-popover-arrow-width)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::after,.bs-popover-start>.popover-arrow::before{border-width:calc(var(--bs-popover-arrow-width) * .5) 0 calc(var(--bs-popover-arrow-width) * .5) var(--bs-popover-arrow-height)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::before,.bs-popover-start>.popover-arrow::before{right:0;border-left-color:var(--bs-popover-arrow-border)}.bs-popover-auto[data-popper-placement^=left]>.popover-arrow::after,.bs-popover-start>.popover-arrow::after{right:var(--bs-popover-border-width);border-left-color:var(--bs-popover-bg)}.popover-header{padding:var(--bs-popover-header-padding-y) var(--bs-popover-header-padding-x);margin-bottom:0;font-size:var(--bs-popover-header-font-size);color:var(--bs-popover-header-color);background-color:var(--bs-popover-header-bg);border-bottom:var(--bs-popover-border-width) solid var(--bs-popover-border-color);border-top-left-radius:var(--bs-popover-inner-border-radius);border-top-right-radius:var(--bs-popover-inner-border-radius)}.popover-header:empty{display:none}.popover-body{padding:var(--bs-popover-body-padding-y) var(--bs-popover-body-padding-x);color:var(--bs-popover-body-color)}.carousel{position:relative}.carousel.pointer-event{touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden;transition:transform .6s ease-in-out}@media (prefers-reduced-motion:reduce){.carousel-item{transition:none}}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.active.carousel-item-end,.carousel-item-next:not(.carousel-item-start){transform:translateX(100%)}.active.carousel-item-start,.carousel-item-prev:not(.carousel-item-end){transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;transform:none}.carousel-fade .carousel-item-next.carousel-item-start,.carousel-fade .carousel-item-prev.carousel-item-end,.carousel-fade .carousel-item.active{z-index:1;opacity:1}.carousel-fade .active.carousel-item-end,.carousel-fade .active.carousel-item-start{z-index:0;opacity:0;transition:opacity 0s .6s}@media (prefers-reduced-motion:reduce){.carousel-fade .active.carousel-item-end,.carousel-fade .active.carousel-item-start{transition:none}}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;z-index:1;display:flex;align-items:center;justify-content:center;width:15%;padding:0;color:#fff;text-align:center;background:0 0;border:0;opacity:.5;transition:opacity .15s ease}@media (prefers-reduced-motion:reduce){.carousel-control-next,.carousel-control-prev{transition:none}}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:2rem;height:2rem;background-repeat:no-repeat;background-position:50%;background-size:100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M11.354 1.646a.5.5 0 0 1 0 .708L5.707 8l5.647 5.646a.5.5 0 0 1-.708.708l-6-6a.5.5 0 0 1 0-.708l6-6a.5.5 0 0 1 .708 0z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' fill='%23fff'%3e%3cpath d='M4.646 1.646a.5.5 0 0 1 .708 0l6 6a.5.5 0 0 1 0 .708l-6 6a.5.5 0 0 1-.708-.708L10.293 8 4.646 2.354a.5.5 0 0 1 0-.708z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:2;display:flex;justify-content:center;padding:0;margin-right:15%;margin-bottom:1rem;margin-left:15%}.carousel-indicators [data-bs-target]{box-sizing:content-box;flex:0 1 auto;width:30px;height:3px;padding:0;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border:0;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5;transition:opacity .6s ease}@media (prefers-reduced-motion:reduce){.carousel-indicators [data-bs-target]{transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:1.25rem;left:15%;padding-top:1.25rem;padding-bottom:1.25rem;color:#fff;text-align:center}.carousel-dark .carousel-control-next-icon,.carousel-dark .carousel-control-prev-icon{filter:invert(1) grayscale(100)}.carousel-dark .carousel-indicators [data-bs-target]{background-color:#000}.carousel-dark .carousel-caption{color:#000}[data-bs-theme=dark] .carousel .carousel-control-next-icon,[data-bs-theme=dark] .carousel .carousel-control-prev-icon,[data-bs-theme=dark].carousel .carousel-control-next-icon,[data-bs-theme=dark].carousel .carousel-control-prev-icon{filter:invert(1) grayscale(100)}[data-bs-theme=dark] .carousel .carousel-indicators [data-bs-target],[data-bs-theme=dark].carousel .carousel-indicators [data-bs-target]{background-color:#000}[data-bs-theme=dark] .carousel .carousel-caption,[data-bs-theme=dark].carousel .carousel-caption{color:#000}.spinner-border,.spinner-grow{display:inline-block;width:var(--bs-spinner-width);height:var(--bs-spinner-height);vertical-align:var(--bs-spinner-vertical-align);border-radius:50%;animation:var(--bs-spinner-animation-speed) linear infinite var(--bs-spinner-animation-name)}@keyframes spinner-border{to{transform:rotate(360deg)}}.spinner-border{--bs-spinner-width:2rem;--bs-spinner-height:2rem;--bs-spinner-vertical-align:-0.125em;--bs-spinner-border-width:0.25em;--bs-spinner-animation-speed:0.75s;--bs-spinner-animation-name:spinner-border;border:var(--bs-spinner-border-width) solid currentcolor;border-right-color:transparent}.spinner-border-sm{--bs-spinner-width:1rem;--bs-spinner-height:1rem;--bs-spinner-border-width:0.2em}@keyframes spinner-grow{0%{transform:scale(0)}50%{opacity:1;transform:none}}.spinner-grow{--bs-spinner-width:2rem;--bs-spinner-height:2rem;--bs-spinner-vertical-align:-0.125em;--bs-spinner-animation-speed:0.75s;--bs-spinner-animation-name:spinner-grow;background-color:currentcolor;opacity:0}.spinner-grow-sm{--bs-spinner-width:1rem;--bs-spinner-height:1rem}@media (prefers-reduced-motion:reduce){.spinner-border,.spinner-grow{--bs-spinner-animation-speed:1.5s}}.offcanvas,.offcanvas-lg,.offcanvas-md,.offcanvas-sm,.offcanvas-xl,.offcanvas-xxl{--bs-offcanvas-zindex:1045;--bs-offcanvas-width:400px;--bs-offcanvas-height:30vh;--bs-offcanvas-padding-x:1rem;--bs-offcanvas-padding-y:1rem;--bs-offcanvas-color:var(--bs-body-color);--bs-offcanvas-bg:var(--bs-body-bg);--bs-offcanvas-border-width:var(--bs-border-width);--bs-offcanvas-border-color:var(--bs-border-color-translucent);--bs-offcanvas-box-shadow:0 0.125rem 0.25rem rgba(0, 0, 0, 0.075);--bs-offcanvas-transition:transform 0.3s ease-in-out;--bs-offcanvas-title-line-height:1.5}@media (max-width:575.98px){.offcanvas-sm{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:575.98px) and (prefers-reduced-motion:reduce){.offcanvas-sm{transition:none}}@media (max-width:575.98px){.offcanvas-sm.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-sm.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-sm.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-sm.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-sm.show:not(.hiding),.offcanvas-sm.showing{transform:none}.offcanvas-sm.hiding,.offcanvas-sm.show,.offcanvas-sm.showing{visibility:visible}}@media (min-width:576px){.offcanvas-sm{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-sm .offcanvas-header{display:none}.offcanvas-sm .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:767.98px){.offcanvas-md{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:767.98px) and (prefers-reduced-motion:reduce){.offcanvas-md{transition:none}}@media (max-width:767.98px){.offcanvas-md.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-md.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-md.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-md.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-md.show:not(.hiding),.offcanvas-md.showing{transform:none}.offcanvas-md.hiding,.offcanvas-md.show,.offcanvas-md.showing{visibility:visible}}@media (min-width:768px){.offcanvas-md{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-md .offcanvas-header{display:none}.offcanvas-md .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:991.98px){.offcanvas-lg{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:991.98px) and (prefers-reduced-motion:reduce){.offcanvas-lg{transition:none}}@media (max-width:991.98px){.offcanvas-lg.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-lg.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-lg.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-lg.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-lg.show:not(.hiding),.offcanvas-lg.showing{transform:none}.offcanvas-lg.hiding,.offcanvas-lg.show,.offcanvas-lg.showing{visibility:visible}}@media (min-width:992px){.offcanvas-lg{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-lg .offcanvas-header{display:none}.offcanvas-lg .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:1199.98px){.offcanvas-xl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:1199.98px) and (prefers-reduced-motion:reduce){.offcanvas-xl{transition:none}}@media (max-width:1199.98px){.offcanvas-xl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xl.show:not(.hiding),.offcanvas-xl.showing{transform:none}.offcanvas-xl.hiding,.offcanvas-xl.show,.offcanvas-xl.showing{visibility:visible}}@media (min-width:1200px){.offcanvas-xl{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-xl .offcanvas-header{display:none}.offcanvas-xl .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}@media (max-width:1399.98px){.offcanvas-xxl{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}}@media (max-width:1399.98px) and (prefers-reduced-motion:reduce){.offcanvas-xxl{transition:none}}@media (max-width:1399.98px){.offcanvas-xxl.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas-xxl.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas-xxl.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas-xxl.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas-xxl.show:not(.hiding),.offcanvas-xxl.showing{transform:none}.offcanvas-xxl.hiding,.offcanvas-xxl.show,.offcanvas-xxl.showing{visibility:visible}}@media (min-width:1400px){.offcanvas-xxl{--bs-offcanvas-height:auto;--bs-offcanvas-border-width:0;background-color:transparent!important}.offcanvas-xxl .offcanvas-header{display:none}.offcanvas-xxl .offcanvas-body{display:flex;flex-grow:0;padding:0;overflow-y:visible;background-color:transparent!important}}.offcanvas{position:fixed;bottom:0;z-index:var(--bs-offcanvas-zindex);display:flex;flex-direction:column;max-width:100%;color:var(--bs-offcanvas-color);visibility:hidden;background-color:var(--bs-offcanvas-bg);background-clip:padding-box;outline:0;transition:var(--bs-offcanvas-transition)}@media (prefers-reduced-motion:reduce){.offcanvas{transition:none}}.offcanvas.offcanvas-start{top:0;left:0;width:var(--bs-offcanvas-width);border-right:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(-100%)}.offcanvas.offcanvas-end{top:0;right:0;width:var(--bs-offcanvas-width);border-left:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateX(100%)}.offcanvas.offcanvas-top{top:0;right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-bottom:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(-100%)}.offcanvas.offcanvas-bottom{right:0;left:0;height:var(--bs-offcanvas-height);max-height:100%;border-top:var(--bs-offcanvas-border-width) solid var(--bs-offcanvas-border-color);transform:translateY(100%)}.offcanvas.show:not(.hiding),.offcanvas.showing{transform:none}.offcanvas.hiding,.offcanvas.show,.offcanvas.showing{visibility:visible}.offcanvas-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.offcanvas-backdrop.fade{opacity:0}.offcanvas-backdrop.show{opacity:.5}.offcanvas-header{display:flex;align-items:center;justify-content:space-between;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x)}.offcanvas-header .btn-close{padding:calc(var(--bs-offcanvas-padding-y) * .5) calc(var(--bs-offcanvas-padding-x) * .5);margin-top:calc(-.5 * var(--bs-offcanvas-padding-y));margin-right:calc(-.5 * var(--bs-offcanvas-padding-x));margin-bottom:calc(-.5 * var(--bs-offcanvas-padding-y))}.offcanvas-title{margin-bottom:0;line-height:var(--bs-offcanvas-title-line-height)}.offcanvas-body{flex-grow:1;padding:var(--bs-offcanvas-padding-y) var(--bs-offcanvas-padding-x);overflow-y:auto}.placeholder{display:inline-block;min-height:1em;vertical-align:middle;cursor:wait;background-color:currentcolor;opacity:.5}.placeholder.btn::before{display:inline-block;content:""}.placeholder-xs{min-height:.6em}.placeholder-sm{min-height:.8em}.placeholder-lg{min-height:1.2em}.placeholder-glow .placeholder{animation:placeholder-glow 2s ease-in-out infinite}@keyframes placeholder-glow{50%{opacity:.2}}.placeholder-wave{-webkit-mask-image:linear-gradient(130deg,#000 55%,rgba(0,0,0,0.8) 75%,#000 95%);mask-image:linear-gradient(130deg,#000 55%,rgba(0,0,0,0.8) 75%,#000 95%);-webkit-mask-size:200% 100%;mask-size:200% 100%;animation:placeholder-wave 2s linear infinite}@keyframes placeholder-wave{100%{-webkit-mask-position:-200% 0%;mask-position:-200% 0%}}.clearfix::after{display:block;clear:both;content:""}.text-bg-primary{color:#fff!important;background-color:RGBA(var(--bs-primary-rgb),var(--bs-bg-opacity,1))!important}.text-bg-secondary{color:#fff!important;background-color:RGBA(var(--bs-secondary-rgb),var(--bs-bg-opacity,1))!important}.text-bg-success{color:#fff!important;background-color:RGBA(var(--bs-success-rgb),var(--bs-bg-opacity,1))!important}.text-bg-info{color:#000!important;background-color:RGBA(var(--bs-info-rgb),var(--bs-bg-opacity,1))!important}.text-bg-warning{color:#000!important;background-color:RGBA(var(--bs-warning-rgb),var(--bs-bg-opacity,1))!important}.text-bg-danger{color:#fff!important;background-color:RGBA(var(--bs-danger-rgb),var(--bs-bg-opacity,1))!important}.text-bg-light{color:#000!important;background-color:RGBA(var(--bs-light-rgb),var(--bs-bg-opacity,1))!important}.text-bg-dark{color:#fff!important;background-color:RGBA(var(--bs-dark-rgb),var(--bs-bg-opacity,1))!important}.link-primary{color:RGBA(var(--bs-primary-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-primary-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-primary-rgb),var(--bs-link-underline-opacity,1))!important}.link-primary:focus,.link-primary:hover{color:RGBA(10,88,202,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(10,88,202,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(10,88,202,var(--bs-link-underline-opacity,1))!important}.link-secondary{color:RGBA(var(--bs-secondary-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-secondary-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-secondary-rgb),var(--bs-link-underline-opacity,1))!important}.link-secondary:focus,.link-secondary:hover{color:RGBA(86,94,100,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(86,94,100,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(86,94,100,var(--bs-link-underline-opacity,1))!important}.link-success{color:RGBA(var(--bs-success-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-success-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-success-rgb),var(--bs-link-underline-opacity,1))!important}.link-success:focus,.link-success:hover{color:RGBA(20,108,67,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(20,108,67,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(20,108,67,var(--bs-link-underline-opacity,1))!important}.link-info{color:RGBA(var(--bs-info-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-info-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-info-rgb),var(--bs-link-underline-opacity,1))!important}.link-info:focus,.link-info:hover{color:RGBA(61,213,243,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(61,213,243,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(61,213,243,var(--bs-link-underline-opacity,1))!important}.link-warning{color:RGBA(var(--bs-warning-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-warning-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-warning-rgb),var(--bs-link-underline-opacity,1))!important}.link-warning:focus,.link-warning:hover{color:RGBA(255,205,57,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(255,205,57,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(255,205,57,var(--bs-link-underline-opacity,1))!important}.link-danger{color:RGBA(var(--bs-danger-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-danger-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-danger-rgb),var(--bs-link-underline-opacity,1))!important}.link-danger:focus,.link-danger:hover{color:RGBA(176,42,55,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(176,42,55,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(176,42,55,var(--bs-link-underline-opacity,1))!important}.link-light{color:RGBA(var(--bs-light-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-light-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-light-rgb),var(--bs-link-underline-opacity,1))!important}.link-light:focus,.link-light:hover{color:RGBA(249,250,251,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(249,250,251,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(249,250,251,var(--bs-link-underline-opacity,1))!important}.link-dark{color:RGBA(var(--bs-dark-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-dark-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-dark-rgb),var(--bs-link-underline-opacity,1))!important}.link-dark:focus,.link-dark:hover{color:RGBA(26,30,33,var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(26,30,33,var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(26,30,33,var(--bs-link-underline-opacity,1))!important}.link-body-emphasis{color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-opacity,1))!important;-webkit-text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,1))!important}.link-body-emphasis:focus,.link-body-emphasis:hover{color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-opacity,.75))!important;-webkit-text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,0.75))!important;text-decoration-color:RGBA(var(--bs-emphasis-color-rgb),var(--bs-link-underline-opacity,0.75))!important}.focus-ring:focus{outline:0;box-shadow:var(--bs-focus-ring-x,0) var(--bs-focus-ring-y,0) var(--bs-focus-ring-blur,0) var(--bs-focus-ring-width) var(--bs-focus-ring-color)}.icon-link{display:inline-flex;gap:.375rem;align-items:center;-webkit-text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-opacity,0.5));text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-opacity,0.5));text-underline-offset:0.25em;-webkit-backface-visibility:hidden;backface-visibility:hidden}.icon-link>.bi{flex-shrink:0;width:1em;height:1em;fill:currentcolor;transition:.2s ease-in-out transform}@media (prefers-reduced-motion:reduce){.icon-link>.bi{transition:none}}.icon-link-hover:focus-visible>.bi,.icon-link-hover:hover>.bi{transform:var(--bs-icon-link-transform,translate3d(.25em,0,0))}.ratio{position:relative;width:100%}.ratio::before{display:block;padding-top:var(--bs-aspect-ratio);content:""}.ratio>*{position:absolute;top:0;left:0;width:100%;height:100%}.ratio-1x1{--bs-aspect-ratio:100%}.ratio-4x3{--bs-aspect-ratio:75%}.ratio-16x9{--bs-aspect-ratio:56.25%}.ratio-21x9{--bs-aspect-ratio:42.8571428571%}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}@media (min-width:576px){.sticky-sm-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-sm-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:768px){.sticky-md-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-md-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:992px){.sticky-lg-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-lg-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:1200px){.sticky-xl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-xl-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}@media (min-width:1400px){.sticky-xxl-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}.sticky-xxl-bottom{position:-webkit-sticky;position:sticky;bottom:0;z-index:1020}}.hstack{display:flex;flex-direction:row;align-items:center;align-self:stretch}.vstack{display:flex;flex:1 1 auto;flex-direction:column;align-self:stretch}.visually-hidden,.visually-hidden-focusable:not(:focus):not(:focus-within){width:1px!important;height:1px!important;padding:0!important;margin:-1px!important;overflow:hidden!important;clip:rect(0,0,0,0)!important;white-space:nowrap!important;border:0!important}.visually-hidden-focusable:not(:focus):not(:focus-within):not(caption),.visually-hidden:not(caption){position:absolute!important}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;content:""}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.vr{display:inline-block;align-self:stretch;width:var(--bs-border-width);min-height:1em;background-color:currentcolor;opacity:.25}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.float-start{float:left!important}.float-end{float:right!important}.float-none{float:none!important}.object-fit-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-none{-o-object-fit:none!important;object-fit:none!important}.opacity-0{opacity:0!important}.opacity-25{opacity:.25!important}.opacity-50{opacity:.5!important}.opacity-75{opacity:.75!important}.opacity-100{opacity:1!important}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.overflow-visible{overflow:visible!important}.overflow-scroll{overflow:scroll!important}.overflow-x-auto{overflow-x:auto!important}.overflow-x-hidden{overflow-x:hidden!important}.overflow-x-visible{overflow-x:visible!important}.overflow-x-scroll{overflow-x:scroll!important}.overflow-y-auto{overflow-y:auto!important}.overflow-y-hidden{overflow-y:hidden!important}.overflow-y-visible{overflow-y:visible!important}.overflow-y-scroll{overflow-y:scroll!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-grid{display:grid!important}.d-inline-grid{display:inline-grid!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:flex!important}.d-inline-flex{display:inline-flex!important}.d-none{display:none!important}.shadow{box-shadow:0 .5rem 1rem rgba(0,0,0,.15)!important}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175)!important}.shadow-none{box-shadow:none!important}.focus-ring-primary{--bs-focus-ring-color:rgba(var(--bs-primary-rgb), var(--bs-focus-ring-opacity))}.focus-ring-secondary{--bs-focus-ring-color:rgba(var(--bs-secondary-rgb), var(--bs-focus-ring-opacity))}.focus-ring-success{--bs-focus-ring-color:rgba(var(--bs-success-rgb), var(--bs-focus-ring-opacity))}.focus-ring-info{--bs-focus-ring-color:rgba(var(--bs-info-rgb), var(--bs-focus-ring-opacity))}.focus-ring-warning{--bs-focus-ring-color:rgba(var(--bs-warning-rgb), var(--bs-focus-ring-opacity))}.focus-ring-danger{--bs-focus-ring-color:rgba(var(--bs-danger-rgb), var(--bs-focus-ring-opacity))}.focus-ring-light{--bs-focus-ring-color:rgba(var(--bs-light-rgb), var(--bs-focus-ring-opacity))}.focus-ring-dark{--bs-focus-ring-color:rgba(var(--bs-dark-rgb), var(--bs-focus-ring-opacity))}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.top-0{top:0!important}.top-50{top:50%!important}.top-100{top:100%!important}.bottom-0{bottom:0!important}.bottom-50{bottom:50%!important}.bottom-100{bottom:100%!important}.start-0{left:0!important}.start-50{left:50%!important}.start-100{left:100%!important}.end-0{right:0!important}.end-50{right:50%!important}.end-100{right:100%!important}.translate-middle{transform:translate(-50%,-50%)!important}.translate-middle-x{transform:translateX(-50%)!important}.translate-middle-y{transform:translateY(-50%)!important}.border{border:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-0{border:0!important}.border-top{border-top:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-top-0{border-top:0!important}.border-end{border-right:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-end-0{border-right:0!important}.border-bottom{border-bottom:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-bottom-0{border-bottom:0!important}.border-start{border-left:var(--bs-border-width) var(--bs-border-style) var(--bs-border-color)!important}.border-start-0{border-left:0!important}.border-primary{--bs-border-opacity:1;border-color:rgba(var(--bs-primary-rgb),var(--bs-border-opacity))!important}.border-secondary{--bs-border-opacity:1;border-color:rgba(var(--bs-secondary-rgb),var(--bs-border-opacity))!important}.border-success{--bs-border-opacity:1;border-color:rgba(var(--bs-success-rgb),var(--bs-border-opacity))!important}.border-info{--bs-border-opacity:1;border-color:rgba(var(--bs-info-rgb),var(--bs-border-opacity))!important}.border-warning{--bs-border-opacity:1;border-color:rgba(var(--bs-warning-rgb),var(--bs-border-opacity))!important}.border-danger{--bs-border-opacity:1;border-color:rgba(var(--bs-danger-rgb),var(--bs-border-opacity))!important}.border-light{--bs-border-opacity:1;border-color:rgba(var(--bs-light-rgb),var(--bs-border-opacity))!important}.border-dark{--bs-border-opacity:1;border-color:rgba(var(--bs-dark-rgb),var(--bs-border-opacity))!important}.border-black{--bs-border-opacity:1;border-color:rgba(var(--bs-black-rgb),var(--bs-border-opacity))!important}.border-white{--bs-border-opacity:1;border-color:rgba(var(--bs-white-rgb),var(--bs-border-opacity))!important}.border-primary-subtle{border-color:var(--bs-primary-border-subtle)!important}.border-secondary-subtle{border-color:var(--bs-secondary-border-subtle)!important}.border-success-subtle{border-color:var(--bs-success-border-subtle)!important}.border-info-subtle{border-color:var(--bs-info-border-subtle)!important}.border-warning-subtle{border-color:var(--bs-warning-border-subtle)!important}.border-danger-subtle{border-color:var(--bs-danger-border-subtle)!important}.border-light-subtle{border-color:var(--bs-light-border-subtle)!important}.border-dark-subtle{border-color:var(--bs-dark-border-subtle)!important}.border-1{border-width:1px!important}.border-2{border-width:2px!important}.border-3{border-width:3px!important}.border-4{border-width:4px!important}.border-5{border-width:5px!important}.border-opacity-10{--bs-border-opacity:0.1}.border-opacity-25{--bs-border-opacity:0.25}.border-opacity-50{--bs-border-opacity:0.5}.border-opacity-75{--bs-border-opacity:0.75}.border-opacity-100{--bs-border-opacity:1}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.mw-100{max-width:100%!important}.vw-100{width:100vw!important}.min-vw-100{min-width:100vw!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mh-100{max-height:100%!important}.vh-100{height:100vh!important}.min-vh-100{min-height:100vh!important}.flex-fill{flex:1 1 auto!important}.flex-row{flex-direction:row!important}.flex-column{flex-direction:column!important}.flex-row-reverse{flex-direction:row-reverse!important}.flex-column-reverse{flex-direction:column-reverse!important}.flex-grow-0{flex-grow:0!important}.flex-grow-1{flex-grow:1!important}.flex-shrink-0{flex-shrink:0!important}.flex-shrink-1{flex-shrink:1!important}.flex-wrap{flex-wrap:wrap!important}.flex-nowrap{flex-wrap:nowrap!important}.flex-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-start{justify-content:flex-start!important}.justify-content-end{justify-content:flex-end!important}.justify-content-center{justify-content:center!important}.justify-content-between{justify-content:space-between!important}.justify-content-around{justify-content:space-around!important}.justify-content-evenly{justify-content:space-evenly!important}.align-items-start{align-items:flex-start!important}.align-items-end{align-items:flex-end!important}.align-items-center{align-items:center!important}.align-items-baseline{align-items:baseline!important}.align-items-stretch{align-items:stretch!important}.align-content-start{align-content:flex-start!important}.align-content-end{align-content:flex-end!important}.align-content-center{align-content:center!important}.align-content-between{align-content:space-between!important}.align-content-around{align-content:space-around!important}.align-content-stretch{align-content:stretch!important}.align-self-auto{align-self:auto!important}.align-self-start{align-self:flex-start!important}.align-self-end{align-self:flex-end!important}.align-self-center{align-self:center!important}.align-self-baseline{align-self:baseline!important}.align-self-stretch{align-self:stretch!important}.order-first{order:-1!important}.order-0{order:0!important}.order-1{order:1!important}.order-2{order:2!important}.order-3{order:3!important}.order-4{order:4!important}.order-5{order:5!important}.order-last{order:6!important}.m-0{margin:0!important}.m-1{margin:.25rem!important}.m-2{margin:.5rem!important}.m-3{margin:1rem!important}.m-4{margin:1.5rem!important}.m-5{margin:3rem!important}.m-auto{margin:auto!important}.mx-0{margin-right:0!important;margin-left:0!important}.mx-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-3{margin-right:1rem!important;margin-left:1rem!important}.mx-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-5{margin-right:3rem!important;margin-left:3rem!important}.mx-auto{margin-right:auto!important;margin-left:auto!important}.my-0{margin-top:0!important;margin-bottom:0!important}.my-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-0{margin-top:0!important}.mt-1{margin-top:.25rem!important}.mt-2{margin-top:.5rem!important}.mt-3{margin-top:1rem!important}.mt-4{margin-top:1.5rem!important}.mt-5{margin-top:3rem!important}.mt-auto{margin-top:auto!important}.me-0{margin-right:0!important}.me-1{margin-right:.25rem!important}.me-2{margin-right:.5rem!important}.me-3{margin-right:1rem!important}.me-4{margin-right:1.5rem!important}.me-5{margin-right:3rem!important}.me-auto{margin-right:auto!important}.mb-0{margin-bottom:0!important}.mb-1{margin-bottom:.25rem!important}.mb-2{margin-bottom:.5rem!important}.mb-3{margin-bottom:1rem!important}.mb-4{margin-bottom:1.5rem!important}.mb-5{margin-bottom:3rem!important}.mb-auto{margin-bottom:auto!important}.ms-0{margin-left:0!important}.ms-1{margin-left:.25rem!important}.ms-2{margin-left:.5rem!important}.ms-3{margin-left:1rem!important}.ms-4{margin-left:1.5rem!important}.ms-5{margin-left:3rem!important}.ms-auto{margin-left:auto!important}.p-0{padding:0!important}.p-1{padding:.25rem!important}.p-2{padding:.5rem!important}.p-3{padding:1rem!important}.p-4{padding:1.5rem!important}.p-5{padding:3rem!important}.px-0{padding-right:0!important;padding-left:0!important}.px-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-3{padding-right:1rem!important;padding-left:1rem!important}.px-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-5{padding-right:3rem!important;padding-left:3rem!important}.py-0{padding-top:0!important;padding-bottom:0!important}.py-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-0{padding-top:0!important}.pt-1{padding-top:.25rem!important}.pt-2{padding-top:.5rem!important}.pt-3{padding-top:1rem!important}.pt-4{padding-top:1.5rem!important}.pt-5{padding-top:3rem!important}.pe-0{padding-right:0!important}.pe-1{padding-right:.25rem!important}.pe-2{padding-right:.5rem!important}.pe-3{padding-right:1rem!important}.pe-4{padding-right:1.5rem!important}.pe-5{padding-right:3rem!important}.pb-0{padding-bottom:0!important}.pb-1{padding-bottom:.25rem!important}.pb-2{padding-bottom:.5rem!important}.pb-3{padding-bottom:1rem!important}.pb-4{padding-bottom:1.5rem!important}.pb-5{padding-bottom:3rem!important}.ps-0{padding-left:0!important}.ps-1{padding-left:.25rem!important}.ps-2{padding-left:.5rem!important}.ps-3{padding-left:1rem!important}.ps-4{padding-left:1.5rem!important}.ps-5{padding-left:3rem!important}.gap-0{gap:0!important}.gap-1{gap:.25rem!important}.gap-2{gap:.5rem!important}.gap-3{gap:1rem!important}.gap-4{gap:1.5rem!important}.gap-5{gap:3rem!important}.row-gap-0{row-gap:0!important}.row-gap-1{row-gap:.25rem!important}.row-gap-2{row-gap:.5rem!important}.row-gap-3{row-gap:1rem!important}.row-gap-4{row-gap:1.5rem!important}.row-gap-5{row-gap:3rem!important}.column-gap-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.font-monospace{font-family:var(--bs-font-monospace)!important}.fs-1{font-size:calc(1.375rem + 1.5vw)!important}.fs-2{font-size:calc(1.325rem + .9vw)!important}.fs-3{font-size:calc(1.3rem + .6vw)!important}.fs-4{font-size:calc(1.275rem + .3vw)!important}.fs-5{font-size:1.25rem!important}.fs-6{font-size:1rem!important}.fst-italic{font-style:italic!important}.fst-normal{font-style:normal!important}.fw-lighter{font-weight:lighter!important}.fw-light{font-weight:300!important}.fw-normal{font-weight:400!important}.fw-medium{font-weight:500!important}.fw-semibold{font-weight:600!important}.fw-bold{font-weight:700!important}.fw-bolder{font-weight:bolder!important}.lh-1{line-height:1!important}.lh-sm{line-height:1.25!important}.lh-base{line-height:1.5!important}.lh-lg{line-height:2!important}.text-start{text-align:left!important}.text-end{text-align:right!important}.text-center{text-align:center!important}.text-decoration-none{text-decoration:none!important}.text-decoration-underline{text-decoration:underline!important}.text-decoration-line-through{text-decoration:line-through!important}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.text-wrap{white-space:normal!important}.text-nowrap{white-space:nowrap!important}.text-break{word-wrap:break-word!important;word-break:break-word!important}.text-primary{--bs-text-opacity:1;color:rgba(var(--bs-primary-rgb),var(--bs-text-opacity))!important}.text-secondary{--bs-text-opacity:1;color:rgba(var(--bs-secondary-rgb),var(--bs-text-opacity))!important}.text-success{--bs-text-opacity:1;color:rgba(var(--bs-success-rgb),var(--bs-text-opacity))!important}.text-info{--bs-text-opacity:1;color:rgba(var(--bs-info-rgb),var(--bs-text-opacity))!important}.text-warning{--bs-text-opacity:1;color:rgba(var(--bs-warning-rgb),var(--bs-text-opacity))!important}.text-danger{--bs-text-opacity:1;color:rgba(var(--bs-danger-rgb),var(--bs-text-opacity))!important}.text-light{--bs-text-opacity:1;color:rgba(var(--bs-light-rgb),var(--bs-text-opacity))!important}.text-dark{--bs-text-opacity:1;color:rgba(var(--bs-dark-rgb),var(--bs-text-opacity))!important}.text-black{--bs-text-opacity:1;color:rgba(var(--bs-black-rgb),var(--bs-text-opacity))!important}.text-white{--bs-text-opacity:1;color:rgba(var(--bs-white-rgb),var(--bs-text-opacity))!important}.text-body{--bs-text-opacity:1;color:rgba(var(--bs-body-color-rgb),var(--bs-text-opacity))!important}.text-muted{--bs-text-opacity:1;color:var(--bs-secondary-color)!important}.text-black-50{--bs-text-opacity:1;color:rgba(0,0,0,.5)!important}.text-white-50{--bs-text-opacity:1;color:rgba(255,255,255,.5)!important}.text-body-secondary{--bs-text-opacity:1;color:var(--bs-secondary-color)!important}.text-body-tertiary{--bs-text-opacity:1;color:var(--bs-tertiary-color)!important}.text-body-emphasis{--bs-text-opacity:1;color:var(--bs-emphasis-color)!important}.text-reset{--bs-text-opacity:1;color:inherit!important}.text-opacity-25{--bs-text-opacity:0.25}.text-opacity-50{--bs-text-opacity:0.5}.text-opacity-75{--bs-text-opacity:0.75}.text-opacity-100{--bs-text-opacity:1}.text-primary-emphasis{color:var(--bs-primary-text-emphasis)!important}.text-secondary-emphasis{color:var(--bs-secondary-text-emphasis)!important}.text-success-emphasis{color:var(--bs-success-text-emphasis)!important}.text-info-emphasis{color:var(--bs-info-text-emphasis)!important}.text-warning-emphasis{color:var(--bs-warning-text-emphasis)!important}.text-danger-emphasis{color:var(--bs-danger-text-emphasis)!important}.text-light-emphasis{color:var(--bs-light-text-emphasis)!important}.text-dark-emphasis{color:var(--bs-dark-text-emphasis)!important}.link-opacity-10{--bs-link-opacity:0.1}.link-opacity-10-hover:hover{--bs-link-opacity:0.1}.link-opacity-25{--bs-link-opacity:0.25}.link-opacity-25-hover:hover{--bs-link-opacity:0.25}.link-opacity-50{--bs-link-opacity:0.5}.link-opacity-50-hover:hover{--bs-link-opacity:0.5}.link-opacity-75{--bs-link-opacity:0.75}.link-opacity-75-hover:hover{--bs-link-opacity:0.75}.link-opacity-100{--bs-link-opacity:1}.link-opacity-100-hover:hover{--bs-link-opacity:1}.link-offset-1{text-underline-offset:0.125em!important}.link-offset-1-hover:hover{text-underline-offset:0.125em!important}.link-offset-2{text-underline-offset:0.25em!important}.link-offset-2-hover:hover{text-underline-offset:0.25em!important}.link-offset-3{text-underline-offset:0.375em!important}.link-offset-3-hover:hover{text-underline-offset:0.375em!important}.link-underline-primary{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-primary-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-primary-rgb),var(--bs-link-underline-opacity))!important}.link-underline-secondary{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-secondary-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-secondary-rgb),var(--bs-link-underline-opacity))!important}.link-underline-success{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-success-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-success-rgb),var(--bs-link-underline-opacity))!important}.link-underline-info{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-info-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-info-rgb),var(--bs-link-underline-opacity))!important}.link-underline-warning{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-warning-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-warning-rgb),var(--bs-link-underline-opacity))!important}.link-underline-danger{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-danger-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-danger-rgb),var(--bs-link-underline-opacity))!important}.link-underline-light{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-light-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-light-rgb),var(--bs-link-underline-opacity))!important}.link-underline-dark{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-dark-rgb),var(--bs-link-underline-opacity))!important;text-decoration-color:rgba(var(--bs-dark-rgb),var(--bs-link-underline-opacity))!important}.link-underline{--bs-link-underline-opacity:1;-webkit-text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-underline-opacity,1))!important;text-decoration-color:rgba(var(--bs-link-color-rgb),var(--bs-link-underline-opacity,1))!important}.link-underline-opacity-0{--bs-link-underline-opacity:0}.link-underline-opacity-0-hover:hover{--bs-link-underline-opacity:0}.link-underline-opacity-10{--bs-link-underline-opacity:0.1}.link-underline-opacity-10-hover:hover{--bs-link-underline-opacity:0.1}.link-underline-opacity-25{--bs-link-underline-opacity:0.25}.link-underline-opacity-25-hover:hover{--bs-link-underline-opacity:0.25}.link-underline-opacity-50{--bs-link-underline-opacity:0.5}.link-underline-opacity-50-hover:hover{--bs-link-underline-opacity:0.5}.link-underline-opacity-75{--bs-link-underline-opacity:0.75}.link-underline-opacity-75-hover:hover{--bs-link-underline-opacity:0.75}.link-underline-opacity-100{--bs-link-underline-opacity:1}.link-underline-opacity-100-hover:hover{--bs-link-underline-opacity:1}.bg-primary{--bs-bg-opacity:1;background-color:rgba(var(--bs-primary-rgb),var(--bs-bg-opacity))!important}.bg-secondary{--bs-bg-opacity:1;background-color:rgba(var(--bs-secondary-rgb),var(--bs-bg-opacity))!important}.bg-success{--bs-bg-opacity:1;background-color:rgba(var(--bs-success-rgb),var(--bs-bg-opacity))!important}.bg-info{--bs-bg-opacity:1;background-color:rgba(var(--bs-info-rgb),var(--bs-bg-opacity))!important}.bg-warning{--bs-bg-opacity:1;background-color:rgba(var(--bs-warning-rgb),var(--bs-bg-opacity))!important}.bg-danger{--bs-bg-opacity:1;background-color:rgba(var(--bs-danger-rgb),var(--bs-bg-opacity))!important}.bg-light{--bs-bg-opacity:1;background-color:rgba(var(--bs-light-rgb),var(--bs-bg-opacity))!important}.bg-dark{--bs-bg-opacity:1;background-color:rgba(var(--bs-dark-rgb),var(--bs-bg-opacity))!important}.bg-black{--bs-bg-opacity:1;background-color:rgba(var(--bs-black-rgb),var(--bs-bg-opacity))!important}.bg-white{--bs-bg-opacity:1;background-color:rgba(var(--bs-white-rgb),var(--bs-bg-opacity))!important}.bg-body{--bs-bg-opacity:1;background-color:rgba(var(--bs-body-bg-rgb),var(--bs-bg-opacity))!important}.bg-transparent{--bs-bg-opacity:1;background-color:transparent!important}.bg-body-secondary{--bs-bg-opacity:1;background-color:rgba(var(--bs-secondary-bg-rgb),var(--bs-bg-opacity))!important}.bg-body-tertiary{--bs-bg-opacity:1;background-color:rgba(var(--bs-tertiary-bg-rgb),var(--bs-bg-opacity))!important}.bg-opacity-10{--bs-bg-opacity:0.1}.bg-opacity-25{--bs-bg-opacity:0.25}.bg-opacity-50{--bs-bg-opacity:0.5}.bg-opacity-75{--bs-bg-opacity:0.75}.bg-opacity-100{--bs-bg-opacity:1}.bg-primary-subtle{background-color:var(--bs-primary-bg-subtle)!important}.bg-secondary-subtle{background-color:var(--bs-secondary-bg-subtle)!important}.bg-success-subtle{background-color:var(--bs-success-bg-subtle)!important}.bg-info-subtle{background-color:var(--bs-info-bg-subtle)!important}.bg-warning-subtle{background-color:var(--bs-warning-bg-subtle)!important}.bg-danger-subtle{background-color:var(--bs-danger-bg-subtle)!important}.bg-light-subtle{background-color:var(--bs-light-bg-subtle)!important}.bg-dark-subtle{background-color:var(--bs-dark-bg-subtle)!important}.bg-gradient{background-image:var(--bs-gradient)!important}.user-select-all{-webkit-user-select:all!important;-moz-user-select:all!important;user-select:all!important}.user-select-auto{-webkit-user-select:auto!important;-moz-user-select:auto!important;user-select:auto!important}.user-select-none{-webkit-user-select:none!important;-moz-user-select:none!important;user-select:none!important}.pe-none{pointer-events:none!important}.pe-auto{pointer-events:auto!important}.rounded{border-radius:var(--bs-border-radius)!important}.rounded-0{border-radius:0!important}.rounded-1{border-radius:var(--bs-border-radius-sm)!important}.rounded-2{border-radius:var(--bs-border-radius)!important}.rounded-3{border-radius:var(--bs-border-radius-lg)!important}.rounded-4{border-radius:var(--bs-border-radius-xl)!important}.rounded-5{border-radius:var(--bs-border-radius-xxl)!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:var(--bs-border-radius-pill)!important}.rounded-top{border-top-left-radius:var(--bs-border-radius)!important;border-top-right-radius:var(--bs-border-radius)!important}.rounded-top-0{border-top-left-radius:0!important;border-top-right-radius:0!important}.rounded-top-1{border-top-left-radius:var(--bs-border-radius-sm)!important;border-top-right-radius:var(--bs-border-radius-sm)!important}.rounded-top-2{border-top-left-radius:var(--bs-border-radius)!important;border-top-right-radius:var(--bs-border-radius)!important}.rounded-top-3{border-top-left-radius:var(--bs-border-radius-lg)!important;border-top-right-radius:var(--bs-border-radius-lg)!important}.rounded-top-4{border-top-left-radius:var(--bs-border-radius-xl)!important;border-top-right-radius:var(--bs-border-radius-xl)!important}.rounded-top-5{border-top-left-radius:var(--bs-border-radius-xxl)!important;border-top-right-radius:var(--bs-border-radius-xxl)!important}.rounded-top-circle{border-top-left-radius:50%!important;border-top-right-radius:50%!important}.rounded-top-pill{border-top-left-radius:var(--bs-border-radius-pill)!important;border-top-right-radius:var(--bs-border-radius-pill)!important}.rounded-end{border-top-right-radius:var(--bs-border-radius)!important;border-bottom-right-radius:var(--bs-border-radius)!important}.rounded-end-0{border-top-right-radius:0!important;border-bottom-right-radius:0!important}.rounded-end-1{border-top-right-radius:var(--bs-border-radius-sm)!important;border-bottom-right-radius:var(--bs-border-radius-sm)!important}.rounded-end-2{border-top-right-radius:var(--bs-border-radius)!important;border-bottom-right-radius:var(--bs-border-radius)!important}.rounded-end-3{border-top-right-radius:var(--bs-border-radius-lg)!important;border-bottom-right-radius:var(--bs-border-radius-lg)!important}.rounded-end-4{border-top-right-radius:var(--bs-border-radius-xl)!important;border-bottom-right-radius:var(--bs-border-radius-xl)!important}.rounded-end-5{border-top-right-radius:var(--bs-border-radius-xxl)!important;border-bottom-right-radius:var(--bs-border-radius-xxl)!important}.rounded-end-circle{border-top-right-radius:50%!important;border-bottom-right-radius:50%!important}.rounded-end-pill{border-top-right-radius:var(--bs-border-radius-pill)!important;border-bottom-right-radius:var(--bs-border-radius-pill)!important}.rounded-bottom{border-bottom-right-radius:var(--bs-border-radius)!important;border-bottom-left-radius:var(--bs-border-radius)!important}.rounded-bottom-0{border-bottom-right-radius:0!important;border-bottom-left-radius:0!important}.rounded-bottom-1{border-bottom-right-radius:var(--bs-border-radius-sm)!important;border-bottom-left-radius:var(--bs-border-radius-sm)!important}.rounded-bottom-2{border-bottom-right-radius:var(--bs-border-radius)!important;border-bottom-left-radius:var(--bs-border-radius)!important}.rounded-bottom-3{border-bottom-right-radius:var(--bs-border-radius-lg)!important;border-bottom-left-radius:var(--bs-border-radius-lg)!important}.rounded-bottom-4{border-bottom-right-radius:var(--bs-border-radius-xl)!important;border-bottom-left-radius:var(--bs-border-radius-xl)!important}.rounded-bottom-5{border-bottom-right-radius:var(--bs-border-radius-xxl)!important;border-bottom-left-radius:var(--bs-border-radius-xxl)!important}.rounded-bottom-circle{border-bottom-right-radius:50%!important;border-bottom-left-radius:50%!important}.rounded-bottom-pill{border-bottom-right-radius:var(--bs-border-radius-pill)!important;border-bottom-left-radius:var(--bs-border-radius-pill)!important}.rounded-start{border-bottom-left-radius:var(--bs-border-radius)!important;border-top-left-radius:var(--bs-border-radius)!important}.rounded-start-0{border-bottom-left-radius:0!important;border-top-left-radius:0!important}.rounded-start-1{border-bottom-left-radius:var(--bs-border-radius-sm)!important;border-top-left-radius:var(--bs-border-radius-sm)!important}.rounded-start-2{border-bottom-left-radius:var(--bs-border-radius)!important;border-top-left-radius:var(--bs-border-radius)!important}.rounded-start-3{border-bottom-left-radius:var(--bs-border-radius-lg)!important;border-top-left-radius:var(--bs-border-radius-lg)!important}.rounded-start-4{border-bottom-left-radius:var(--bs-border-radius-xl)!important;border-top-left-radius:var(--bs-border-radius-xl)!important}.rounded-start-5{border-bottom-left-radius:var(--bs-border-radius-xxl)!important;border-top-left-radius:var(--bs-border-radius-xxl)!important}.rounded-start-circle{border-bottom-left-radius:50%!important;border-top-left-radius:50%!important}.rounded-start-pill{border-bottom-left-radius:var(--bs-border-radius-pill)!important;border-top-left-radius:var(--bs-border-radius-pill)!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}.z-n1{z-index:-1!important}.z-0{z-index:0!important}.z-1{z-index:1!important}.z-2{z-index:2!important}.z-3{z-index:3!important}@media (min-width:576px){.float-sm-start{float:left!important}.float-sm-end{float:right!important}.float-sm-none{float:none!important}.object-fit-sm-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-sm-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-sm-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-sm-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-sm-none{-o-object-fit:none!important;object-fit:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-grid{display:grid!important}.d-sm-inline-grid{display:inline-grid!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:flex!important}.d-sm-inline-flex{display:inline-flex!important}.d-sm-none{display:none!important}.flex-sm-fill{flex:1 1 auto!important}.flex-sm-row{flex-direction:row!important}.flex-sm-column{flex-direction:column!important}.flex-sm-row-reverse{flex-direction:row-reverse!important}.flex-sm-column-reverse{flex-direction:column-reverse!important}.flex-sm-grow-0{flex-grow:0!important}.flex-sm-grow-1{flex-grow:1!important}.flex-sm-shrink-0{flex-shrink:0!important}.flex-sm-shrink-1{flex-shrink:1!important}.flex-sm-wrap{flex-wrap:wrap!important}.flex-sm-nowrap{flex-wrap:nowrap!important}.flex-sm-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-sm-start{justify-content:flex-start!important}.justify-content-sm-end{justify-content:flex-end!important}.justify-content-sm-center{justify-content:center!important}.justify-content-sm-between{justify-content:space-between!important}.justify-content-sm-around{justify-content:space-around!important}.justify-content-sm-evenly{justify-content:space-evenly!important}.align-items-sm-start{align-items:flex-start!important}.align-items-sm-end{align-items:flex-end!important}.align-items-sm-center{align-items:center!important}.align-items-sm-baseline{align-items:baseline!important}.align-items-sm-stretch{align-items:stretch!important}.align-content-sm-start{align-content:flex-start!important}.align-content-sm-end{align-content:flex-end!important}.align-content-sm-center{align-content:center!important}.align-content-sm-between{align-content:space-between!important}.align-content-sm-around{align-content:space-around!important}.align-content-sm-stretch{align-content:stretch!important}.align-self-sm-auto{align-self:auto!important}.align-self-sm-start{align-self:flex-start!important}.align-self-sm-end{align-self:flex-end!important}.align-self-sm-center{align-self:center!important}.align-self-sm-baseline{align-self:baseline!important}.align-self-sm-stretch{align-self:stretch!important}.order-sm-first{order:-1!important}.order-sm-0{order:0!important}.order-sm-1{order:1!important}.order-sm-2{order:2!important}.order-sm-3{order:3!important}.order-sm-4{order:4!important}.order-sm-5{order:5!important}.order-sm-last{order:6!important}.m-sm-0{margin:0!important}.m-sm-1{margin:.25rem!important}.m-sm-2{margin:.5rem!important}.m-sm-3{margin:1rem!important}.m-sm-4{margin:1.5rem!important}.m-sm-5{margin:3rem!important}.m-sm-auto{margin:auto!important}.mx-sm-0{margin-right:0!important;margin-left:0!important}.mx-sm-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-sm-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-sm-3{margin-right:1rem!important;margin-left:1rem!important}.mx-sm-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-sm-5{margin-right:3rem!important;margin-left:3rem!important}.mx-sm-auto{margin-right:auto!important;margin-left:auto!important}.my-sm-0{margin-top:0!important;margin-bottom:0!important}.my-sm-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-sm-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-sm-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-sm-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-sm-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-sm-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-sm-0{margin-top:0!important}.mt-sm-1{margin-top:.25rem!important}.mt-sm-2{margin-top:.5rem!important}.mt-sm-3{margin-top:1rem!important}.mt-sm-4{margin-top:1.5rem!important}.mt-sm-5{margin-top:3rem!important}.mt-sm-auto{margin-top:auto!important}.me-sm-0{margin-right:0!important}.me-sm-1{margin-right:.25rem!important}.me-sm-2{margin-right:.5rem!important}.me-sm-3{margin-right:1rem!important}.me-sm-4{margin-right:1.5rem!important}.me-sm-5{margin-right:3rem!important}.me-sm-auto{margin-right:auto!important}.mb-sm-0{margin-bottom:0!important}.mb-sm-1{margin-bottom:.25rem!important}.mb-sm-2{margin-bottom:.5rem!important}.mb-sm-3{margin-bottom:1rem!important}.mb-sm-4{margin-bottom:1.5rem!important}.mb-sm-5{margin-bottom:3rem!important}.mb-sm-auto{margin-bottom:auto!important}.ms-sm-0{margin-left:0!important}.ms-sm-1{margin-left:.25rem!important}.ms-sm-2{margin-left:.5rem!important}.ms-sm-3{margin-left:1rem!important}.ms-sm-4{margin-left:1.5rem!important}.ms-sm-5{margin-left:3rem!important}.ms-sm-auto{margin-left:auto!important}.p-sm-0{padding:0!important}.p-sm-1{padding:.25rem!important}.p-sm-2{padding:.5rem!important}.p-sm-3{padding:1rem!important}.p-sm-4{padding:1.5rem!important}.p-sm-5{padding:3rem!important}.px-sm-0{padding-right:0!important;padding-left:0!important}.px-sm-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-sm-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-sm-3{padding-right:1rem!important;padding-left:1rem!important}.px-sm-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-sm-5{padding-right:3rem!important;padding-left:3rem!important}.py-sm-0{padding-top:0!important;padding-bottom:0!important}.py-sm-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-sm-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-sm-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-sm-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-sm-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-sm-0{padding-top:0!important}.pt-sm-1{padding-top:.25rem!important}.pt-sm-2{padding-top:.5rem!important}.pt-sm-3{padding-top:1rem!important}.pt-sm-4{padding-top:1.5rem!important}.pt-sm-5{padding-top:3rem!important}.pe-sm-0{padding-right:0!important}.pe-sm-1{padding-right:.25rem!important}.pe-sm-2{padding-right:.5rem!important}.pe-sm-3{padding-right:1rem!important}.pe-sm-4{padding-right:1.5rem!important}.pe-sm-5{padding-right:3rem!important}.pb-sm-0{padding-bottom:0!important}.pb-sm-1{padding-bottom:.25rem!important}.pb-sm-2{padding-bottom:.5rem!important}.pb-sm-3{padding-bottom:1rem!important}.pb-sm-4{padding-bottom:1.5rem!important}.pb-sm-5{padding-bottom:3rem!important}.ps-sm-0{padding-left:0!important}.ps-sm-1{padding-left:.25rem!important}.ps-sm-2{padding-left:.5rem!important}.ps-sm-3{padding-left:1rem!important}.ps-sm-4{padding-left:1.5rem!important}.ps-sm-5{padding-left:3rem!important}.gap-sm-0{gap:0!important}.gap-sm-1{gap:.25rem!important}.gap-sm-2{gap:.5rem!important}.gap-sm-3{gap:1rem!important}.gap-sm-4{gap:1.5rem!important}.gap-sm-5{gap:3rem!important}.row-gap-sm-0{row-gap:0!important}.row-gap-sm-1{row-gap:.25rem!important}.row-gap-sm-2{row-gap:.5rem!important}.row-gap-sm-3{row-gap:1rem!important}.row-gap-sm-4{row-gap:1.5rem!important}.row-gap-sm-5{row-gap:3rem!important}.column-gap-sm-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-sm-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-sm-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-sm-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-sm-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-sm-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-sm-start{text-align:left!important}.text-sm-end{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.float-md-start{float:left!important}.float-md-end{float:right!important}.float-md-none{float:none!important}.object-fit-md-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-md-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-md-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-md-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-md-none{-o-object-fit:none!important;object-fit:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-grid{display:grid!important}.d-md-inline-grid{display:inline-grid!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:flex!important}.d-md-inline-flex{display:inline-flex!important}.d-md-none{display:none!important}.flex-md-fill{flex:1 1 auto!important}.flex-md-row{flex-direction:row!important}.flex-md-column{flex-direction:column!important}.flex-md-row-reverse{flex-direction:row-reverse!important}.flex-md-column-reverse{flex-direction:column-reverse!important}.flex-md-grow-0{flex-grow:0!important}.flex-md-grow-1{flex-grow:1!important}.flex-md-shrink-0{flex-shrink:0!important}.flex-md-shrink-1{flex-shrink:1!important}.flex-md-wrap{flex-wrap:wrap!important}.flex-md-nowrap{flex-wrap:nowrap!important}.flex-md-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-md-start{justify-content:flex-start!important}.justify-content-md-end{justify-content:flex-end!important}.justify-content-md-center{justify-content:center!important}.justify-content-md-between{justify-content:space-between!important}.justify-content-md-around{justify-content:space-around!important}.justify-content-md-evenly{justify-content:space-evenly!important}.align-items-md-start{align-items:flex-start!important}.align-items-md-end{align-items:flex-end!important}.align-items-md-center{align-items:center!important}.align-items-md-baseline{align-items:baseline!important}.align-items-md-stretch{align-items:stretch!important}.align-content-md-start{align-content:flex-start!important}.align-content-md-end{align-content:flex-end!important}.align-content-md-center{align-content:center!important}.align-content-md-between{align-content:space-between!important}.align-content-md-around{align-content:space-around!important}.align-content-md-stretch{align-content:stretch!important}.align-self-md-auto{align-self:auto!important}.align-self-md-start{align-self:flex-start!important}.align-self-md-end{align-self:flex-end!important}.align-self-md-center{align-self:center!important}.align-self-md-baseline{align-self:baseline!important}.align-self-md-stretch{align-self:stretch!important}.order-md-first{order:-1!important}.order-md-0{order:0!important}.order-md-1{order:1!important}.order-md-2{order:2!important}.order-md-3{order:3!important}.order-md-4{order:4!important}.order-md-5{order:5!important}.order-md-last{order:6!important}.m-md-0{margin:0!important}.m-md-1{margin:.25rem!important}.m-md-2{margin:.5rem!important}.m-md-3{margin:1rem!important}.m-md-4{margin:1.5rem!important}.m-md-5{margin:3rem!important}.m-md-auto{margin:auto!important}.mx-md-0{margin-right:0!important;margin-left:0!important}.mx-md-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-md-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-md-3{margin-right:1rem!important;margin-left:1rem!important}.mx-md-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-md-5{margin-right:3rem!important;margin-left:3rem!important}.mx-md-auto{margin-right:auto!important;margin-left:auto!important}.my-md-0{margin-top:0!important;margin-bottom:0!important}.my-md-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-md-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-md-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-md-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-md-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-md-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-md-0{margin-top:0!important}.mt-md-1{margin-top:.25rem!important}.mt-md-2{margin-top:.5rem!important}.mt-md-3{margin-top:1rem!important}.mt-md-4{margin-top:1.5rem!important}.mt-md-5{margin-top:3rem!important}.mt-md-auto{margin-top:auto!important}.me-md-0{margin-right:0!important}.me-md-1{margin-right:.25rem!important}.me-md-2{margin-right:.5rem!important}.me-md-3{margin-right:1rem!important}.me-md-4{margin-right:1.5rem!important}.me-md-5{margin-right:3rem!important}.me-md-auto{margin-right:auto!important}.mb-md-0{margin-bottom:0!important}.mb-md-1{margin-bottom:.25rem!important}.mb-md-2{margin-bottom:.5rem!important}.mb-md-3{margin-bottom:1rem!important}.mb-md-4{margin-bottom:1.5rem!important}.mb-md-5{margin-bottom:3rem!important}.mb-md-auto{margin-bottom:auto!important}.ms-md-0{margin-left:0!important}.ms-md-1{margin-left:.25rem!important}.ms-md-2{margin-left:.5rem!important}.ms-md-3{margin-left:1rem!important}.ms-md-4{margin-left:1.5rem!important}.ms-md-5{margin-left:3rem!important}.ms-md-auto{margin-left:auto!important}.p-md-0{padding:0!important}.p-md-1{padding:.25rem!important}.p-md-2{padding:.5rem!important}.p-md-3{padding:1rem!important}.p-md-4{padding:1.5rem!important}.p-md-5{padding:3rem!important}.px-md-0{padding-right:0!important;padding-left:0!important}.px-md-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-md-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-md-3{padding-right:1rem!important;padding-left:1rem!important}.px-md-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-md-5{padding-right:3rem!important;padding-left:3rem!important}.py-md-0{padding-top:0!important;padding-bottom:0!important}.py-md-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-md-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-md-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-md-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-md-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-md-0{padding-top:0!important}.pt-md-1{padding-top:.25rem!important}.pt-md-2{padding-top:.5rem!important}.pt-md-3{padding-top:1rem!important}.pt-md-4{padding-top:1.5rem!important}.pt-md-5{padding-top:3rem!important}.pe-md-0{padding-right:0!important}.pe-md-1{padding-right:.25rem!important}.pe-md-2{padding-right:.5rem!important}.pe-md-3{padding-right:1rem!important}.pe-md-4{padding-right:1.5rem!important}.pe-md-5{padding-right:3rem!important}.pb-md-0{padding-bottom:0!important}.pb-md-1{padding-bottom:.25rem!important}.pb-md-2{padding-bottom:.5rem!important}.pb-md-3{padding-bottom:1rem!important}.pb-md-4{padding-bottom:1.5rem!important}.pb-md-5{padding-bottom:3rem!important}.ps-md-0{padding-left:0!important}.ps-md-1{padding-left:.25rem!important}.ps-md-2{padding-left:.5rem!important}.ps-md-3{padding-left:1rem!important}.ps-md-4{padding-left:1.5rem!important}.ps-md-5{padding-left:3rem!important}.gap-md-0{gap:0!important}.gap-md-1{gap:.25rem!important}.gap-md-2{gap:.5rem!important}.gap-md-3{gap:1rem!important}.gap-md-4{gap:1.5rem!important}.gap-md-5{gap:3rem!important}.row-gap-md-0{row-gap:0!important}.row-gap-md-1{row-gap:.25rem!important}.row-gap-md-2{row-gap:.5rem!important}.row-gap-md-3{row-gap:1rem!important}.row-gap-md-4{row-gap:1.5rem!important}.row-gap-md-5{row-gap:3rem!important}.column-gap-md-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-md-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-md-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-md-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-md-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-md-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-md-start{text-align:left!important}.text-md-end{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.float-lg-start{float:left!important}.float-lg-end{float:right!important}.float-lg-none{float:none!important}.object-fit-lg-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-lg-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-lg-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-lg-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-lg-none{-o-object-fit:none!important;object-fit:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-grid{display:grid!important}.d-lg-inline-grid{display:inline-grid!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:flex!important}.d-lg-inline-flex{display:inline-flex!important}.d-lg-none{display:none!important}.flex-lg-fill{flex:1 1 auto!important}.flex-lg-row{flex-direction:row!important}.flex-lg-column{flex-direction:column!important}.flex-lg-row-reverse{flex-direction:row-reverse!important}.flex-lg-column-reverse{flex-direction:column-reverse!important}.flex-lg-grow-0{flex-grow:0!important}.flex-lg-grow-1{flex-grow:1!important}.flex-lg-shrink-0{flex-shrink:0!important}.flex-lg-shrink-1{flex-shrink:1!important}.flex-lg-wrap{flex-wrap:wrap!important}.flex-lg-nowrap{flex-wrap:nowrap!important}.flex-lg-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-lg-start{justify-content:flex-start!important}.justify-content-lg-end{justify-content:flex-end!important}.justify-content-lg-center{justify-content:center!important}.justify-content-lg-between{justify-content:space-between!important}.justify-content-lg-around{justify-content:space-around!important}.justify-content-lg-evenly{justify-content:space-evenly!important}.align-items-lg-start{align-items:flex-start!important}.align-items-lg-end{align-items:flex-end!important}.align-items-lg-center{align-items:center!important}.align-items-lg-baseline{align-items:baseline!important}.align-items-lg-stretch{align-items:stretch!important}.align-content-lg-start{align-content:flex-start!important}.align-content-lg-end{align-content:flex-end!important}.align-content-lg-center{align-content:center!important}.align-content-lg-between{align-content:space-between!important}.align-content-lg-around{align-content:space-around!important}.align-content-lg-stretch{align-content:stretch!important}.align-self-lg-auto{align-self:auto!important}.align-self-lg-start{align-self:flex-start!important}.align-self-lg-end{align-self:flex-end!important}.align-self-lg-center{align-self:center!important}.align-self-lg-baseline{align-self:baseline!important}.align-self-lg-stretch{align-self:stretch!important}.order-lg-first{order:-1!important}.order-lg-0{order:0!important}.order-lg-1{order:1!important}.order-lg-2{order:2!important}.order-lg-3{order:3!important}.order-lg-4{order:4!important}.order-lg-5{order:5!important}.order-lg-last{order:6!important}.m-lg-0{margin:0!important}.m-lg-1{margin:.25rem!important}.m-lg-2{margin:.5rem!important}.m-lg-3{margin:1rem!important}.m-lg-4{margin:1.5rem!important}.m-lg-5{margin:3rem!important}.m-lg-auto{margin:auto!important}.mx-lg-0{margin-right:0!important;margin-left:0!important}.mx-lg-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-lg-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-lg-3{margin-right:1rem!important;margin-left:1rem!important}.mx-lg-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-lg-5{margin-right:3rem!important;margin-left:3rem!important}.mx-lg-auto{margin-right:auto!important;margin-left:auto!important}.my-lg-0{margin-top:0!important;margin-bottom:0!important}.my-lg-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-lg-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-lg-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-lg-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-lg-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-lg-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-lg-0{margin-top:0!important}.mt-lg-1{margin-top:.25rem!important}.mt-lg-2{margin-top:.5rem!important}.mt-lg-3{margin-top:1rem!important}.mt-lg-4{margin-top:1.5rem!important}.mt-lg-5{margin-top:3rem!important}.mt-lg-auto{margin-top:auto!important}.me-lg-0{margin-right:0!important}.me-lg-1{margin-right:.25rem!important}.me-lg-2{margin-right:.5rem!important}.me-lg-3{margin-right:1rem!important}.me-lg-4{margin-right:1.5rem!important}.me-lg-5{margin-right:3rem!important}.me-lg-auto{margin-right:auto!important}.mb-lg-0{margin-bottom:0!important}.mb-lg-1{margin-bottom:.25rem!important}.mb-lg-2{margin-bottom:.5rem!important}.mb-lg-3{margin-bottom:1rem!important}.mb-lg-4{margin-bottom:1.5rem!important}.mb-lg-5{margin-bottom:3rem!important}.mb-lg-auto{margin-bottom:auto!important}.ms-lg-0{margin-left:0!important}.ms-lg-1{margin-left:.25rem!important}.ms-lg-2{margin-left:.5rem!important}.ms-lg-3{margin-left:1rem!important}.ms-lg-4{margin-left:1.5rem!important}.ms-lg-5{margin-left:3rem!important}.ms-lg-auto{margin-left:auto!important}.p-lg-0{padding:0!important}.p-lg-1{padding:.25rem!important}.p-lg-2{padding:.5rem!important}.p-lg-3{padding:1rem!important}.p-lg-4{padding:1.5rem!important}.p-lg-5{padding:3rem!important}.px-lg-0{padding-right:0!important;padding-left:0!important}.px-lg-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-lg-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-lg-3{padding-right:1rem!important;padding-left:1rem!important}.px-lg-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-lg-5{padding-right:3rem!important;padding-left:3rem!important}.py-lg-0{padding-top:0!important;padding-bottom:0!important}.py-lg-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-lg-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-lg-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-lg-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-lg-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-lg-0{padding-top:0!important}.pt-lg-1{padding-top:.25rem!important}.pt-lg-2{padding-top:.5rem!important}.pt-lg-3{padding-top:1rem!important}.pt-lg-4{padding-top:1.5rem!important}.pt-lg-5{padding-top:3rem!important}.pe-lg-0{padding-right:0!important}.pe-lg-1{padding-right:.25rem!important}.pe-lg-2{padding-right:.5rem!important}.pe-lg-3{padding-right:1rem!important}.pe-lg-4{padding-right:1.5rem!important}.pe-lg-5{padding-right:3rem!important}.pb-lg-0{padding-bottom:0!important}.pb-lg-1{padding-bottom:.25rem!important}.pb-lg-2{padding-bottom:.5rem!important}.pb-lg-3{padding-bottom:1rem!important}.pb-lg-4{padding-bottom:1.5rem!important}.pb-lg-5{padding-bottom:3rem!important}.ps-lg-0{padding-left:0!important}.ps-lg-1{padding-left:.25rem!important}.ps-lg-2{padding-left:.5rem!important}.ps-lg-3{padding-left:1rem!important}.ps-lg-4{padding-left:1.5rem!important}.ps-lg-5{padding-left:3rem!important}.gap-lg-0{gap:0!important}.gap-lg-1{gap:.25rem!important}.gap-lg-2{gap:.5rem!important}.gap-lg-3{gap:1rem!important}.gap-lg-4{gap:1.5rem!important}.gap-lg-5{gap:3rem!important}.row-gap-lg-0{row-gap:0!important}.row-gap-lg-1{row-gap:.25rem!important}.row-gap-lg-2{row-gap:.5rem!important}.row-gap-lg-3{row-gap:1rem!important}.row-gap-lg-4{row-gap:1.5rem!important}.row-gap-lg-5{row-gap:3rem!important}.column-gap-lg-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-lg-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-lg-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-lg-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-lg-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-lg-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-lg-start{text-align:left!important}.text-lg-end{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.float-xl-start{float:left!important}.float-xl-end{float:right!important}.float-xl-none{float:none!important}.object-fit-xl-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-xl-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-xl-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-xl-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-xl-none{-o-object-fit:none!important;object-fit:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-grid{display:grid!important}.d-xl-inline-grid{display:inline-grid!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:flex!important}.d-xl-inline-flex{display:inline-flex!important}.d-xl-none{display:none!important}.flex-xl-fill{flex:1 1 auto!important}.flex-xl-row{flex-direction:row!important}.flex-xl-column{flex-direction:column!important}.flex-xl-row-reverse{flex-direction:row-reverse!important}.flex-xl-column-reverse{flex-direction:column-reverse!important}.flex-xl-grow-0{flex-grow:0!important}.flex-xl-grow-1{flex-grow:1!important}.flex-xl-shrink-0{flex-shrink:0!important}.flex-xl-shrink-1{flex-shrink:1!important}.flex-xl-wrap{flex-wrap:wrap!important}.flex-xl-nowrap{flex-wrap:nowrap!important}.flex-xl-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-xl-start{justify-content:flex-start!important}.justify-content-xl-end{justify-content:flex-end!important}.justify-content-xl-center{justify-content:center!important}.justify-content-xl-between{justify-content:space-between!important}.justify-content-xl-around{justify-content:space-around!important}.justify-content-xl-evenly{justify-content:space-evenly!important}.align-items-xl-start{align-items:flex-start!important}.align-items-xl-end{align-items:flex-end!important}.align-items-xl-center{align-items:center!important}.align-items-xl-baseline{align-items:baseline!important}.align-items-xl-stretch{align-items:stretch!important}.align-content-xl-start{align-content:flex-start!important}.align-content-xl-end{align-content:flex-end!important}.align-content-xl-center{align-content:center!important}.align-content-xl-between{align-content:space-between!important}.align-content-xl-around{align-content:space-around!important}.align-content-xl-stretch{align-content:stretch!important}.align-self-xl-auto{align-self:auto!important}.align-self-xl-start{align-self:flex-start!important}.align-self-xl-end{align-self:flex-end!important}.align-self-xl-center{align-self:center!important}.align-self-xl-baseline{align-self:baseline!important}.align-self-xl-stretch{align-self:stretch!important}.order-xl-first{order:-1!important}.order-xl-0{order:0!important}.order-xl-1{order:1!important}.order-xl-2{order:2!important}.order-xl-3{order:3!important}.order-xl-4{order:4!important}.order-xl-5{order:5!important}.order-xl-last{order:6!important}.m-xl-0{margin:0!important}.m-xl-1{margin:.25rem!important}.m-xl-2{margin:.5rem!important}.m-xl-3{margin:1rem!important}.m-xl-4{margin:1.5rem!important}.m-xl-5{margin:3rem!important}.m-xl-auto{margin:auto!important}.mx-xl-0{margin-right:0!important;margin-left:0!important}.mx-xl-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-xl-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-xl-3{margin-right:1rem!important;margin-left:1rem!important}.mx-xl-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-xl-5{margin-right:3rem!important;margin-left:3rem!important}.mx-xl-auto{margin-right:auto!important;margin-left:auto!important}.my-xl-0{margin-top:0!important;margin-bottom:0!important}.my-xl-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-xl-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-xl-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-xl-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-xl-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-xl-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-xl-0{margin-top:0!important}.mt-xl-1{margin-top:.25rem!important}.mt-xl-2{margin-top:.5rem!important}.mt-xl-3{margin-top:1rem!important}.mt-xl-4{margin-top:1.5rem!important}.mt-xl-5{margin-top:3rem!important}.mt-xl-auto{margin-top:auto!important}.me-xl-0{margin-right:0!important}.me-xl-1{margin-right:.25rem!important}.me-xl-2{margin-right:.5rem!important}.me-xl-3{margin-right:1rem!important}.me-xl-4{margin-right:1.5rem!important}.me-xl-5{margin-right:3rem!important}.me-xl-auto{margin-right:auto!important}.mb-xl-0{margin-bottom:0!important}.mb-xl-1{margin-bottom:.25rem!important}.mb-xl-2{margin-bottom:.5rem!important}.mb-xl-3{margin-bottom:1rem!important}.mb-xl-4{margin-bottom:1.5rem!important}.mb-xl-5{margin-bottom:3rem!important}.mb-xl-auto{margin-bottom:auto!important}.ms-xl-0{margin-left:0!important}.ms-xl-1{margin-left:.25rem!important}.ms-xl-2{margin-left:.5rem!important}.ms-xl-3{margin-left:1rem!important}.ms-xl-4{margin-left:1.5rem!important}.ms-xl-5{margin-left:3rem!important}.ms-xl-auto{margin-left:auto!important}.p-xl-0{padding:0!important}.p-xl-1{padding:.25rem!important}.p-xl-2{padding:.5rem!important}.p-xl-3{padding:1rem!important}.p-xl-4{padding:1.5rem!important}.p-xl-5{padding:3rem!important}.px-xl-0{padding-right:0!important;padding-left:0!important}.px-xl-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-xl-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-xl-3{padding-right:1rem!important;padding-left:1rem!important}.px-xl-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-xl-5{padding-right:3rem!important;padding-left:3rem!important}.py-xl-0{padding-top:0!important;padding-bottom:0!important}.py-xl-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-xl-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-xl-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-xl-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-xl-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-xl-0{padding-top:0!important}.pt-xl-1{padding-top:.25rem!important}.pt-xl-2{padding-top:.5rem!important}.pt-xl-3{padding-top:1rem!important}.pt-xl-4{padding-top:1.5rem!important}.pt-xl-5{padding-top:3rem!important}.pe-xl-0{padding-right:0!important}.pe-xl-1{padding-right:.25rem!important}.pe-xl-2{padding-right:.5rem!important}.pe-xl-3{padding-right:1rem!important}.pe-xl-4{padding-right:1.5rem!important}.pe-xl-5{padding-right:3rem!important}.pb-xl-0{padding-bottom:0!important}.pb-xl-1{padding-bottom:.25rem!important}.pb-xl-2{padding-bottom:.5rem!important}.pb-xl-3{padding-bottom:1rem!important}.pb-xl-4{padding-bottom:1.5rem!important}.pb-xl-5{padding-bottom:3rem!important}.ps-xl-0{padding-left:0!important}.ps-xl-1{padding-left:.25rem!important}.ps-xl-2{padding-left:.5rem!important}.ps-xl-3{padding-left:1rem!important}.ps-xl-4{padding-left:1.5rem!important}.ps-xl-5{padding-left:3rem!important}.gap-xl-0{gap:0!important}.gap-xl-1{gap:.25rem!important}.gap-xl-2{gap:.5rem!important}.gap-xl-3{gap:1rem!important}.gap-xl-4{gap:1.5rem!important}.gap-xl-5{gap:3rem!important}.row-gap-xl-0{row-gap:0!important}.row-gap-xl-1{row-gap:.25rem!important}.row-gap-xl-2{row-gap:.5rem!important}.row-gap-xl-3{row-gap:1rem!important}.row-gap-xl-4{row-gap:1.5rem!important}.row-gap-xl-5{row-gap:3rem!important}.column-gap-xl-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-xl-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-xl-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-xl-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-xl-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-xl-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-xl-start{text-align:left!important}.text-xl-end{text-align:right!important}.text-xl-center{text-align:center!important}}@media (min-width:1400px){.float-xxl-start{float:left!important}.float-xxl-end{float:right!important}.float-xxl-none{float:none!important}.object-fit-xxl-contain{-o-object-fit:contain!important;object-fit:contain!important}.object-fit-xxl-cover{-o-object-fit:cover!important;object-fit:cover!important}.object-fit-xxl-fill{-o-object-fit:fill!important;object-fit:fill!important}.object-fit-xxl-scale{-o-object-fit:scale-down!important;object-fit:scale-down!important}.object-fit-xxl-none{-o-object-fit:none!important;object-fit:none!important}.d-xxl-inline{display:inline!important}.d-xxl-inline-block{display:inline-block!important}.d-xxl-block{display:block!important}.d-xxl-grid{display:grid!important}.d-xxl-inline-grid{display:inline-grid!important}.d-xxl-table{display:table!important}.d-xxl-table-row{display:table-row!important}.d-xxl-table-cell{display:table-cell!important}.d-xxl-flex{display:flex!important}.d-xxl-inline-flex{display:inline-flex!important}.d-xxl-none{display:none!important}.flex-xxl-fill{flex:1 1 auto!important}.flex-xxl-row{flex-direction:row!important}.flex-xxl-column{flex-direction:column!important}.flex-xxl-row-reverse{flex-direction:row-reverse!important}.flex-xxl-column-reverse{flex-direction:column-reverse!important}.flex-xxl-grow-0{flex-grow:0!important}.flex-xxl-grow-1{flex-grow:1!important}.flex-xxl-shrink-0{flex-shrink:0!important}.flex-xxl-shrink-1{flex-shrink:1!important}.flex-xxl-wrap{flex-wrap:wrap!important}.flex-xxl-nowrap{flex-wrap:nowrap!important}.flex-xxl-wrap-reverse{flex-wrap:wrap-reverse!important}.justify-content-xxl-start{justify-content:flex-start!important}.justify-content-xxl-end{justify-content:flex-end!important}.justify-content-xxl-center{justify-content:center!important}.justify-content-xxl-between{justify-content:space-between!important}.justify-content-xxl-around{justify-content:space-around!important}.justify-content-xxl-evenly{justify-content:space-evenly!important}.align-items-xxl-start{align-items:flex-start!important}.align-items-xxl-end{align-items:flex-end!important}.align-items-xxl-center{align-items:center!important}.align-items-xxl-baseline{align-items:baseline!important}.align-items-xxl-stretch{align-items:stretch!important}.align-content-xxl-start{align-content:flex-start!important}.align-content-xxl-end{align-content:flex-end!important}.align-content-xxl-center{align-content:center!important}.align-content-xxl-between{align-content:space-between!important}.align-content-xxl-around{align-content:space-around!important}.align-content-xxl-stretch{align-content:stretch!important}.align-self-xxl-auto{align-self:auto!important}.align-self-xxl-start{align-self:flex-start!important}.align-self-xxl-end{align-self:flex-end!important}.align-self-xxl-center{align-self:center!important}.align-self-xxl-baseline{align-self:baseline!important}.align-self-xxl-stretch{align-self:stretch!important}.order-xxl-first{order:-1!important}.order-xxl-0{order:0!important}.order-xxl-1{order:1!important}.order-xxl-2{order:2!important}.order-xxl-3{order:3!important}.order-xxl-4{order:4!important}.order-xxl-5{order:5!important}.order-xxl-last{order:6!important}.m-xxl-0{margin:0!important}.m-xxl-1{margin:.25rem!important}.m-xxl-2{margin:.5rem!important}.m-xxl-3{margin:1rem!important}.m-xxl-4{margin:1.5rem!important}.m-xxl-5{margin:3rem!important}.m-xxl-auto{margin:auto!important}.mx-xxl-0{margin-right:0!important;margin-left:0!important}.mx-xxl-1{margin-right:.25rem!important;margin-left:.25rem!important}.mx-xxl-2{margin-right:.5rem!important;margin-left:.5rem!important}.mx-xxl-3{margin-right:1rem!important;margin-left:1rem!important}.mx-xxl-4{margin-right:1.5rem!important;margin-left:1.5rem!important}.mx-xxl-5{margin-right:3rem!important;margin-left:3rem!important}.mx-xxl-auto{margin-right:auto!important;margin-left:auto!important}.my-xxl-0{margin-top:0!important;margin-bottom:0!important}.my-xxl-1{margin-top:.25rem!important;margin-bottom:.25rem!important}.my-xxl-2{margin-top:.5rem!important;margin-bottom:.5rem!important}.my-xxl-3{margin-top:1rem!important;margin-bottom:1rem!important}.my-xxl-4{margin-top:1.5rem!important;margin-bottom:1.5rem!important}.my-xxl-5{margin-top:3rem!important;margin-bottom:3rem!important}.my-xxl-auto{margin-top:auto!important;margin-bottom:auto!important}.mt-xxl-0{margin-top:0!important}.mt-xxl-1{margin-top:.25rem!important}.mt-xxl-2{margin-top:.5rem!important}.mt-xxl-3{margin-top:1rem!important}.mt-xxl-4{margin-top:1.5rem!important}.mt-xxl-5{margin-top:3rem!important}.mt-xxl-auto{margin-top:auto!important}.me-xxl-0{margin-right:0!important}.me-xxl-1{margin-right:.25rem!important}.me-xxl-2{margin-right:.5rem!important}.me-xxl-3{margin-right:1rem!important}.me-xxl-4{margin-right:1.5rem!important}.me-xxl-5{margin-right:3rem!important}.me-xxl-auto{margin-right:auto!important}.mb-xxl-0{margin-bottom:0!important}.mb-xxl-1{margin-bottom:.25rem!important}.mb-xxl-2{margin-bottom:.5rem!important}.mb-xxl-3{margin-bottom:1rem!important}.mb-xxl-4{margin-bottom:1.5rem!important}.mb-xxl-5{margin-bottom:3rem!important}.mb-xxl-auto{margin-bottom:auto!important}.ms-xxl-0{margin-left:0!important}.ms-xxl-1{margin-left:.25rem!important}.ms-xxl-2{margin-left:.5rem!important}.ms-xxl-3{margin-left:1rem!important}.ms-xxl-4{margin-left:1.5rem!important}.ms-xxl-5{margin-left:3rem!important}.ms-xxl-auto{margin-left:auto!important}.p-xxl-0{padding:0!important}.p-xxl-1{padding:.25rem!important}.p-xxl-2{padding:.5rem!important}.p-xxl-3{padding:1rem!important}.p-xxl-4{padding:1.5rem!important}.p-xxl-5{padding:3rem!important}.px-xxl-0{padding-right:0!important;padding-left:0!important}.px-xxl-1{padding-right:.25rem!important;padding-left:.25rem!important}.px-xxl-2{padding-right:.5rem!important;padding-left:.5rem!important}.px-xxl-3{padding-right:1rem!important;padding-left:1rem!important}.px-xxl-4{padding-right:1.5rem!important;padding-left:1.5rem!important}.px-xxl-5{padding-right:3rem!important;padding-left:3rem!important}.py-xxl-0{padding-top:0!important;padding-bottom:0!important}.py-xxl-1{padding-top:.25rem!important;padding-bottom:.25rem!important}.py-xxl-2{padding-top:.5rem!important;padding-bottom:.5rem!important}.py-xxl-3{padding-top:1rem!important;padding-bottom:1rem!important}.py-xxl-4{padding-top:1.5rem!important;padding-bottom:1.5rem!important}.py-xxl-5{padding-top:3rem!important;padding-bottom:3rem!important}.pt-xxl-0{padding-top:0!important}.pt-xxl-1{padding-top:.25rem!important}.pt-xxl-2{padding-top:.5rem!important}.pt-xxl-3{padding-top:1rem!important}.pt-xxl-4{padding-top:1.5rem!important}.pt-xxl-5{padding-top:3rem!important}.pe-xxl-0{padding-right:0!important}.pe-xxl-1{padding-right:.25rem!important}.pe-xxl-2{padding-right:.5rem!important}.pe-xxl-3{padding-right:1rem!important}.pe-xxl-4{padding-right:1.5rem!important}.pe-xxl-5{padding-right:3rem!important}.pb-xxl-0{padding-bottom:0!important}.pb-xxl-1{padding-bottom:.25rem!important}.pb-xxl-2{padding-bottom:.5rem!important}.pb-xxl-3{padding-bottom:1rem!important}.pb-xxl-4{padding-bottom:1.5rem!important}.pb-xxl-5{padding-bottom:3rem!important}.ps-xxl-0{padding-left:0!important}.ps-xxl-1{padding-left:.25rem!important}.ps-xxl-2{padding-left:.5rem!important}.ps-xxl-3{padding-left:1rem!important}.ps-xxl-4{padding-left:1.5rem!important}.ps-xxl-5{padding-left:3rem!important}.gap-xxl-0{gap:0!important}.gap-xxl-1{gap:.25rem!important}.gap-xxl-2{gap:.5rem!important}.gap-xxl-3{gap:1rem!important}.gap-xxl-4{gap:1.5rem!important}.gap-xxl-5{gap:3rem!important}.row-gap-xxl-0{row-gap:0!important}.row-gap-xxl-1{row-gap:.25rem!important}.row-gap-xxl-2{row-gap:.5rem!important}.row-gap-xxl-3{row-gap:1rem!important}.row-gap-xxl-4{row-gap:1.5rem!important}.row-gap-xxl-5{row-gap:3rem!important}.column-gap-xxl-0{-moz-column-gap:0!important;column-gap:0!important}.column-gap-xxl-1{-moz-column-gap:0.25rem!important;column-gap:.25rem!important}.column-gap-xxl-2{-moz-column-gap:0.5rem!important;column-gap:.5rem!important}.column-gap-xxl-3{-moz-column-gap:1rem!important;column-gap:1rem!important}.column-gap-xxl-4{-moz-column-gap:1.5rem!important;column-gap:1.5rem!important}.column-gap-xxl-5{-moz-column-gap:3rem!important;column-gap:3rem!important}.text-xxl-start{text-align:left!important}.text-xxl-end{text-align:right!important}.text-xxl-center{text-align:center!important}}@media (min-width:1200px){.fs-1{font-size:2.5rem!important}.fs-2{font-size:2rem!important}.fs-3{font-size:1.75rem!important}.fs-4{font-size:1.5rem!important}}@media print{.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-grid{display:grid!important}.d-print-inline-grid{display:inline-grid!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:flex!important}.d-print-inline-flex{display:inline-flex!important}.d-print-none{display:none!important}} +/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file diff --git a/app/static/css/fonts/bootstrap-icons.woff b/app/static/css/fonts/bootstrap-icons.woff new file mode 100644 index 0000000000000000000000000000000000000000..6e72a590a26bc71c7011e3c50da2ca766a853344 Binary files /dev/null and b/app/static/css/fonts/bootstrap-icons.woff differ diff --git a/app/static/css/fonts/bootstrap-icons.woff2 b/app/static/css/fonts/bootstrap-icons.woff2 new file mode 100644 index 0000000000000000000000000000000000000000..3b957d5a764cc096fcee42de5ab6a48dfee9af4a Binary files /dev/null and b/app/static/css/fonts/bootstrap-icons.woff2 differ diff --git a/app/static/img/tech-audio-logo.png b/app/static/img/tech-audio-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..1c70bf1c0f1bb2b5166109e2be63ef00910aa8ea Binary files /dev/null and b/app/static/img/tech-audio-logo.png differ diff --git a/app/static/js/bootstrap.bundle.min.js b/app/static/js/bootstrap.bundle.min.js new file mode 100644 index 0000000000000000000000000000000000000000..e8f21f703f7bb4e9ab84daca93c9ee1d5358a316 --- /dev/null +++ b/app/static/js/bootstrap.bundle.min.js @@ -0,0 +1,7 @@ +/*! + * Bootstrap v5.3.1 (https://getbootstrap.com/) + * Copyright 2011-2023 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors) + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */ +!function(t,e){"object"==typeof exports&&"undefined"!=typeof module?module.exports=e():"function"==typeof define&&define.amd?define(e):(t="undefined"!=typeof globalThis?globalThis:t||self).bootstrap=e()}(this,(function(){"use strict";const t=new Map,e={set(e,i,n){t.has(e)||t.set(e,new Map);const s=t.get(e);s.has(i)||0===s.size?s.set(i,n):console.error(`Bootstrap doesn't allow more than one instance per element. Bound instance: ${Array.from(s.keys())[0]}.`)},get:(e,i)=>t.has(e)&&t.get(e).get(i)||null,remove(e,i){if(!t.has(e))return;const n=t.get(e);n.delete(i),0===n.size&&t.delete(e)}},i="transitionend",n=t=>(t&&window.CSS&&window.CSS.escape&&(t=t.replace(/#([^\s"#']+)/g,((t,e)=>`#${CSS.escape(e)}`))),t),s=t=>{t.dispatchEvent(new Event(i))},o=t=>!(!t||"object"!=typeof t)&&(void 0!==t.jquery&&(t=t[0]),void 0!==t.nodeType),r=t=>o(t)?t.jquery?t[0]:t:"string"==typeof t&&t.length>0?document.querySelector(n(t)):null,a=t=>{if(!o(t)||0===t.getClientRects().length)return!1;const e="visible"===getComputedStyle(t).getPropertyValue("visibility"),i=t.closest("details:not([open])");if(!i)return e;if(i!==t){const e=t.closest("summary");if(e&&e.parentNode!==i)return!1;if(null===e)return!1}return e},l=t=>!t||t.nodeType!==Node.ELEMENT_NODE||!!t.classList.contains("disabled")||(void 0!==t.disabled?t.disabled:t.hasAttribute("disabled")&&"false"!==t.getAttribute("disabled")),c=t=>{if(!document.documentElement.attachShadow)return null;if("function"==typeof t.getRootNode){const e=t.getRootNode();return e instanceof ShadowRoot?e:null}return t instanceof ShadowRoot?t:t.parentNode?c(t.parentNode):null},h=()=>{},d=t=>{t.offsetHeight},u=()=>window.jQuery&&!document.body.hasAttribute("data-bs-no-jquery")?window.jQuery:null,f=[],p=()=>"rtl"===document.documentElement.dir,m=t=>{var e;e=()=>{const e=u();if(e){const i=t.NAME,n=e.fn[i];e.fn[i]=t.jQueryInterface,e.fn[i].Constructor=t,e.fn[i].noConflict=()=>(e.fn[i]=n,t.jQueryInterface)}},"loading"===document.readyState?(f.length||document.addEventListener("DOMContentLoaded",(()=>{for(const t of f)t()})),f.push(e)):e()},g=(t,e=[],i=t)=>"function"==typeof t?t(...e):i,_=(t,e,n=!0)=>{if(!n)return void g(t);const o=(t=>{if(!t)return 0;let{transitionDuration:e,transitionDelay:i}=window.getComputedStyle(t);const n=Number.parseFloat(e),s=Number.parseFloat(i);return n||s?(e=e.split(",")[0],i=i.split(",")[0],1e3*(Number.parseFloat(e)+Number.parseFloat(i))):0})(e)+5;let r=!1;const a=({target:n})=>{n===e&&(r=!0,e.removeEventListener(i,a),g(t))};e.addEventListener(i,a),setTimeout((()=>{r||s(e)}),o)},b=(t,e,i,n)=>{const s=t.length;let o=t.indexOf(e);return-1===o?!i&&n?t[s-1]:t[0]:(o+=i?1:-1,n&&(o=(o+s)%s),t[Math.max(0,Math.min(o,s-1))])},v=/[^.]*(?=\..*)\.|.*/,y=/\..*/,w=/::\d+$/,A={};let E=1;const T={mouseenter:"mouseover",mouseleave:"mouseout"},C=new Set(["click","dblclick","mouseup","mousedown","contextmenu","mousewheel","DOMMouseScroll","mouseover","mouseout","mousemove","selectstart","selectend","keydown","keypress","keyup","orientationchange","touchstart","touchmove","touchend","touchcancel","pointerdown","pointermove","pointerup","pointerleave","pointercancel","gesturestart","gesturechange","gestureend","focus","blur","change","reset","select","submit","focusin","focusout","load","unload","beforeunload","resize","move","DOMContentLoaded","readystatechange","error","abort","scroll"]);function O(t,e){return e&&`${e}::${E++}`||t.uidEvent||E++}function x(t){const e=O(t);return t.uidEvent=e,A[e]=A[e]||{},A[e]}function k(t,e,i=null){return Object.values(t).find((t=>t.callable===e&&t.delegationSelector===i))}function L(t,e,i){const n="string"==typeof e,s=n?i:e||i;let o=I(t);return C.has(o)||(o=t),[n,s,o]}function S(t,e,i,n,s){if("string"!=typeof e||!t)return;let[o,r,a]=L(e,i,n);if(e in T){const t=t=>function(e){if(!e.relatedTarget||e.relatedTarget!==e.delegateTarget&&!e.delegateTarget.contains(e.relatedTarget))return t.call(this,e)};r=t(r)}const l=x(t),c=l[a]||(l[a]={}),h=k(c,r,o?i:null);if(h)return void(h.oneOff=h.oneOff&&s);const d=O(r,e.replace(v,"")),u=o?function(t,e,i){return function n(s){const o=t.querySelectorAll(e);for(let{target:r}=s;r&&r!==this;r=r.parentNode)for(const a of o)if(a===r)return P(s,{delegateTarget:r}),n.oneOff&&N.off(t,s.type,e,i),i.apply(r,[s])}}(t,i,r):function(t,e){return function i(n){return P(n,{delegateTarget:t}),i.oneOff&&N.off(t,n.type,e),e.apply(t,[n])}}(t,r);u.delegationSelector=o?i:null,u.callable=r,u.oneOff=s,u.uidEvent=d,c[d]=u,t.addEventListener(a,u,o)}function D(t,e,i,n,s){const o=k(e[i],n,s);o&&(t.removeEventListener(i,o,Boolean(s)),delete e[i][o.uidEvent])}function $(t,e,i,n){const s=e[i]||{};for(const[o,r]of Object.entries(s))o.includes(n)&&D(t,e,i,r.callable,r.delegationSelector)}function I(t){return t=t.replace(y,""),T[t]||t}const N={on(t,e,i,n){S(t,e,i,n,!1)},one(t,e,i,n){S(t,e,i,n,!0)},off(t,e,i,n){if("string"!=typeof e||!t)return;const[s,o,r]=L(e,i,n),a=r!==e,l=x(t),c=l[r]||{},h=e.startsWith(".");if(void 0===o){if(h)for(const i of Object.keys(l))$(t,l,i,e.slice(1));for(const[i,n]of Object.entries(c)){const s=i.replace(w,"");a&&!e.includes(s)||D(t,l,r,n.callable,n.delegationSelector)}}else{if(!Object.keys(c).length)return;D(t,l,r,o,s?i:null)}},trigger(t,e,i){if("string"!=typeof e||!t)return null;const n=u();let s=null,o=!0,r=!0,a=!1;e!==I(e)&&n&&(s=n.Event(e,i),n(t).trigger(s),o=!s.isPropagationStopped(),r=!s.isImmediatePropagationStopped(),a=s.isDefaultPrevented());const l=P(new Event(e,{bubbles:o,cancelable:!0}),i);return a&&l.preventDefault(),r&&t.dispatchEvent(l),l.defaultPrevented&&s&&s.preventDefault(),l}};function P(t,e={}){for(const[i,n]of Object.entries(e))try{t[i]=n}catch(e){Object.defineProperty(t,i,{configurable:!0,get:()=>n})}return t}function M(t){if("true"===t)return!0;if("false"===t)return!1;if(t===Number(t).toString())return Number(t);if(""===t||"null"===t)return null;if("string"!=typeof t)return t;try{return JSON.parse(decodeURIComponent(t))}catch(e){return t}}function j(t){return t.replace(/[A-Z]/g,(t=>`-${t.toLowerCase()}`))}const F={setDataAttribute(t,e,i){t.setAttribute(`data-bs-${j(e)}`,i)},removeDataAttribute(t,e){t.removeAttribute(`data-bs-${j(e)}`)},getDataAttributes(t){if(!t)return{};const e={},i=Object.keys(t.dataset).filter((t=>t.startsWith("bs")&&!t.startsWith("bsConfig")));for(const n of i){let i=n.replace(/^bs/,"");i=i.charAt(0).toLowerCase()+i.slice(1,i.length),e[i]=M(t.dataset[n])}return e},getDataAttribute:(t,e)=>M(t.getAttribute(`data-bs-${j(e)}`))};class H{static get Default(){return{}}static get DefaultType(){return{}}static get NAME(){throw new Error('You have to implement the static method "NAME", for each component!')}_getConfig(t){return t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t}_mergeConfigObj(t,e){const i=o(e)?F.getDataAttribute(e,"config"):{};return{...this.constructor.Default,..."object"==typeof i?i:{},...o(e)?F.getDataAttributes(e):{},..."object"==typeof t?t:{}}}_typeCheckConfig(t,e=this.constructor.DefaultType){for(const[n,s]of Object.entries(e)){const e=t[n],r=o(e)?"element":null==(i=e)?`${i}`:Object.prototype.toString.call(i).match(/\s([a-z]+)/i)[1].toLowerCase();if(!new RegExp(s).test(r))throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option "${n}" provided type "${r}" but expected type "${s}".`)}var i}}class W extends H{constructor(t,i){super(),(t=r(t))&&(this._element=t,this._config=this._getConfig(i),e.set(this._element,this.constructor.DATA_KEY,this))}dispose(){e.remove(this._element,this.constructor.DATA_KEY),N.off(this._element,this.constructor.EVENT_KEY);for(const t of Object.getOwnPropertyNames(this))this[t]=null}_queueCallback(t,e,i=!0){_(t,e,i)}_getConfig(t){return t=this._mergeConfigObj(t,this._element),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}static getInstance(t){return e.get(r(t),this.DATA_KEY)}static getOrCreateInstance(t,e={}){return this.getInstance(t)||new this(t,"object"==typeof e?e:null)}static get VERSION(){return"5.3.1"}static get DATA_KEY(){return`bs.${this.NAME}`}static get EVENT_KEY(){return`.${this.DATA_KEY}`}static eventName(t){return`${t}${this.EVENT_KEY}`}}const B=t=>{let e=t.getAttribute("data-bs-target");if(!e||"#"===e){let i=t.getAttribute("href");if(!i||!i.includes("#")&&!i.startsWith("."))return null;i.includes("#")&&!i.startsWith("#")&&(i=`#${i.split("#")[1]}`),e=i&&"#"!==i?i.trim():null}return n(e)},z={find:(t,e=document.documentElement)=>[].concat(...Element.prototype.querySelectorAll.call(e,t)),findOne:(t,e=document.documentElement)=>Element.prototype.querySelector.call(e,t),children:(t,e)=>[].concat(...t.children).filter((t=>t.matches(e))),parents(t,e){const i=[];let n=t.parentNode.closest(e);for(;n;)i.push(n),n=n.parentNode.closest(e);return i},prev(t,e){let i=t.previousElementSibling;for(;i;){if(i.matches(e))return[i];i=i.previousElementSibling}return[]},next(t,e){let i=t.nextElementSibling;for(;i;){if(i.matches(e))return[i];i=i.nextElementSibling}return[]},focusableChildren(t){const e=["a","button","input","textarea","select","details","[tabindex]",'[contenteditable="true"]'].map((t=>`${t}:not([tabindex^="-"])`)).join(",");return this.find(e,t).filter((t=>!l(t)&&a(t)))},getSelectorFromElement(t){const e=B(t);return e&&z.findOne(e)?e:null},getElementFromSelector(t){const e=B(t);return e?z.findOne(e):null},getMultipleElementsFromSelector(t){const e=B(t);return e?z.find(e):[]}},R=(t,e="hide")=>{const i=`click.dismiss${t.EVENT_KEY}`,n=t.NAME;N.on(document,i,`[data-bs-dismiss="${n}"]`,(function(i){if(["A","AREA"].includes(this.tagName)&&i.preventDefault(),l(this))return;const s=z.getElementFromSelector(this)||this.closest(`.${n}`);t.getOrCreateInstance(s)[e]()}))},q=".bs.alert",V=`close${q}`,K=`closed${q}`;class Q extends W{static get NAME(){return"alert"}close(){if(N.trigger(this._element,V).defaultPrevented)return;this._element.classList.remove("show");const t=this._element.classList.contains("fade");this._queueCallback((()=>this._destroyElement()),this._element,t)}_destroyElement(){this._element.remove(),N.trigger(this._element,K),this.dispose()}static jQueryInterface(t){return this.each((function(){const e=Q.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}R(Q,"close"),m(Q);const X='[data-bs-toggle="button"]';class Y extends W{static get NAME(){return"button"}toggle(){this._element.setAttribute("aria-pressed",this._element.classList.toggle("active"))}static jQueryInterface(t){return this.each((function(){const e=Y.getOrCreateInstance(this);"toggle"===t&&e[t]()}))}}N.on(document,"click.bs.button.data-api",X,(t=>{t.preventDefault();const e=t.target.closest(X);Y.getOrCreateInstance(e).toggle()})),m(Y);const U=".bs.swipe",G=`touchstart${U}`,J=`touchmove${U}`,Z=`touchend${U}`,tt=`pointerdown${U}`,et=`pointerup${U}`,it={endCallback:null,leftCallback:null,rightCallback:null},nt={endCallback:"(function|null)",leftCallback:"(function|null)",rightCallback:"(function|null)"};class st extends H{constructor(t,e){super(),this._element=t,t&&st.isSupported()&&(this._config=this._getConfig(e),this._deltaX=0,this._supportPointerEvents=Boolean(window.PointerEvent),this._initEvents())}static get Default(){return it}static get DefaultType(){return nt}static get NAME(){return"swipe"}dispose(){N.off(this._element,U)}_start(t){this._supportPointerEvents?this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX):this._deltaX=t.touches[0].clientX}_end(t){this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX-this._deltaX),this._handleSwipe(),g(this._config.endCallback)}_move(t){this._deltaX=t.touches&&t.touches.length>1?0:t.touches[0].clientX-this._deltaX}_handleSwipe(){const t=Math.abs(this._deltaX);if(t<=40)return;const e=t/this._deltaX;this._deltaX=0,e&&g(e>0?this._config.rightCallback:this._config.leftCallback)}_initEvents(){this._supportPointerEvents?(N.on(this._element,tt,(t=>this._start(t))),N.on(this._element,et,(t=>this._end(t))),this._element.classList.add("pointer-event")):(N.on(this._element,G,(t=>this._start(t))),N.on(this._element,J,(t=>this._move(t))),N.on(this._element,Z,(t=>this._end(t))))}_eventIsPointerPenTouch(t){return this._supportPointerEvents&&("pen"===t.pointerType||"touch"===t.pointerType)}static isSupported(){return"ontouchstart"in document.documentElement||navigator.maxTouchPoints>0}}const ot=".bs.carousel",rt=".data-api",at="next",lt="prev",ct="left",ht="right",dt=`slide${ot}`,ut=`slid${ot}`,ft=`keydown${ot}`,pt=`mouseenter${ot}`,mt=`mouseleave${ot}`,gt=`dragstart${ot}`,_t=`load${ot}${rt}`,bt=`click${ot}${rt}`,vt="carousel",yt="active",wt=".active",At=".carousel-item",Et=wt+At,Tt={ArrowLeft:ht,ArrowRight:ct},Ct={interval:5e3,keyboard:!0,pause:"hover",ride:!1,touch:!0,wrap:!0},Ot={interval:"(number|boolean)",keyboard:"boolean",pause:"(string|boolean)",ride:"(boolean|string)",touch:"boolean",wrap:"boolean"};class xt extends W{constructor(t,e){super(t,e),this._interval=null,this._activeElement=null,this._isSliding=!1,this.touchTimeout=null,this._swipeHelper=null,this._indicatorsElement=z.findOne(".carousel-indicators",this._element),this._addEventListeners(),this._config.ride===vt&&this.cycle()}static get Default(){return Ct}static get DefaultType(){return Ot}static get NAME(){return"carousel"}next(){this._slide(at)}nextWhenVisible(){!document.hidden&&a(this._element)&&this.next()}prev(){this._slide(lt)}pause(){this._isSliding&&s(this._element),this._clearInterval()}cycle(){this._clearInterval(),this._updateInterval(),this._interval=setInterval((()=>this.nextWhenVisible()),this._config.interval)}_maybeEnableCycle(){this._config.ride&&(this._isSliding?N.one(this._element,ut,(()=>this.cycle())):this.cycle())}to(t){const e=this._getItems();if(t>e.length-1||t<0)return;if(this._isSliding)return void N.one(this._element,ut,(()=>this.to(t)));const i=this._getItemIndex(this._getActive());if(i===t)return;const n=t>i?at:lt;this._slide(n,e[t])}dispose(){this._swipeHelper&&this._swipeHelper.dispose(),super.dispose()}_configAfterMerge(t){return t.defaultInterval=t.interval,t}_addEventListeners(){this._config.keyboard&&N.on(this._element,ft,(t=>this._keydown(t))),"hover"===this._config.pause&&(N.on(this._element,pt,(()=>this.pause())),N.on(this._element,mt,(()=>this._maybeEnableCycle()))),this._config.touch&&st.isSupported()&&this._addTouchEventListeners()}_addTouchEventListeners(){for(const t of z.find(".carousel-item img",this._element))N.on(t,gt,(t=>t.preventDefault()));const t={leftCallback:()=>this._slide(this._directionToOrder(ct)),rightCallback:()=>this._slide(this._directionToOrder(ht)),endCallback:()=>{"hover"===this._config.pause&&(this.pause(),this.touchTimeout&&clearTimeout(this.touchTimeout),this.touchTimeout=setTimeout((()=>this._maybeEnableCycle()),500+this._config.interval))}};this._swipeHelper=new st(this._element,t)}_keydown(t){if(/input|textarea/i.test(t.target.tagName))return;const e=Tt[t.key];e&&(t.preventDefault(),this._slide(this._directionToOrder(e)))}_getItemIndex(t){return this._getItems().indexOf(t)}_setActiveIndicatorElement(t){if(!this._indicatorsElement)return;const e=z.findOne(wt,this._indicatorsElement);e.classList.remove(yt),e.removeAttribute("aria-current");const i=z.findOne(`[data-bs-slide-to="${t}"]`,this._indicatorsElement);i&&(i.classList.add(yt),i.setAttribute("aria-current","true"))}_updateInterval(){const t=this._activeElement||this._getActive();if(!t)return;const e=Number.parseInt(t.getAttribute("data-bs-interval"),10);this._config.interval=e||this._config.defaultInterval}_slide(t,e=null){if(this._isSliding)return;const i=this._getActive(),n=t===at,s=e||b(this._getItems(),i,n,this._config.wrap);if(s===i)return;const o=this._getItemIndex(s),r=e=>N.trigger(this._element,e,{relatedTarget:s,direction:this._orderToDirection(t),from:this._getItemIndex(i),to:o});if(r(dt).defaultPrevented)return;if(!i||!s)return;const a=Boolean(this._interval);this.pause(),this._isSliding=!0,this._setActiveIndicatorElement(o),this._activeElement=s;const l=n?"carousel-item-start":"carousel-item-end",c=n?"carousel-item-next":"carousel-item-prev";s.classList.add(c),d(s),i.classList.add(l),s.classList.add(l),this._queueCallback((()=>{s.classList.remove(l,c),s.classList.add(yt),i.classList.remove(yt,c,l),this._isSliding=!1,r(ut)}),i,this._isAnimated()),a&&this.cycle()}_isAnimated(){return this._element.classList.contains("slide")}_getActive(){return z.findOne(Et,this._element)}_getItems(){return z.find(At,this._element)}_clearInterval(){this._interval&&(clearInterval(this._interval),this._interval=null)}_directionToOrder(t){return p()?t===ct?lt:at:t===ct?at:lt}_orderToDirection(t){return p()?t===lt?ct:ht:t===lt?ht:ct}static jQueryInterface(t){return this.each((function(){const e=xt.getOrCreateInstance(this,t);if("number"!=typeof t){if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}else e.to(t)}))}}N.on(document,bt,"[data-bs-slide], [data-bs-slide-to]",(function(t){const e=z.getElementFromSelector(this);if(!e||!e.classList.contains(vt))return;t.preventDefault();const i=xt.getOrCreateInstance(e),n=this.getAttribute("data-bs-slide-to");return n?(i.to(n),void i._maybeEnableCycle()):"next"===F.getDataAttribute(this,"slide")?(i.next(),void i._maybeEnableCycle()):(i.prev(),void i._maybeEnableCycle())})),N.on(window,_t,(()=>{const t=z.find('[data-bs-ride="carousel"]');for(const e of t)xt.getOrCreateInstance(e)})),m(xt);const kt=".bs.collapse",Lt=`show${kt}`,St=`shown${kt}`,Dt=`hide${kt}`,$t=`hidden${kt}`,It=`click${kt}.data-api`,Nt="show",Pt="collapse",Mt="collapsing",jt=`:scope .${Pt} .${Pt}`,Ft='[data-bs-toggle="collapse"]',Ht={parent:null,toggle:!0},Wt={parent:"(null|element)",toggle:"boolean"};class Bt extends W{constructor(t,e){super(t,e),this._isTransitioning=!1,this._triggerArray=[];const i=z.find(Ft);for(const t of i){const e=z.getSelectorFromElement(t),i=z.find(e).filter((t=>t===this._element));null!==e&&i.length&&this._triggerArray.push(t)}this._initializeChildren(),this._config.parent||this._addAriaAndCollapsedClass(this._triggerArray,this._isShown()),this._config.toggle&&this.toggle()}static get Default(){return Ht}static get DefaultType(){return Wt}static get NAME(){return"collapse"}toggle(){this._isShown()?this.hide():this.show()}show(){if(this._isTransitioning||this._isShown())return;let t=[];if(this._config.parent&&(t=this._getFirstLevelChildren(".collapse.show, .collapse.collapsing").filter((t=>t!==this._element)).map((t=>Bt.getOrCreateInstance(t,{toggle:!1})))),t.length&&t[0]._isTransitioning)return;if(N.trigger(this._element,Lt).defaultPrevented)return;for(const e of t)e.hide();const e=this._getDimension();this._element.classList.remove(Pt),this._element.classList.add(Mt),this._element.style[e]=0,this._addAriaAndCollapsedClass(this._triggerArray,!0),this._isTransitioning=!0;const i=`scroll${e[0].toUpperCase()+e.slice(1)}`;this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(Mt),this._element.classList.add(Pt,Nt),this._element.style[e]="",N.trigger(this._element,St)}),this._element,!0),this._element.style[e]=`${this._element[i]}px`}hide(){if(this._isTransitioning||!this._isShown())return;if(N.trigger(this._element,Dt).defaultPrevented)return;const t=this._getDimension();this._element.style[t]=`${this._element.getBoundingClientRect()[t]}px`,d(this._element),this._element.classList.add(Mt),this._element.classList.remove(Pt,Nt);for(const t of this._triggerArray){const e=z.getElementFromSelector(t);e&&!this._isShown(e)&&this._addAriaAndCollapsedClass([t],!1)}this._isTransitioning=!0,this._element.style[t]="",this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(Mt),this._element.classList.add(Pt),N.trigger(this._element,$t)}),this._element,!0)}_isShown(t=this._element){return t.classList.contains(Nt)}_configAfterMerge(t){return t.toggle=Boolean(t.toggle),t.parent=r(t.parent),t}_getDimension(){return this._element.classList.contains("collapse-horizontal")?"width":"height"}_initializeChildren(){if(!this._config.parent)return;const t=this._getFirstLevelChildren(Ft);for(const e of t){const t=z.getElementFromSelector(e);t&&this._addAriaAndCollapsedClass([e],this._isShown(t))}}_getFirstLevelChildren(t){const e=z.find(jt,this._config.parent);return z.find(t,this._config.parent).filter((t=>!e.includes(t)))}_addAriaAndCollapsedClass(t,e){if(t.length)for(const i of t)i.classList.toggle("collapsed",!e),i.setAttribute("aria-expanded",e)}static jQueryInterface(t){const e={};return"string"==typeof t&&/show|hide/.test(t)&&(e.toggle=!1),this.each((function(){const i=Bt.getOrCreateInstance(this,e);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t]()}}))}}N.on(document,It,Ft,(function(t){("A"===t.target.tagName||t.delegateTarget&&"A"===t.delegateTarget.tagName)&&t.preventDefault();for(const t of z.getMultipleElementsFromSelector(this))Bt.getOrCreateInstance(t,{toggle:!1}).toggle()})),m(Bt);var zt="top",Rt="bottom",qt="right",Vt="left",Kt="auto",Qt=[zt,Rt,qt,Vt],Xt="start",Yt="end",Ut="clippingParents",Gt="viewport",Jt="popper",Zt="reference",te=Qt.reduce((function(t,e){return t.concat([e+"-"+Xt,e+"-"+Yt])}),[]),ee=[].concat(Qt,[Kt]).reduce((function(t,e){return t.concat([e,e+"-"+Xt,e+"-"+Yt])}),[]),ie="beforeRead",ne="read",se="afterRead",oe="beforeMain",re="main",ae="afterMain",le="beforeWrite",ce="write",he="afterWrite",de=[ie,ne,se,oe,re,ae,le,ce,he];function ue(t){return t?(t.nodeName||"").toLowerCase():null}function fe(t){if(null==t)return window;if("[object Window]"!==t.toString()){var e=t.ownerDocument;return e&&e.defaultView||window}return t}function pe(t){return t instanceof fe(t).Element||t instanceof Element}function me(t){return t instanceof fe(t).HTMLElement||t instanceof HTMLElement}function ge(t){return"undefined"!=typeof ShadowRoot&&(t instanceof fe(t).ShadowRoot||t instanceof ShadowRoot)}const _e={name:"applyStyles",enabled:!0,phase:"write",fn:function(t){var e=t.state;Object.keys(e.elements).forEach((function(t){var i=e.styles[t]||{},n=e.attributes[t]||{},s=e.elements[t];me(s)&&ue(s)&&(Object.assign(s.style,i),Object.keys(n).forEach((function(t){var e=n[t];!1===e?s.removeAttribute(t):s.setAttribute(t,!0===e?"":e)})))}))},effect:function(t){var e=t.state,i={popper:{position:e.options.strategy,left:"0",top:"0",margin:"0"},arrow:{position:"absolute"},reference:{}};return Object.assign(e.elements.popper.style,i.popper),e.styles=i,e.elements.arrow&&Object.assign(e.elements.arrow.style,i.arrow),function(){Object.keys(e.elements).forEach((function(t){var n=e.elements[t],s=e.attributes[t]||{},o=Object.keys(e.styles.hasOwnProperty(t)?e.styles[t]:i[t]).reduce((function(t,e){return t[e]="",t}),{});me(n)&&ue(n)&&(Object.assign(n.style,o),Object.keys(s).forEach((function(t){n.removeAttribute(t)})))}))}},requires:["computeStyles"]};function be(t){return t.split("-")[0]}var ve=Math.max,ye=Math.min,we=Math.round;function Ae(){var t=navigator.userAgentData;return null!=t&&t.brands&&Array.isArray(t.brands)?t.brands.map((function(t){return t.brand+"/"+t.version})).join(" "):navigator.userAgent}function Ee(){return!/^((?!chrome|android).)*safari/i.test(Ae())}function Te(t,e,i){void 0===e&&(e=!1),void 0===i&&(i=!1);var n=t.getBoundingClientRect(),s=1,o=1;e&&me(t)&&(s=t.offsetWidth>0&&we(n.width)/t.offsetWidth||1,o=t.offsetHeight>0&&we(n.height)/t.offsetHeight||1);var r=(pe(t)?fe(t):window).visualViewport,a=!Ee()&&i,l=(n.left+(a&&r?r.offsetLeft:0))/s,c=(n.top+(a&&r?r.offsetTop:0))/o,h=n.width/s,d=n.height/o;return{width:h,height:d,top:c,right:l+h,bottom:c+d,left:l,x:l,y:c}}function Ce(t){var e=Te(t),i=t.offsetWidth,n=t.offsetHeight;return Math.abs(e.width-i)<=1&&(i=e.width),Math.abs(e.height-n)<=1&&(n=e.height),{x:t.offsetLeft,y:t.offsetTop,width:i,height:n}}function Oe(t,e){var i=e.getRootNode&&e.getRootNode();if(t.contains(e))return!0;if(i&&ge(i)){var n=e;do{if(n&&t.isSameNode(n))return!0;n=n.parentNode||n.host}while(n)}return!1}function xe(t){return fe(t).getComputedStyle(t)}function ke(t){return["table","td","th"].indexOf(ue(t))>=0}function Le(t){return((pe(t)?t.ownerDocument:t.document)||window.document).documentElement}function Se(t){return"html"===ue(t)?t:t.assignedSlot||t.parentNode||(ge(t)?t.host:null)||Le(t)}function De(t){return me(t)&&"fixed"!==xe(t).position?t.offsetParent:null}function $e(t){for(var e=fe(t),i=De(t);i&&ke(i)&&"static"===xe(i).position;)i=De(i);return i&&("html"===ue(i)||"body"===ue(i)&&"static"===xe(i).position)?e:i||function(t){var e=/firefox/i.test(Ae());if(/Trident/i.test(Ae())&&me(t)&&"fixed"===xe(t).position)return null;var i=Se(t);for(ge(i)&&(i=i.host);me(i)&&["html","body"].indexOf(ue(i))<0;){var n=xe(i);if("none"!==n.transform||"none"!==n.perspective||"paint"===n.contain||-1!==["transform","perspective"].indexOf(n.willChange)||e&&"filter"===n.willChange||e&&n.filter&&"none"!==n.filter)return i;i=i.parentNode}return null}(t)||e}function Ie(t){return["top","bottom"].indexOf(t)>=0?"x":"y"}function Ne(t,e,i){return ve(t,ye(e,i))}function Pe(t){return Object.assign({},{top:0,right:0,bottom:0,left:0},t)}function Me(t,e){return e.reduce((function(e,i){return e[i]=t,e}),{})}const je={name:"arrow",enabled:!0,phase:"main",fn:function(t){var e,i=t.state,n=t.name,s=t.options,o=i.elements.arrow,r=i.modifiersData.popperOffsets,a=be(i.placement),l=Ie(a),c=[Vt,qt].indexOf(a)>=0?"height":"width";if(o&&r){var h=function(t,e){return Pe("number"!=typeof(t="function"==typeof t?t(Object.assign({},e.rects,{placement:e.placement})):t)?t:Me(t,Qt))}(s.padding,i),d=Ce(o),u="y"===l?zt:Vt,f="y"===l?Rt:qt,p=i.rects.reference[c]+i.rects.reference[l]-r[l]-i.rects.popper[c],m=r[l]-i.rects.reference[l],g=$e(o),_=g?"y"===l?g.clientHeight||0:g.clientWidth||0:0,b=p/2-m/2,v=h[u],y=_-d[c]-h[f],w=_/2-d[c]/2+b,A=Ne(v,w,y),E=l;i.modifiersData[n]=((e={})[E]=A,e.centerOffset=A-w,e)}},effect:function(t){var e=t.state,i=t.options.element,n=void 0===i?"[data-popper-arrow]":i;null!=n&&("string"!=typeof n||(n=e.elements.popper.querySelector(n)))&&Oe(e.elements.popper,n)&&(e.elements.arrow=n)},requires:["popperOffsets"],requiresIfExists:["preventOverflow"]};function Fe(t){return t.split("-")[1]}var He={top:"auto",right:"auto",bottom:"auto",left:"auto"};function We(t){var e,i=t.popper,n=t.popperRect,s=t.placement,o=t.variation,r=t.offsets,a=t.position,l=t.gpuAcceleration,c=t.adaptive,h=t.roundOffsets,d=t.isFixed,u=r.x,f=void 0===u?0:u,p=r.y,m=void 0===p?0:p,g="function"==typeof h?h({x:f,y:m}):{x:f,y:m};f=g.x,m=g.y;var _=r.hasOwnProperty("x"),b=r.hasOwnProperty("y"),v=Vt,y=zt,w=window;if(c){var A=$e(i),E="clientHeight",T="clientWidth";A===fe(i)&&"static"!==xe(A=Le(i)).position&&"absolute"===a&&(E="scrollHeight",T="scrollWidth"),(s===zt||(s===Vt||s===qt)&&o===Yt)&&(y=Rt,m-=(d&&A===w&&w.visualViewport?w.visualViewport.height:A[E])-n.height,m*=l?1:-1),s!==Vt&&(s!==zt&&s!==Rt||o!==Yt)||(v=qt,f-=(d&&A===w&&w.visualViewport?w.visualViewport.width:A[T])-n.width,f*=l?1:-1)}var C,O=Object.assign({position:a},c&&He),x=!0===h?function(t,e){var i=t.x,n=t.y,s=e.devicePixelRatio||1;return{x:we(i*s)/s||0,y:we(n*s)/s||0}}({x:f,y:m},fe(i)):{x:f,y:m};return f=x.x,m=x.y,l?Object.assign({},O,((C={})[y]=b?"0":"",C[v]=_?"0":"",C.transform=(w.devicePixelRatio||1)<=1?"translate("+f+"px, "+m+"px)":"translate3d("+f+"px, "+m+"px, 0)",C)):Object.assign({},O,((e={})[y]=b?m+"px":"",e[v]=_?f+"px":"",e.transform="",e))}const Be={name:"computeStyles",enabled:!0,phase:"beforeWrite",fn:function(t){var e=t.state,i=t.options,n=i.gpuAcceleration,s=void 0===n||n,o=i.adaptive,r=void 0===o||o,a=i.roundOffsets,l=void 0===a||a,c={placement:be(e.placement),variation:Fe(e.placement),popper:e.elements.popper,popperRect:e.rects.popper,gpuAcceleration:s,isFixed:"fixed"===e.options.strategy};null!=e.modifiersData.popperOffsets&&(e.styles.popper=Object.assign({},e.styles.popper,We(Object.assign({},c,{offsets:e.modifiersData.popperOffsets,position:e.options.strategy,adaptive:r,roundOffsets:l})))),null!=e.modifiersData.arrow&&(e.styles.arrow=Object.assign({},e.styles.arrow,We(Object.assign({},c,{offsets:e.modifiersData.arrow,position:"absolute",adaptive:!1,roundOffsets:l})))),e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-placement":e.placement})},data:{}};var ze={passive:!0};const Re={name:"eventListeners",enabled:!0,phase:"write",fn:function(){},effect:function(t){var e=t.state,i=t.instance,n=t.options,s=n.scroll,o=void 0===s||s,r=n.resize,a=void 0===r||r,l=fe(e.elements.popper),c=[].concat(e.scrollParents.reference,e.scrollParents.popper);return o&&c.forEach((function(t){t.addEventListener("scroll",i.update,ze)})),a&&l.addEventListener("resize",i.update,ze),function(){o&&c.forEach((function(t){t.removeEventListener("scroll",i.update,ze)})),a&&l.removeEventListener("resize",i.update,ze)}},data:{}};var qe={left:"right",right:"left",bottom:"top",top:"bottom"};function Ve(t){return t.replace(/left|right|bottom|top/g,(function(t){return qe[t]}))}var Ke={start:"end",end:"start"};function Qe(t){return t.replace(/start|end/g,(function(t){return Ke[t]}))}function Xe(t){var e=fe(t);return{scrollLeft:e.pageXOffset,scrollTop:e.pageYOffset}}function Ye(t){return Te(Le(t)).left+Xe(t).scrollLeft}function Ue(t){var e=xe(t),i=e.overflow,n=e.overflowX,s=e.overflowY;return/auto|scroll|overlay|hidden/.test(i+s+n)}function Ge(t){return["html","body","#document"].indexOf(ue(t))>=0?t.ownerDocument.body:me(t)&&Ue(t)?t:Ge(Se(t))}function Je(t,e){var i;void 0===e&&(e=[]);var n=Ge(t),s=n===(null==(i=t.ownerDocument)?void 0:i.body),o=fe(n),r=s?[o].concat(o.visualViewport||[],Ue(n)?n:[]):n,a=e.concat(r);return s?a:a.concat(Je(Se(r)))}function Ze(t){return Object.assign({},t,{left:t.x,top:t.y,right:t.x+t.width,bottom:t.y+t.height})}function ti(t,e,i){return e===Gt?Ze(function(t,e){var i=fe(t),n=Le(t),s=i.visualViewport,o=n.clientWidth,r=n.clientHeight,a=0,l=0;if(s){o=s.width,r=s.height;var c=Ee();(c||!c&&"fixed"===e)&&(a=s.offsetLeft,l=s.offsetTop)}return{width:o,height:r,x:a+Ye(t),y:l}}(t,i)):pe(e)?function(t,e){var i=Te(t,!1,"fixed"===e);return i.top=i.top+t.clientTop,i.left=i.left+t.clientLeft,i.bottom=i.top+t.clientHeight,i.right=i.left+t.clientWidth,i.width=t.clientWidth,i.height=t.clientHeight,i.x=i.left,i.y=i.top,i}(e,i):Ze(function(t){var e,i=Le(t),n=Xe(t),s=null==(e=t.ownerDocument)?void 0:e.body,o=ve(i.scrollWidth,i.clientWidth,s?s.scrollWidth:0,s?s.clientWidth:0),r=ve(i.scrollHeight,i.clientHeight,s?s.scrollHeight:0,s?s.clientHeight:0),a=-n.scrollLeft+Ye(t),l=-n.scrollTop;return"rtl"===xe(s||i).direction&&(a+=ve(i.clientWidth,s?s.clientWidth:0)-o),{width:o,height:r,x:a,y:l}}(Le(t)))}function ei(t){var e,i=t.reference,n=t.element,s=t.placement,o=s?be(s):null,r=s?Fe(s):null,a=i.x+i.width/2-n.width/2,l=i.y+i.height/2-n.height/2;switch(o){case zt:e={x:a,y:i.y-n.height};break;case Rt:e={x:a,y:i.y+i.height};break;case qt:e={x:i.x+i.width,y:l};break;case Vt:e={x:i.x-n.width,y:l};break;default:e={x:i.x,y:i.y}}var c=o?Ie(o):null;if(null!=c){var h="y"===c?"height":"width";switch(r){case Xt:e[c]=e[c]-(i[h]/2-n[h]/2);break;case Yt:e[c]=e[c]+(i[h]/2-n[h]/2)}}return e}function ii(t,e){void 0===e&&(e={});var i=e,n=i.placement,s=void 0===n?t.placement:n,o=i.strategy,r=void 0===o?t.strategy:o,a=i.boundary,l=void 0===a?Ut:a,c=i.rootBoundary,h=void 0===c?Gt:c,d=i.elementContext,u=void 0===d?Jt:d,f=i.altBoundary,p=void 0!==f&&f,m=i.padding,g=void 0===m?0:m,_=Pe("number"!=typeof g?g:Me(g,Qt)),b=u===Jt?Zt:Jt,v=t.rects.popper,y=t.elements[p?b:u],w=function(t,e,i,n){var s="clippingParents"===e?function(t){var e=Je(Se(t)),i=["absolute","fixed"].indexOf(xe(t).position)>=0&&me(t)?$e(t):t;return pe(i)?e.filter((function(t){return pe(t)&&Oe(t,i)&&"body"!==ue(t)})):[]}(t):[].concat(e),o=[].concat(s,[i]),r=o[0],a=o.reduce((function(e,i){var s=ti(t,i,n);return e.top=ve(s.top,e.top),e.right=ye(s.right,e.right),e.bottom=ye(s.bottom,e.bottom),e.left=ve(s.left,e.left),e}),ti(t,r,n));return a.width=a.right-a.left,a.height=a.bottom-a.top,a.x=a.left,a.y=a.top,a}(pe(y)?y:y.contextElement||Le(t.elements.popper),l,h,r),A=Te(t.elements.reference),E=ei({reference:A,element:v,strategy:"absolute",placement:s}),T=Ze(Object.assign({},v,E)),C=u===Jt?T:A,O={top:w.top-C.top+_.top,bottom:C.bottom-w.bottom+_.bottom,left:w.left-C.left+_.left,right:C.right-w.right+_.right},x=t.modifiersData.offset;if(u===Jt&&x){var k=x[s];Object.keys(O).forEach((function(t){var e=[qt,Rt].indexOf(t)>=0?1:-1,i=[zt,Rt].indexOf(t)>=0?"y":"x";O[t]+=k[i]*e}))}return O}function ni(t,e){void 0===e&&(e={});var i=e,n=i.placement,s=i.boundary,o=i.rootBoundary,r=i.padding,a=i.flipVariations,l=i.allowedAutoPlacements,c=void 0===l?ee:l,h=Fe(n),d=h?a?te:te.filter((function(t){return Fe(t)===h})):Qt,u=d.filter((function(t){return c.indexOf(t)>=0}));0===u.length&&(u=d);var f=u.reduce((function(e,i){return e[i]=ii(t,{placement:i,boundary:s,rootBoundary:o,padding:r})[be(i)],e}),{});return Object.keys(f).sort((function(t,e){return f[t]-f[e]}))}const si={name:"flip",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,n=t.name;if(!e.modifiersData[n]._skip){for(var s=i.mainAxis,o=void 0===s||s,r=i.altAxis,a=void 0===r||r,l=i.fallbackPlacements,c=i.padding,h=i.boundary,d=i.rootBoundary,u=i.altBoundary,f=i.flipVariations,p=void 0===f||f,m=i.allowedAutoPlacements,g=e.options.placement,_=be(g),b=l||(_!==g&&p?function(t){if(be(t)===Kt)return[];var e=Ve(t);return[Qe(t),e,Qe(e)]}(g):[Ve(g)]),v=[g].concat(b).reduce((function(t,i){return t.concat(be(i)===Kt?ni(e,{placement:i,boundary:h,rootBoundary:d,padding:c,flipVariations:p,allowedAutoPlacements:m}):i)}),[]),y=e.rects.reference,w=e.rects.popper,A=new Map,E=!0,T=v[0],C=0;C=0,S=L?"width":"height",D=ii(e,{placement:O,boundary:h,rootBoundary:d,altBoundary:u,padding:c}),$=L?k?qt:Vt:k?Rt:zt;y[S]>w[S]&&($=Ve($));var I=Ve($),N=[];if(o&&N.push(D[x]<=0),a&&N.push(D[$]<=0,D[I]<=0),N.every((function(t){return t}))){T=O,E=!1;break}A.set(O,N)}if(E)for(var P=function(t){var e=v.find((function(e){var i=A.get(e);if(i)return i.slice(0,t).every((function(t){return t}))}));if(e)return T=e,"break"},M=p?3:1;M>0&&"break"!==P(M);M--);e.placement!==T&&(e.modifiersData[n]._skip=!0,e.placement=T,e.reset=!0)}},requiresIfExists:["offset"],data:{_skip:!1}};function oi(t,e,i){return void 0===i&&(i={x:0,y:0}),{top:t.top-e.height-i.y,right:t.right-e.width+i.x,bottom:t.bottom-e.height+i.y,left:t.left-e.width-i.x}}function ri(t){return[zt,qt,Rt,Vt].some((function(e){return t[e]>=0}))}const ai={name:"hide",enabled:!0,phase:"main",requiresIfExists:["preventOverflow"],fn:function(t){var e=t.state,i=t.name,n=e.rects.reference,s=e.rects.popper,o=e.modifiersData.preventOverflow,r=ii(e,{elementContext:"reference"}),a=ii(e,{altBoundary:!0}),l=oi(r,n),c=oi(a,s,o),h=ri(l),d=ri(c);e.modifiersData[i]={referenceClippingOffsets:l,popperEscapeOffsets:c,isReferenceHidden:h,hasPopperEscaped:d},e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-reference-hidden":h,"data-popper-escaped":d})}},li={name:"offset",enabled:!0,phase:"main",requires:["popperOffsets"],fn:function(t){var e=t.state,i=t.options,n=t.name,s=i.offset,o=void 0===s?[0,0]:s,r=ee.reduce((function(t,i){return t[i]=function(t,e,i){var n=be(t),s=[Vt,zt].indexOf(n)>=0?-1:1,o="function"==typeof i?i(Object.assign({},e,{placement:t})):i,r=o[0],a=o[1];return r=r||0,a=(a||0)*s,[Vt,qt].indexOf(n)>=0?{x:a,y:r}:{x:r,y:a}}(i,e.rects,o),t}),{}),a=r[e.placement],l=a.x,c=a.y;null!=e.modifiersData.popperOffsets&&(e.modifiersData.popperOffsets.x+=l,e.modifiersData.popperOffsets.y+=c),e.modifiersData[n]=r}},ci={name:"popperOffsets",enabled:!0,phase:"read",fn:function(t){var e=t.state,i=t.name;e.modifiersData[i]=ei({reference:e.rects.reference,element:e.rects.popper,strategy:"absolute",placement:e.placement})},data:{}},hi={name:"preventOverflow",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,n=t.name,s=i.mainAxis,o=void 0===s||s,r=i.altAxis,a=void 0!==r&&r,l=i.boundary,c=i.rootBoundary,h=i.altBoundary,d=i.padding,u=i.tether,f=void 0===u||u,p=i.tetherOffset,m=void 0===p?0:p,g=ii(e,{boundary:l,rootBoundary:c,padding:d,altBoundary:h}),_=be(e.placement),b=Fe(e.placement),v=!b,y=Ie(_),w="x"===y?"y":"x",A=e.modifiersData.popperOffsets,E=e.rects.reference,T=e.rects.popper,C="function"==typeof m?m(Object.assign({},e.rects,{placement:e.placement})):m,O="number"==typeof C?{mainAxis:C,altAxis:C}:Object.assign({mainAxis:0,altAxis:0},C),x=e.modifiersData.offset?e.modifiersData.offset[e.placement]:null,k={x:0,y:0};if(A){if(o){var L,S="y"===y?zt:Vt,D="y"===y?Rt:qt,$="y"===y?"height":"width",I=A[y],N=I+g[S],P=I-g[D],M=f?-T[$]/2:0,j=b===Xt?E[$]:T[$],F=b===Xt?-T[$]:-E[$],H=e.elements.arrow,W=f&&H?Ce(H):{width:0,height:0},B=e.modifiersData["arrow#persistent"]?e.modifiersData["arrow#persistent"].padding:{top:0,right:0,bottom:0,left:0},z=B[S],R=B[D],q=Ne(0,E[$],W[$]),V=v?E[$]/2-M-q-z-O.mainAxis:j-q-z-O.mainAxis,K=v?-E[$]/2+M+q+R+O.mainAxis:F+q+R+O.mainAxis,Q=e.elements.arrow&&$e(e.elements.arrow),X=Q?"y"===y?Q.clientTop||0:Q.clientLeft||0:0,Y=null!=(L=null==x?void 0:x[y])?L:0,U=I+K-Y,G=Ne(f?ye(N,I+V-Y-X):N,I,f?ve(P,U):P);A[y]=G,k[y]=G-I}if(a){var J,Z="x"===y?zt:Vt,tt="x"===y?Rt:qt,et=A[w],it="y"===w?"height":"width",nt=et+g[Z],st=et-g[tt],ot=-1!==[zt,Vt].indexOf(_),rt=null!=(J=null==x?void 0:x[w])?J:0,at=ot?nt:et-E[it]-T[it]-rt+O.altAxis,lt=ot?et+E[it]+T[it]-rt-O.altAxis:st,ct=f&&ot?function(t,e,i){var n=Ne(t,e,i);return n>i?i:n}(at,et,lt):Ne(f?at:nt,et,f?lt:st);A[w]=ct,k[w]=ct-et}e.modifiersData[n]=k}},requiresIfExists:["offset"]};function di(t,e,i){void 0===i&&(i=!1);var n,s,o=me(e),r=me(e)&&function(t){var e=t.getBoundingClientRect(),i=we(e.width)/t.offsetWidth||1,n=we(e.height)/t.offsetHeight||1;return 1!==i||1!==n}(e),a=Le(e),l=Te(t,r,i),c={scrollLeft:0,scrollTop:0},h={x:0,y:0};return(o||!o&&!i)&&(("body"!==ue(e)||Ue(a))&&(c=(n=e)!==fe(n)&&me(n)?{scrollLeft:(s=n).scrollLeft,scrollTop:s.scrollTop}:Xe(n)),me(e)?((h=Te(e,!0)).x+=e.clientLeft,h.y+=e.clientTop):a&&(h.x=Ye(a))),{x:l.left+c.scrollLeft-h.x,y:l.top+c.scrollTop-h.y,width:l.width,height:l.height}}function ui(t){var e=new Map,i=new Set,n=[];function s(t){i.add(t.name),[].concat(t.requires||[],t.requiresIfExists||[]).forEach((function(t){if(!i.has(t)){var n=e.get(t);n&&s(n)}})),n.push(t)}return t.forEach((function(t){e.set(t.name,t)})),t.forEach((function(t){i.has(t.name)||s(t)})),n}var fi={placement:"bottom",modifiers:[],strategy:"absolute"};function pi(){for(var t=arguments.length,e=new Array(t),i=0;iNumber.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_getPopperConfig(){const t={placement:this._getPlacement(),modifiers:[{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"offset",options:{offset:this._getOffset()}}]};return(this._inNavbar||"static"===this._config.display)&&(F.setDataAttribute(this._menu,"popper","static"),t.modifiers=[{name:"applyStyles",enabled:!1}]),{...t,...g(this._config.popperConfig,[t])}}_selectMenuItem({key:t,target:e}){const i=z.find(".dropdown-menu .dropdown-item:not(.disabled):not(:disabled)",this._menu).filter((t=>a(t)));i.length&&b(i,e,t===Ti,!i.includes(e)).focus()}static jQueryInterface(t){return this.each((function(){const e=qi.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}static clearMenus(t){if(2===t.button||"keyup"===t.type&&"Tab"!==t.key)return;const e=z.find(Ni);for(const i of e){const e=qi.getInstance(i);if(!e||!1===e._config.autoClose)continue;const n=t.composedPath(),s=n.includes(e._menu);if(n.includes(e._element)||"inside"===e._config.autoClose&&!s||"outside"===e._config.autoClose&&s)continue;if(e._menu.contains(t.target)&&("keyup"===t.type&&"Tab"===t.key||/input|select|option|textarea|form/i.test(t.target.tagName)))continue;const o={relatedTarget:e._element};"click"===t.type&&(o.clickEvent=t),e._completeHide(o)}}static dataApiKeydownHandler(t){const e=/input|textarea/i.test(t.target.tagName),i="Escape"===t.key,n=[Ei,Ti].includes(t.key);if(!n&&!i)return;if(e&&!i)return;t.preventDefault();const s=this.matches(Ii)?this:z.prev(this,Ii)[0]||z.next(this,Ii)[0]||z.findOne(Ii,t.delegateTarget.parentNode),o=qi.getOrCreateInstance(s);if(n)return t.stopPropagation(),o.show(),void o._selectMenuItem(t);o._isShown()&&(t.stopPropagation(),o.hide(),s.focus())}}N.on(document,Si,Ii,qi.dataApiKeydownHandler),N.on(document,Si,Pi,qi.dataApiKeydownHandler),N.on(document,Li,qi.clearMenus),N.on(document,Di,qi.clearMenus),N.on(document,Li,Ii,(function(t){t.preventDefault(),qi.getOrCreateInstance(this).toggle()})),m(qi);const Vi="backdrop",Ki="show",Qi=`mousedown.bs.${Vi}`,Xi={className:"modal-backdrop",clickCallback:null,isAnimated:!1,isVisible:!0,rootElement:"body"},Yi={className:"string",clickCallback:"(function|null)",isAnimated:"boolean",isVisible:"boolean",rootElement:"(element|string)"};class Ui extends H{constructor(t){super(),this._config=this._getConfig(t),this._isAppended=!1,this._element=null}static get Default(){return Xi}static get DefaultType(){return Yi}static get NAME(){return Vi}show(t){if(!this._config.isVisible)return void g(t);this._append();const e=this._getElement();this._config.isAnimated&&d(e),e.classList.add(Ki),this._emulateAnimation((()=>{g(t)}))}hide(t){this._config.isVisible?(this._getElement().classList.remove(Ki),this._emulateAnimation((()=>{this.dispose(),g(t)}))):g(t)}dispose(){this._isAppended&&(N.off(this._element,Qi),this._element.remove(),this._isAppended=!1)}_getElement(){if(!this._element){const t=document.createElement("div");t.className=this._config.className,this._config.isAnimated&&t.classList.add("fade"),this._element=t}return this._element}_configAfterMerge(t){return t.rootElement=r(t.rootElement),t}_append(){if(this._isAppended)return;const t=this._getElement();this._config.rootElement.append(t),N.on(t,Qi,(()=>{g(this._config.clickCallback)})),this._isAppended=!0}_emulateAnimation(t){_(t,this._getElement(),this._config.isAnimated)}}const Gi=".bs.focustrap",Ji=`focusin${Gi}`,Zi=`keydown.tab${Gi}`,tn="backward",en={autofocus:!0,trapElement:null},nn={autofocus:"boolean",trapElement:"element"};class sn extends H{constructor(t){super(),this._config=this._getConfig(t),this._isActive=!1,this._lastTabNavDirection=null}static get Default(){return en}static get DefaultType(){return nn}static get NAME(){return"focustrap"}activate(){this._isActive||(this._config.autofocus&&this._config.trapElement.focus(),N.off(document,Gi),N.on(document,Ji,(t=>this._handleFocusin(t))),N.on(document,Zi,(t=>this._handleKeydown(t))),this._isActive=!0)}deactivate(){this._isActive&&(this._isActive=!1,N.off(document,Gi))}_handleFocusin(t){const{trapElement:e}=this._config;if(t.target===document||t.target===e||e.contains(t.target))return;const i=z.focusableChildren(e);0===i.length?e.focus():this._lastTabNavDirection===tn?i[i.length-1].focus():i[0].focus()}_handleKeydown(t){"Tab"===t.key&&(this._lastTabNavDirection=t.shiftKey?tn:"forward")}}const on=".fixed-top, .fixed-bottom, .is-fixed, .sticky-top",rn=".sticky-top",an="padding-right",ln="margin-right";class cn{constructor(){this._element=document.body}getWidth(){const t=document.documentElement.clientWidth;return Math.abs(window.innerWidth-t)}hide(){const t=this.getWidth();this._disableOverFlow(),this._setElementAttributes(this._element,an,(e=>e+t)),this._setElementAttributes(on,an,(e=>e+t)),this._setElementAttributes(rn,ln,(e=>e-t))}reset(){this._resetElementAttributes(this._element,"overflow"),this._resetElementAttributes(this._element,an),this._resetElementAttributes(on,an),this._resetElementAttributes(rn,ln)}isOverflowing(){return this.getWidth()>0}_disableOverFlow(){this._saveInitialAttribute(this._element,"overflow"),this._element.style.overflow="hidden"}_setElementAttributes(t,e,i){const n=this.getWidth();this._applyManipulationCallback(t,(t=>{if(t!==this._element&&window.innerWidth>t.clientWidth+n)return;this._saveInitialAttribute(t,e);const s=window.getComputedStyle(t).getPropertyValue(e);t.style.setProperty(e,`${i(Number.parseFloat(s))}px`)}))}_saveInitialAttribute(t,e){const i=t.style.getPropertyValue(e);i&&F.setDataAttribute(t,e,i)}_resetElementAttributes(t,e){this._applyManipulationCallback(t,(t=>{const i=F.getDataAttribute(t,e);null!==i?(F.removeDataAttribute(t,e),t.style.setProperty(e,i)):t.style.removeProperty(e)}))}_applyManipulationCallback(t,e){if(o(t))e(t);else for(const i of z.find(t,this._element))e(i)}}const hn=".bs.modal",dn=`hide${hn}`,un=`hidePrevented${hn}`,fn=`hidden${hn}`,pn=`show${hn}`,mn=`shown${hn}`,gn=`resize${hn}`,_n=`click.dismiss${hn}`,bn=`mousedown.dismiss${hn}`,vn=`keydown.dismiss${hn}`,yn=`click${hn}.data-api`,wn="modal-open",An="show",En="modal-static",Tn={backdrop:!0,focus:!0,keyboard:!0},Cn={backdrop:"(boolean|string)",focus:"boolean",keyboard:"boolean"};class On extends W{constructor(t,e){super(t,e),this._dialog=z.findOne(".modal-dialog",this._element),this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._isShown=!1,this._isTransitioning=!1,this._scrollBar=new cn,this._addEventListeners()}static get Default(){return Tn}static get DefaultType(){return Cn}static get NAME(){return"modal"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||this._isTransitioning||N.trigger(this._element,pn,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._isTransitioning=!0,this._scrollBar.hide(),document.body.classList.add(wn),this._adjustDialog(),this._backdrop.show((()=>this._showElement(t))))}hide(){this._isShown&&!this._isTransitioning&&(N.trigger(this._element,dn).defaultPrevented||(this._isShown=!1,this._isTransitioning=!0,this._focustrap.deactivate(),this._element.classList.remove(An),this._queueCallback((()=>this._hideModal()),this._element,this._isAnimated())))}dispose(){N.off(window,hn),N.off(this._dialog,hn),this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}handleUpdate(){this._adjustDialog()}_initializeBackDrop(){return new Ui({isVisible:Boolean(this._config.backdrop),isAnimated:this._isAnimated()})}_initializeFocusTrap(){return new sn({trapElement:this._element})}_showElement(t){document.body.contains(this._element)||document.body.append(this._element),this._element.style.display="block",this._element.removeAttribute("aria-hidden"),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.scrollTop=0;const e=z.findOne(".modal-body",this._dialog);e&&(e.scrollTop=0),d(this._element),this._element.classList.add(An),this._queueCallback((()=>{this._config.focus&&this._focustrap.activate(),this._isTransitioning=!1,N.trigger(this._element,mn,{relatedTarget:t})}),this._dialog,this._isAnimated())}_addEventListeners(){N.on(this._element,vn,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():this._triggerBackdropTransition())})),N.on(window,gn,(()=>{this._isShown&&!this._isTransitioning&&this._adjustDialog()})),N.on(this._element,bn,(t=>{N.one(this._element,_n,(e=>{this._element===t.target&&this._element===e.target&&("static"!==this._config.backdrop?this._config.backdrop&&this.hide():this._triggerBackdropTransition())}))}))}_hideModal(){this._element.style.display="none",this._element.setAttribute("aria-hidden",!0),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._isTransitioning=!1,this._backdrop.hide((()=>{document.body.classList.remove(wn),this._resetAdjustments(),this._scrollBar.reset(),N.trigger(this._element,fn)}))}_isAnimated(){return this._element.classList.contains("fade")}_triggerBackdropTransition(){if(N.trigger(this._element,un).defaultPrevented)return;const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._element.style.overflowY;"hidden"===e||this._element.classList.contains(En)||(t||(this._element.style.overflowY="hidden"),this._element.classList.add(En),this._queueCallback((()=>{this._element.classList.remove(En),this._queueCallback((()=>{this._element.style.overflowY=e}),this._dialog)}),this._dialog),this._element.focus())}_adjustDialog(){const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._scrollBar.getWidth(),i=e>0;if(i&&!t){const t=p()?"paddingLeft":"paddingRight";this._element.style[t]=`${e}px`}if(!i&&t){const t=p()?"paddingRight":"paddingLeft";this._element.style[t]=`${e}px`}}_resetAdjustments(){this._element.style.paddingLeft="",this._element.style.paddingRight=""}static jQueryInterface(t,e){return this.each((function(){const i=On.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t](e)}}))}}N.on(document,yn,'[data-bs-toggle="modal"]',(function(t){const e=z.getElementFromSelector(this);["A","AREA"].includes(this.tagName)&&t.preventDefault(),N.one(e,pn,(t=>{t.defaultPrevented||N.one(e,fn,(()=>{a(this)&&this.focus()}))}));const i=z.findOne(".modal.show");i&&On.getInstance(i).hide(),On.getOrCreateInstance(e).toggle(this)})),R(On),m(On);const xn=".bs.offcanvas",kn=".data-api",Ln=`load${xn}${kn}`,Sn="show",Dn="showing",$n="hiding",In=".offcanvas.show",Nn=`show${xn}`,Pn=`shown${xn}`,Mn=`hide${xn}`,jn=`hidePrevented${xn}`,Fn=`hidden${xn}`,Hn=`resize${xn}`,Wn=`click${xn}${kn}`,Bn=`keydown.dismiss${xn}`,zn={backdrop:!0,keyboard:!0,scroll:!1},Rn={backdrop:"(boolean|string)",keyboard:"boolean",scroll:"boolean"};class qn extends W{constructor(t,e){super(t,e),this._isShown=!1,this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._addEventListeners()}static get Default(){return zn}static get DefaultType(){return Rn}static get NAME(){return"offcanvas"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||N.trigger(this._element,Nn,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._backdrop.show(),this._config.scroll||(new cn).hide(),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.classList.add(Dn),this._queueCallback((()=>{this._config.scroll&&!this._config.backdrop||this._focustrap.activate(),this._element.classList.add(Sn),this._element.classList.remove(Dn),N.trigger(this._element,Pn,{relatedTarget:t})}),this._element,!0))}hide(){this._isShown&&(N.trigger(this._element,Mn).defaultPrevented||(this._focustrap.deactivate(),this._element.blur(),this._isShown=!1,this._element.classList.add($n),this._backdrop.hide(),this._queueCallback((()=>{this._element.classList.remove(Sn,$n),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._config.scroll||(new cn).reset(),N.trigger(this._element,Fn)}),this._element,!0)))}dispose(){this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}_initializeBackDrop(){const t=Boolean(this._config.backdrop);return new Ui({className:"offcanvas-backdrop",isVisible:t,isAnimated:!0,rootElement:this._element.parentNode,clickCallback:t?()=>{"static"!==this._config.backdrop?this.hide():N.trigger(this._element,jn)}:null})}_initializeFocusTrap(){return new sn({trapElement:this._element})}_addEventListeners(){N.on(this._element,Bn,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():N.trigger(this._element,jn))}))}static jQueryInterface(t){return this.each((function(){const e=qn.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}N.on(document,Wn,'[data-bs-toggle="offcanvas"]',(function(t){const e=z.getElementFromSelector(this);if(["A","AREA"].includes(this.tagName)&&t.preventDefault(),l(this))return;N.one(e,Fn,(()=>{a(this)&&this.focus()}));const i=z.findOne(In);i&&i!==e&&qn.getInstance(i).hide(),qn.getOrCreateInstance(e).toggle(this)})),N.on(window,Ln,(()=>{for(const t of z.find(In))qn.getOrCreateInstance(t).show()})),N.on(window,Hn,(()=>{for(const t of z.find("[aria-modal][class*=show][class*=offcanvas-]"))"fixed"!==getComputedStyle(t).position&&qn.getOrCreateInstance(t).hide()})),R(qn),m(qn);const Vn={"*":["class","dir","id","lang","role",/^aria-[\w-]*$/i],a:["target","href","title","rel"],area:[],b:[],br:[],col:[],code:[],div:[],em:[],hr:[],h1:[],h2:[],h3:[],h4:[],h5:[],h6:[],i:[],img:["src","srcset","alt","title","width","height"],li:[],ol:[],p:[],pre:[],s:[],small:[],span:[],sub:[],sup:[],strong:[],u:[],ul:[]},Kn=new Set(["background","cite","href","itemtype","longdesc","poster","src","xlink:href"]),Qn=/^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i,Xn=(t,e)=>{const i=t.nodeName.toLowerCase();return e.includes(i)?!Kn.has(i)||Boolean(Qn.test(t.nodeValue)):e.filter((t=>t instanceof RegExp)).some((t=>t.test(i)))},Yn={allowList:Vn,content:{},extraClass:"",html:!1,sanitize:!0,sanitizeFn:null,template:"
"},Un={allowList:"object",content:"object",extraClass:"(string|function)",html:"boolean",sanitize:"boolean",sanitizeFn:"(null|function)",template:"string"},Gn={entry:"(string|element|function|null)",selector:"(string|element)"};class Jn extends H{constructor(t){super(),this._config=this._getConfig(t)}static get Default(){return Yn}static get DefaultType(){return Un}static get NAME(){return"TemplateFactory"}getContent(){return Object.values(this._config.content).map((t=>this._resolvePossibleFunction(t))).filter(Boolean)}hasContent(){return this.getContent().length>0}changeContent(t){return this._checkContent(t),this._config.content={...this._config.content,...t},this}toHtml(){const t=document.createElement("div");t.innerHTML=this._maybeSanitize(this._config.template);for(const[e,i]of Object.entries(this._config.content))this._setContent(t,i,e);const e=t.children[0],i=this._resolvePossibleFunction(this._config.extraClass);return i&&e.classList.add(...i.split(" ")),e}_typeCheckConfig(t){super._typeCheckConfig(t),this._checkContent(t.content)}_checkContent(t){for(const[e,i]of Object.entries(t))super._typeCheckConfig({selector:e,entry:i},Gn)}_setContent(t,e,i){const n=z.findOne(i,t);n&&((e=this._resolvePossibleFunction(e))?o(e)?this._putElementInTemplate(r(e),n):this._config.html?n.innerHTML=this._maybeSanitize(e):n.textContent=e:n.remove())}_maybeSanitize(t){return this._config.sanitize?function(t,e,i){if(!t.length)return t;if(i&&"function"==typeof i)return i(t);const n=(new window.DOMParser).parseFromString(t,"text/html"),s=[].concat(...n.body.querySelectorAll("*"));for(const t of s){const i=t.nodeName.toLowerCase();if(!Object.keys(e).includes(i)){t.remove();continue}const n=[].concat(...t.attributes),s=[].concat(e["*"]||[],e[i]||[]);for(const e of n)Xn(e,s)||t.removeAttribute(e.nodeName)}return n.body.innerHTML}(t,this._config.allowList,this._config.sanitizeFn):t}_resolvePossibleFunction(t){return g(t,[this])}_putElementInTemplate(t,e){if(this._config.html)return e.innerHTML="",void e.append(t);e.textContent=t.textContent}}const Zn=new Set(["sanitize","allowList","sanitizeFn"]),ts="fade",es="show",is=".modal",ns="hide.bs.modal",ss="hover",os="focus",rs={AUTO:"auto",TOP:"top",RIGHT:p()?"left":"right",BOTTOM:"bottom",LEFT:p()?"right":"left"},as={allowList:Vn,animation:!0,boundary:"clippingParents",container:!1,customClass:"",delay:0,fallbackPlacements:["top","right","bottom","left"],html:!1,offset:[0,6],placement:"top",popperConfig:null,sanitize:!0,sanitizeFn:null,selector:!1,template:'',title:"",trigger:"hover focus"},ls={allowList:"object",animation:"boolean",boundary:"(string|element)",container:"(string|element|boolean)",customClass:"(string|function)",delay:"(number|object)",fallbackPlacements:"array",html:"boolean",offset:"(array|string|function)",placement:"(string|function)",popperConfig:"(null|object|function)",sanitize:"boolean",sanitizeFn:"(null|function)",selector:"(string|boolean)",template:"string",title:"(string|element|function)",trigger:"string"};class cs extends W{constructor(t,e){if(void 0===vi)throw new TypeError("Bootstrap's tooltips require Popper (https://popper.js.org)");super(t,e),this._isEnabled=!0,this._timeout=0,this._isHovered=null,this._activeTrigger={},this._popper=null,this._templateFactory=null,this._newContent=null,this.tip=null,this._setListeners(),this._config.selector||this._fixTitle()}static get Default(){return as}static get DefaultType(){return ls}static get NAME(){return"tooltip"}enable(){this._isEnabled=!0}disable(){this._isEnabled=!1}toggleEnabled(){this._isEnabled=!this._isEnabled}toggle(){this._isEnabled&&(this._activeTrigger.click=!this._activeTrigger.click,this._isShown()?this._leave():this._enter())}dispose(){clearTimeout(this._timeout),N.off(this._element.closest(is),ns,this._hideModalHandler),this._element.getAttribute("data-bs-original-title")&&this._element.setAttribute("title",this._element.getAttribute("data-bs-original-title")),this._disposePopper(),super.dispose()}show(){if("none"===this._element.style.display)throw new Error("Please use show on visible elements");if(!this._isWithContent()||!this._isEnabled)return;const t=N.trigger(this._element,this.constructor.eventName("show")),e=(c(this._element)||this._element.ownerDocument.documentElement).contains(this._element);if(t.defaultPrevented||!e)return;this._disposePopper();const i=this._getTipElement();this._element.setAttribute("aria-describedby",i.getAttribute("id"));const{container:n}=this._config;if(this._element.ownerDocument.documentElement.contains(this.tip)||(n.append(i),N.trigger(this._element,this.constructor.eventName("inserted"))),this._popper=this._createPopper(i),i.classList.add(es),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))N.on(t,"mouseover",h);this._queueCallback((()=>{N.trigger(this._element,this.constructor.eventName("shown")),!1===this._isHovered&&this._leave(),this._isHovered=!1}),this.tip,this._isAnimated())}hide(){if(this._isShown()&&!N.trigger(this._element,this.constructor.eventName("hide")).defaultPrevented){if(this._getTipElement().classList.remove(es),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))N.off(t,"mouseover",h);this._activeTrigger.click=!1,this._activeTrigger[os]=!1,this._activeTrigger[ss]=!1,this._isHovered=null,this._queueCallback((()=>{this._isWithActiveTrigger()||(this._isHovered||this._disposePopper(),this._element.removeAttribute("aria-describedby"),N.trigger(this._element,this.constructor.eventName("hidden")))}),this.tip,this._isAnimated())}}update(){this._popper&&this._popper.update()}_isWithContent(){return Boolean(this._getTitle())}_getTipElement(){return this.tip||(this.tip=this._createTipElement(this._newContent||this._getContentForTemplate())),this.tip}_createTipElement(t){const e=this._getTemplateFactory(t).toHtml();if(!e)return null;e.classList.remove(ts,es),e.classList.add(`bs-${this.constructor.NAME}-auto`);const i=(t=>{do{t+=Math.floor(1e6*Math.random())}while(document.getElementById(t));return t})(this.constructor.NAME).toString();return e.setAttribute("id",i),this._isAnimated()&&e.classList.add(ts),e}setContent(t){this._newContent=t,this._isShown()&&(this._disposePopper(),this.show())}_getTemplateFactory(t){return this._templateFactory?this._templateFactory.changeContent(t):this._templateFactory=new Jn({...this._config,content:t,extraClass:this._resolvePossibleFunction(this._config.customClass)}),this._templateFactory}_getContentForTemplate(){return{".tooltip-inner":this._getTitle()}}_getTitle(){return this._resolvePossibleFunction(this._config.title)||this._element.getAttribute("data-bs-original-title")}_initializeOnDelegatedTarget(t){return this.constructor.getOrCreateInstance(t.delegateTarget,this._getDelegateConfig())}_isAnimated(){return this._config.animation||this.tip&&this.tip.classList.contains(ts)}_isShown(){return this.tip&&this.tip.classList.contains(es)}_createPopper(t){const e=g(this._config.placement,[this,t,this._element]),i=rs[e.toUpperCase()];return bi(this._element,t,this._getPopperConfig(i))}_getOffset(){const{offset:t}=this._config;return"string"==typeof t?t.split(",").map((t=>Number.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_resolvePossibleFunction(t){return g(t,[this._element])}_getPopperConfig(t){const e={placement:t,modifiers:[{name:"flip",options:{fallbackPlacements:this._config.fallbackPlacements}},{name:"offset",options:{offset:this._getOffset()}},{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"arrow",options:{element:`.${this.constructor.NAME}-arrow`}},{name:"preSetPlacement",enabled:!0,phase:"beforeMain",fn:t=>{this._getTipElement().setAttribute("data-popper-placement",t.state.placement)}}]};return{...e,...g(this._config.popperConfig,[e])}}_setListeners(){const t=this._config.trigger.split(" ");for(const e of t)if("click"===e)N.on(this._element,this.constructor.eventName("click"),this._config.selector,(t=>{this._initializeOnDelegatedTarget(t).toggle()}));else if("manual"!==e){const t=e===ss?this.constructor.eventName("mouseenter"):this.constructor.eventName("focusin"),i=e===ss?this.constructor.eventName("mouseleave"):this.constructor.eventName("focusout");N.on(this._element,t,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusin"===t.type?os:ss]=!0,e._enter()})),N.on(this._element,i,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusout"===t.type?os:ss]=e._element.contains(t.relatedTarget),e._leave()}))}this._hideModalHandler=()=>{this._element&&this.hide()},N.on(this._element.closest(is),ns,this._hideModalHandler)}_fixTitle(){const t=this._element.getAttribute("title");t&&(this._element.getAttribute("aria-label")||this._element.textContent.trim()||this._element.setAttribute("aria-label",t),this._element.setAttribute("data-bs-original-title",t),this._element.removeAttribute("title"))}_enter(){this._isShown()||this._isHovered?this._isHovered=!0:(this._isHovered=!0,this._setTimeout((()=>{this._isHovered&&this.show()}),this._config.delay.show))}_leave(){this._isWithActiveTrigger()||(this._isHovered=!1,this._setTimeout((()=>{this._isHovered||this.hide()}),this._config.delay.hide))}_setTimeout(t,e){clearTimeout(this._timeout),this._timeout=setTimeout(t,e)}_isWithActiveTrigger(){return Object.values(this._activeTrigger).includes(!0)}_getConfig(t){const e=F.getDataAttributes(this._element);for(const t of Object.keys(e))Zn.has(t)&&delete e[t];return t={...e,..."object"==typeof t&&t?t:{}},t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t.container=!1===t.container?document.body:r(t.container),"number"==typeof t.delay&&(t.delay={show:t.delay,hide:t.delay}),"number"==typeof t.title&&(t.title=t.title.toString()),"number"==typeof t.content&&(t.content=t.content.toString()),t}_getDelegateConfig(){const t={};for(const[e,i]of Object.entries(this._config))this.constructor.Default[e]!==i&&(t[e]=i);return t.selector=!1,t.trigger="manual",t}_disposePopper(){this._popper&&(this._popper.destroy(),this._popper=null),this.tip&&(this.tip.remove(),this.tip=null)}static jQueryInterface(t){return this.each((function(){const e=cs.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}m(cs);const hs={...cs.Default,content:"",offset:[0,8],placement:"right",template:'',trigger:"click"},ds={...cs.DefaultType,content:"(null|string|element|function)"};class us extends cs{static get Default(){return hs}static get DefaultType(){return ds}static get NAME(){return"popover"}_isWithContent(){return this._getTitle()||this._getContent()}_getContentForTemplate(){return{".popover-header":this._getTitle(),".popover-body":this._getContent()}}_getContent(){return this._resolvePossibleFunction(this._config.content)}static jQueryInterface(t){return this.each((function(){const e=us.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}m(us);const fs=".bs.scrollspy",ps=`activate${fs}`,ms=`click${fs}`,gs=`load${fs}.data-api`,_s="active",bs="[href]",vs=".nav-link",ys=`${vs}, .nav-item > ${vs}, .list-group-item`,ws={offset:null,rootMargin:"0px 0px -25%",smoothScroll:!1,target:null,threshold:[.1,.5,1]},As={offset:"(number|null)",rootMargin:"string",smoothScroll:"boolean",target:"element",threshold:"array"};class Es extends W{constructor(t,e){super(t,e),this._targetLinks=new Map,this._observableSections=new Map,this._rootElement="visible"===getComputedStyle(this._element).overflowY?null:this._element,this._activeTarget=null,this._observer=null,this._previousScrollData={visibleEntryTop:0,parentScrollTop:0},this.refresh()}static get Default(){return ws}static get DefaultType(){return As}static get NAME(){return"scrollspy"}refresh(){this._initializeTargetsAndObservables(),this._maybeEnableSmoothScroll(),this._observer?this._observer.disconnect():this._observer=this._getNewObserver();for(const t of this._observableSections.values())this._observer.observe(t)}dispose(){this._observer.disconnect(),super.dispose()}_configAfterMerge(t){return t.target=r(t.target)||document.body,t.rootMargin=t.offset?`${t.offset}px 0px -30%`:t.rootMargin,"string"==typeof t.threshold&&(t.threshold=t.threshold.split(",").map((t=>Number.parseFloat(t)))),t}_maybeEnableSmoothScroll(){this._config.smoothScroll&&(N.off(this._config.target,ms),N.on(this._config.target,ms,bs,(t=>{const e=this._observableSections.get(t.target.hash);if(e){t.preventDefault();const i=this._rootElement||window,n=e.offsetTop-this._element.offsetTop;if(i.scrollTo)return void i.scrollTo({top:n,behavior:"smooth"});i.scrollTop=n}})))}_getNewObserver(){const t={root:this._rootElement,threshold:this._config.threshold,rootMargin:this._config.rootMargin};return new IntersectionObserver((t=>this._observerCallback(t)),t)}_observerCallback(t){const e=t=>this._targetLinks.get(`#${t.target.id}`),i=t=>{this._previousScrollData.visibleEntryTop=t.target.offsetTop,this._process(e(t))},n=(this._rootElement||document.documentElement).scrollTop,s=n>=this._previousScrollData.parentScrollTop;this._previousScrollData.parentScrollTop=n;for(const o of t){if(!o.isIntersecting){this._activeTarget=null,this._clearActiveClass(e(o));continue}const t=o.target.offsetTop>=this._previousScrollData.visibleEntryTop;if(s&&t){if(i(o),!n)return}else s||t||i(o)}}_initializeTargetsAndObservables(){this._targetLinks=new Map,this._observableSections=new Map;const t=z.find(bs,this._config.target);for(const e of t){if(!e.hash||l(e))continue;const t=z.findOne(decodeURI(e.hash),this._element);a(t)&&(this._targetLinks.set(decodeURI(e.hash),e),this._observableSections.set(e.hash,t))}}_process(t){this._activeTarget!==t&&(this._clearActiveClass(this._config.target),this._activeTarget=t,t.classList.add(_s),this._activateParents(t),N.trigger(this._element,ps,{relatedTarget:t}))}_activateParents(t){if(t.classList.contains("dropdown-item"))z.findOne(".dropdown-toggle",t.closest(".dropdown")).classList.add(_s);else for(const e of z.parents(t,".nav, .list-group"))for(const t of z.prev(e,ys))t.classList.add(_s)}_clearActiveClass(t){t.classList.remove(_s);const e=z.find(`${bs}.${_s}`,t);for(const t of e)t.classList.remove(_s)}static jQueryInterface(t){return this.each((function(){const e=Es.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}N.on(window,gs,(()=>{for(const t of z.find('[data-bs-spy="scroll"]'))Es.getOrCreateInstance(t)})),m(Es);const Ts=".bs.tab",Cs=`hide${Ts}`,Os=`hidden${Ts}`,xs=`show${Ts}`,ks=`shown${Ts}`,Ls=`click${Ts}`,Ss=`keydown${Ts}`,Ds=`load${Ts}`,$s="ArrowLeft",Is="ArrowRight",Ns="ArrowUp",Ps="ArrowDown",Ms="Home",js="End",Fs="active",Hs="fade",Ws="show",Bs=":not(.dropdown-toggle)",zs='[data-bs-toggle="tab"], [data-bs-toggle="pill"], [data-bs-toggle="list"]',Rs=`.nav-link${Bs}, .list-group-item${Bs}, [role="tab"]${Bs}, ${zs}`,qs=`.${Fs}[data-bs-toggle="tab"], .${Fs}[data-bs-toggle="pill"], .${Fs}[data-bs-toggle="list"]`;class Vs extends W{constructor(t){super(t),this._parent=this._element.closest('.list-group, .nav, [role="tablist"]'),this._parent&&(this._setInitialAttributes(this._parent,this._getChildren()),N.on(this._element,Ss,(t=>this._keydown(t))))}static get NAME(){return"tab"}show(){const t=this._element;if(this._elemIsActive(t))return;const e=this._getActiveElem(),i=e?N.trigger(e,Cs,{relatedTarget:t}):null;N.trigger(t,xs,{relatedTarget:e}).defaultPrevented||i&&i.defaultPrevented||(this._deactivate(e,t),this._activate(t,e))}_activate(t,e){t&&(t.classList.add(Fs),this._activate(z.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.removeAttribute("tabindex"),t.setAttribute("aria-selected",!0),this._toggleDropDown(t,!0),N.trigger(t,ks,{relatedTarget:e})):t.classList.add(Ws)}),t,t.classList.contains(Hs)))}_deactivate(t,e){t&&(t.classList.remove(Fs),t.blur(),this._deactivate(z.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.setAttribute("aria-selected",!1),t.setAttribute("tabindex","-1"),this._toggleDropDown(t,!1),N.trigger(t,Os,{relatedTarget:e})):t.classList.remove(Ws)}),t,t.classList.contains(Hs)))}_keydown(t){if(![$s,Is,Ns,Ps,Ms,js].includes(t.key))return;t.stopPropagation(),t.preventDefault();const e=this._getChildren().filter((t=>!l(t)));let i;if([Ms,js].includes(t.key))i=e[t.key===Ms?0:e.length-1];else{const n=[Is,Ps].includes(t.key);i=b(e,t.target,n,!0)}i&&(i.focus({preventScroll:!0}),Vs.getOrCreateInstance(i).show())}_getChildren(){return z.find(Rs,this._parent)}_getActiveElem(){return this._getChildren().find((t=>this._elemIsActive(t)))||null}_setInitialAttributes(t,e){this._setAttributeIfNotExists(t,"role","tablist");for(const t of e)this._setInitialAttributesOnChild(t)}_setInitialAttributesOnChild(t){t=this._getInnerElement(t);const e=this._elemIsActive(t),i=this._getOuterElement(t);t.setAttribute("aria-selected",e),i!==t&&this._setAttributeIfNotExists(i,"role","presentation"),e||t.setAttribute("tabindex","-1"),this._setAttributeIfNotExists(t,"role","tab"),this._setInitialAttributesOnTargetPanel(t)}_setInitialAttributesOnTargetPanel(t){const e=z.getElementFromSelector(t);e&&(this._setAttributeIfNotExists(e,"role","tabpanel"),t.id&&this._setAttributeIfNotExists(e,"aria-labelledby",`${t.id}`))}_toggleDropDown(t,e){const i=this._getOuterElement(t);if(!i.classList.contains("dropdown"))return;const n=(t,n)=>{const s=z.findOne(t,i);s&&s.classList.toggle(n,e)};n(".dropdown-toggle",Fs),n(".dropdown-menu",Ws),i.setAttribute("aria-expanded",e)}_setAttributeIfNotExists(t,e,i){t.hasAttribute(e)||t.setAttribute(e,i)}_elemIsActive(t){return t.classList.contains(Fs)}_getInnerElement(t){return t.matches(Rs)?t:z.findOne(Rs,t)}_getOuterElement(t){return t.closest(".nav-item, .list-group-item")||t}static jQueryInterface(t){return this.each((function(){const e=Vs.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}N.on(document,Ls,zs,(function(t){["A","AREA"].includes(this.tagName)&&t.preventDefault(),l(this)||Vs.getOrCreateInstance(this).show()})),N.on(window,Ds,(()=>{for(const t of z.find(qs))Vs.getOrCreateInstance(t)})),m(Vs);const Ks=".bs.toast",Qs=`mouseover${Ks}`,Xs=`mouseout${Ks}`,Ys=`focusin${Ks}`,Us=`focusout${Ks}`,Gs=`hide${Ks}`,Js=`hidden${Ks}`,Zs=`show${Ks}`,to=`shown${Ks}`,eo="hide",io="show",no="showing",so={animation:"boolean",autohide:"boolean",delay:"number"},oo={animation:!0,autohide:!0,delay:5e3};class ro extends W{constructor(t,e){super(t,e),this._timeout=null,this._hasMouseInteraction=!1,this._hasKeyboardInteraction=!1,this._setListeners()}static get Default(){return oo}static get DefaultType(){return so}static get NAME(){return"toast"}show(){N.trigger(this._element,Zs).defaultPrevented||(this._clearTimeout(),this._config.animation&&this._element.classList.add("fade"),this._element.classList.remove(eo),d(this._element),this._element.classList.add(io,no),this._queueCallback((()=>{this._element.classList.remove(no),N.trigger(this._element,to),this._maybeScheduleHide()}),this._element,this._config.animation))}hide(){this.isShown()&&(N.trigger(this._element,Gs).defaultPrevented||(this._element.classList.add(no),this._queueCallback((()=>{this._element.classList.add(eo),this._element.classList.remove(no,io),N.trigger(this._element,Js)}),this._element,this._config.animation)))}dispose(){this._clearTimeout(),this.isShown()&&this._element.classList.remove(io),super.dispose()}isShown(){return this._element.classList.contains(io)}_maybeScheduleHide(){this._config.autohide&&(this._hasMouseInteraction||this._hasKeyboardInteraction||(this._timeout=setTimeout((()=>{this.hide()}),this._config.delay)))}_onInteraction(t,e){switch(t.type){case"mouseover":case"mouseout":this._hasMouseInteraction=e;break;case"focusin":case"focusout":this._hasKeyboardInteraction=e}if(e)return void this._clearTimeout();const i=t.relatedTarget;this._element===i||this._element.contains(i)||this._maybeScheduleHide()}_setListeners(){N.on(this._element,Qs,(t=>this._onInteraction(t,!0))),N.on(this._element,Xs,(t=>this._onInteraction(t,!1))),N.on(this._element,Ys,(t=>this._onInteraction(t,!0))),N.on(this._element,Us,(t=>this._onInteraction(t,!1)))}_clearTimeout(){clearTimeout(this._timeout),this._timeout=null}static jQueryInterface(t){return this.each((function(){const e=ro.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}return R(ro),m(ro),{Alert:Q,Button:Y,Carousel:xt,Collapse:Bt,Dropdown:qi,Modal:On,Offcanvas:qn,Popover:us,ScrollSpy:Es,Tab:Vs,Toast:ro,Tooltip:cs}})); +//# sourceMappingURL=bootstrap.bundle.min.js.map \ No newline at end of file diff --git a/app/static/reascripts/FXPerm/.gitignore b/app/static/reascripts/FXPerm/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..31b688b99ba26b91c4b57041499bf153a418a2a9 --- /dev/null +++ b/app/static/reascripts/FXPerm/.gitignore @@ -0,0 +1 @@ +*.luac diff --git a/app/static/reascripts/ReaSpeech/.gitignore b/app/static/reascripts/ReaSpeech/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..31b688b99ba26b91c4b57041499bf153a418a2a9 --- /dev/null +++ b/app/static/reascripts/ReaSpeech/.gitignore @@ -0,0 +1 @@ +*.luac diff --git a/app/static/reascripts/ReaSpeech/icons.ttf b/app/static/reascripts/ReaSpeech/icons.ttf new file mode 100644 index 0000000000000000000000000000000000000000..d918491cccfb0be57947cf82cf92d32dbc6568b2 Binary files /dev/null and b/app/static/reascripts/ReaSpeech/icons.ttf differ diff --git a/app/templates/base.html b/app/templates/base.html new file mode 100644 index 0000000000000000000000000000000000000000..3fb35ad14c875d5448a38d476ba73f603f138a1b --- /dev/null +++ b/app/templates/base.html @@ -0,0 +1,33 @@ + + + + + + {% if title %}{{ title }} - {% endif %}ReaSpeech + + + + + + + + + {% include "nav.html" %} + + {% block content %}{% endblock %} + + + + diff --git a/app/templates/index.html b/app/templates/index.html new file mode 100644 index 0000000000000000000000000000000000000000..6adb34bc3e63236f24f3f84a2735288a96eb2e72 --- /dev/null +++ b/app/templates/index.html @@ -0,0 +1,48 @@ +{% extends "base.html" %} + +{% block content %} +
+
+

+ + ReaSpeech +

+ +

+ ReaSpeech is a REAPER extension for Automatic Speech Recognition. +

+ +

1. Install ReaImGui

+

+ ReaSpeech depends on the ReaImGui REAPER extension, which is distributed + via ReaPack through the default ReaTeam Extensions repository. +

+
    +
  • + + Install ReaPack if necessary +
  • +
  • In REAPER, go to Extensions > ReaPack > Browse packages...
  • +
  • Type "imgui" into the Filter box
  • +
  • Right-click on "ReaImGui: ReaScript binding for Dear ImGui"
  • +
  • Select "Install", and then click the "Apply" button on the bottom-right
  • +
+ +

2. Install ReaScript

+
    +
  • + + Download ReaScript +
  • +
  • Save the script to your REAPER Scripts folder (%APPDATA%\REAPER\Scripts on Windows)
  • +
  • Actions > Show action list... > New action... > Load ReaScript...
  • +
+ +

3. Run ReaSpeech

+
    +
  • Actions > Show action list...
  • +
  • Click "Script: ReaSpeech.lua" and then click "Run"
  • +
+
+
+{% endblock %} diff --git a/app/templates/nav.html b/app/templates/nav.html new file mode 100644 index 0000000000000000000000000000000000000000..c4abd9c26758d306dc484a1d065e1e8d7df41de8 --- /dev/null +++ b/app/templates/nav.html @@ -0,0 +1,15 @@ + diff --git a/app/templates/reascript.lua b/app/templates/reascript.lua new file mode 100644 index 0000000000000000000000000000000000000000..59f28ad6ba717deed68e2342fc02f959547654f0 --- /dev/null +++ b/app/templates/reascript.lua @@ -0,0 +1,33 @@ +-- ReaScript launcher for {{ name }} + +Script = { + name = "{{ name }}", + host = "{{ host }}", + lua = _VERSION:match('[%d.]+'), + timeout = 30000, +} + +function Script:load() + local curl = "curl" + local tempfile = os.tmpname() + if reaper.GetOS():find("Win") then + tempfile = os.getenv("TEMP") .. tempfile + else + curl = "/usr/bin/curl" + end + local command = curl + .. " -sSf http://" .. self.host .. "/static/reascripts/" .. self.name + .. "/" .. self.name .. "-" .. self.lua .. '.luac -o "' .. tempfile .. '"' + local result = reaper.ExecProcess(command, self.timeout) + local offset = result:find("\n") + local code = tonumber(result:sub(1, offset - 1)) + local output = result:sub(offset + 1, -1) + if code == 0 then + loadfile(tempfile)() + else + reaper.ShowConsoleMsg(output) + end + os.remove(tempfile) +end + +Script:load() diff --git a/app/util/__pycache__/audio.cpython-310.pyc b/app/util/__pycache__/audio.cpython-310.pyc new file mode 100644 index 0000000000000000000000000000000000000000..49d4b350981cd00d6245551c185d57673038bd8a Binary files /dev/null and b/app/util/__pycache__/audio.cpython-310.pyc differ diff --git a/app/util/apierror.py b/app/util/apierror.py new file mode 100644 index 0000000000000000000000000000000000000000..2fe2f9f38ee02e0a405abfad0e0e1c567c5c6a47 --- /dev/null +++ b/app/util/apierror.py @@ -0,0 +1,14 @@ +from fastapi.responses import JSONResponse + +class APIError(Exception): + def __init__(self, error): + self.error = error + + def to_response(self): + return error_response(self.error) + +def error_dict(error): + return {"error": str(error)} + +def error_response(error): + return JSONResponse(status_code=500, content=error_dict(error)) diff --git a/app/util/audio.py b/app/util/audio.py new file mode 100644 index 0000000000000000000000000000000000000000..0661fbcbef17456c3081a52c356973e562fc844a --- /dev/null +++ b/app/util/audio.py @@ -0,0 +1,47 @@ +import os +from typing import BinaryIO + +import ffmpeg +import numpy as np + +SAMPLE_RATE = 16000 + +FFMPEG_BIN = os.getenv("FFMPEG_BIN", "ffmpeg") + +def load_audio(file: BinaryIO, encode=True, sr: int = SAMPLE_RATE): + """ + Open an audio file object and read as mono waveform, resampling as necessary. + Modified from https://github.com/openai/whisper/blob/main/whisper/audio.py to accept a file object + Parameters + ---------- + file: BinaryIO + The audio file like object + encode: Boolean + If true, encode audio stream to WAV before sending to whisper + sr: int + The sample rate to resample the audio if necessary + Returns + ------- + A NumPy array containing the audio waveform, in float32 dtype. + """ + if encode: + try: + # This launches a subprocess to decode audio while down-mixing and resampling as necessary. + # Requires the ffmpeg CLI and `ffmpeg-python` package to be installed. + out, _ = ( + ffmpeg.input("pipe:", threads=0) + .output("-", format="s16le", acodec="pcm_s16le", ac=1, ar=sr) + .run(cmd=FFMPEG_BIN, capture_stdout=True, capture_stderr=True, input=file.read()) + ) + except ffmpeg.Error as e: + raise RuntimeError(f"Failed to load audio: {e.stderr.decode()}") from e + else: + out = file.read() + + try: + return np.frombuffer(out, np.int16).flatten().astype(np.float32) / 32768.0 + except Exception as e: + # TODO: Unsupported file formats can raise the following exception: + # ValueError: buffer size must be a multiple of element size + # This should be made more robust. + raise RuntimeError("Failed to load audio") from e diff --git a/app/webservice.py b/app/webservice.py new file mode 100644 index 0000000000000000000000000000000000000000..1e8923b2ad930635fe279c5bf31910368e7104ad --- /dev/null +++ b/app/webservice.py @@ -0,0 +1,183 @@ +from typing import Union, Annotated +import importlib.metadata +import logging +import os +import tempfile + +from celery.result import AsyncResult +from fastapi import FastAPI, File, Query, Request, UploadFile, applications +from fastapi.openapi.docs import get_swagger_ui_html +from fastapi.responses import HTMLResponse, JSONResponse, PlainTextResponse, RedirectResponse, StreamingResponse +from fastapi.staticfiles import StaticFiles +from fastapi.templating import Jinja2Templates +from whisper import tokenizer +import aiofiles + +from .util import apierror +from .worker import transcribe + +logging.basicConfig(format='[%(asctime)s] [%(name)s] [%(levelname)s] %(message)s', level=logging.INFO, force=True) +logger = logging.getLogger(__name__) + +ASR_ENGINE = os.getenv("ASR_ENGINE", "faster_whisper") + +ASR_OPTIONS = frozenset([ + "task", + "language", + "initial_prompt", + "encode", + "output", + "vad_filter", + "word_timestamps", + "model_name", +]) + +DEFAULT_MODEL_NAME = os.getenv("ASR_MODEL", "small") + +LANGUAGE_CODES = sorted(list(tokenizer.LANGUAGES.keys())) + +projectMetadata = importlib.metadata.metadata('reaspeech') +app = FastAPI( + # docs_url=None, + # redoc_url=None, + title=projectMetadata['Name'].title().replace('-', ' '), + description=projectMetadata['Summary'], + version=projectMetadata['Version'], + contact={ + "url": projectMetadata['Home-page'] + }, + swagger_ui_parameters={"defaultModelsExpandDepth": -1}, + license_info={ + "name": "MIT License", + "url": projectMetadata['License'] + } +) + +assets_path = os.getcwd() + "/swagger-ui-assets" +if os.path.exists(assets_path + "/swagger-ui.css") and os.path.exists(assets_path + "/swagger-ui-bundle.js"): + app.mount("/assets", StaticFiles(directory=assets_path), name="static") + + def swagger_monkey_patch(*args, **kwargs): + return get_swagger_ui_html( + *args, + **kwargs, + swagger_favicon_url="", + swagger_css_url="/assets/swagger-ui.css", + swagger_js_url="/assets/swagger-ui-bundle.js", + ) + + applications.get_swagger_ui_html = swagger_monkey_patch + +static_path = os.getcwd() + "/app/static" +app.mount("/static", StaticFiles(directory=static_path), name="static") + +templates_path = os.getcwd() + "/app/templates" +templates = Jinja2Templates(directory=templates_path) + +output_directory = os.environ.get("OUTPUT_DIRECTORY", os.getcwd() + "/app/output") +output_url_prefix = os.environ.get("OUTPUT_URL_PREFIX", "/output") +app.mount(output_url_prefix, StaticFiles(directory=output_directory), name="output") + +@app.exception_handler(apierror.APIError) +async def api_exception_handler(request: Request, exc: apierror.APIError): + return exc.to_response() + +@app.exception_handler(500) +async def internal_exception_handler(request: Request, exc: Exception): + return apierror.error_response(exc) + +@app.get("/", response_class=RedirectResponse, include_in_schema=False) +async def index(): + return "/reaspeech" + +@app.get("/reaspeech", response_class=HTMLResponse, include_in_schema=False) +async def reaspeech(request: Request): + return templates.TemplateResponse("index.html", {"request": request}) + +@app.get("/reascript", response_class=PlainTextResponse, include_in_schema=False) +async def reascript(request: Request, name: str, host: str): + return templates.TemplateResponse("reascript.lua", { + "request": request, + "name": name, + "host": host + }, + media_type='application/x-lua', + headers={ + 'Content-Disposition': f'attachment; filename="{name}.lua"' + } + ) + +@app.post("/asr", tags=["Endpoints"]) +async def asr( + task: Union[str, None] = Query(default="transcribe", enum=["transcribe", "translate"]), + language: Union[str, None] = Query(default=None, enum=LANGUAGE_CODES), + initial_prompt: Union[str, None] = Query(default=None), + audio_file: UploadFile = File(...), + encode: bool = Query(default=True, description="Encode audio first through ffmpeg"), + output: Union[str, None] = Query(default="txt", enum=["txt", "vtt", "srt", "tsv", "json"]), + vad_filter: Annotated[bool | None, Query( + description="Enable the voice activity detection (VAD) to filter out parts of the audio without speech", + include_in_schema=(True if ASR_ENGINE == "faster_whisper" else False) + )] = False, + word_timestamps: bool = Query(default=False, description="Word level timestamps"), + model_name: Union[str, None] = Query(default=None, description="Model name to use for transcription"), + use_async: bool = Query(default=False, description="Use asynchronous processing") +): + asr_options = {k: v for k, v in locals().items() if k in ASR_OPTIONS} + async_str = " (async)" if use_async else "" + logger.info(f"Transcribing{async_str} {audio_file.filename} with {asr_options}") + + with tempfile.NamedTemporaryFile(delete=False) as temp_file: + temp_file_path = temp_file.name + + async with aiofiles.open(temp_file_path, 'wb') as out_file: + while content := await audio_file.read(1024 * 1024): # Read in chunks of 1MB + await out_file.write(content) + + transcriber = transcribe.si(temp_file_path, audio_file.filename, asr_options) + + if use_async: + job = transcriber.apply_async() + return JSONResponse({"job_id": job.id}) + + else: + result = transcriber.apply().get() + + def reader(): + with open(result['output_path'], "r") as file: + yield from file + + filename = result['output_filename'] + return StreamingResponse( + reader(), + media_type="text/plain", + headers={ + 'Asr-Engine': ASR_ENGINE, + 'Content-Disposition': f'attachment; filename="{filename}"' + }) + +@app.get("/jobs/{job_id}", tags=["Endpoints"]) +async def job_status(job_id: str): + job = AsyncResult(job_id) + + result = { + "job_id": job_id, + "job_status": job.status, + "job_result": job.result + } + + if job.status == "FAILURE": + result["job_result"] = apierror.error_dict(result["job_result"]) + + return JSONResponse(result) + +@app.delete("/jobs/{job_id}", tags=["Endpoints"]) +async def revoke_job(job_id: str): + job = AsyncResult(job_id) + job.revoke(terminate=True) + + result = { + "job_id": job_id, + "job_status": job.status + } + return JSONResponse(result) diff --git a/app/worker.py b/app/worker.py new file mode 100644 index 0000000000000000000000000000000000000000..72cd7ba1cb0fde4f63db0da999a048c6422ca80d --- /dev/null +++ b/app/worker.py @@ -0,0 +1,129 @@ +import logging +import os + +from celery import Celery +from typing import Union, Callable +from whisper import tokenizer +import tqdm + +from .util.audio import load_audio + +logging.basicConfig(format='[%(asctime)s] [%(name)s] [%(levelname)s] %(message)s', level=logging.INFO, force=True) +logger = logging.getLogger(__name__) + +# monkeypatch tqdm to fool whisper's `transcribe` function +class _TQDM(tqdm.tqdm): + _tqdm = tqdm.tqdm + progress_function = None + + def __init__(self, *argv, total=0, unit="", **kwargs): + logger.debug(f"Creating TQDM with total={total}, unit={unit}") + self._total = total + self._unit = unit + self._progress = 0 + self.progress_function = _TQDM.progress_function or None + super().__init__(*argv, **kwargs) + + def set_progress_function(progress_function: Callable[[str, int, int], None]): + logger.debug(f"Setting progress function to {progress_function}") + _TQDM.progress_function = progress_function + + def update(self, progress): + logger.debug(f"Updating TQDM with progress={progress}") + self._progress += progress + if self.progress_function is not None: + self.progress_function(self._unit, self._total, self._progress) + else: + _TQDM._tqdm.update(self, progress) + +tqdm.tqdm = _TQDM + +ASR_ENGINE = os.getenv("ASR_ENGINE", "faster_whisper") +if ASR_ENGINE == "faster_whisper": + from .faster_whisper.core import load_model, transcribe as whisper_transcribe +else: + from .openai_whisper.core import load_model, transcribe as whisper_transcribe + +LANGUAGE_CODES = sorted(list(tokenizer.LANGUAGES.keys())) + +DEFAULT_MODEL_NAME = os.getenv("ASR_MODEL", "small") + +STATES = { + 'loading_model': 'LOADING_MODEL', + 'encoding': 'ENCODING', + 'transcribing': 'TRANSCRIBING', +} +celery = Celery(__name__) +celery.conf.broker_connection_retry_on_startup = True +celery.conf.broker_url = os.environ.get("CELERY_BROKER_URL", "redis://localhost:6379") +celery.conf.result_backend = os.environ.get("CELERY_RESULT_BACKEND", "redis://localhost:6379") +celery.conf.worker_hijack_root_logger = False +celery.conf.worker_redirect_stdouts_level = "DEBUG" + +@celery.task(name="transcribe", bind=True) +def transcribe( + self, + audio_file_path: str, + original_filename: str, + asr_options: dict, +): + logger.info(f"Transcribing {audio_file_path} with {asr_options}") + output_format = asr_options["output"] + + with open(audio_file_path, "rb") as audio_file: + _TQDM.set_progress_function(update_progress(self)) + + try: + model_name = asr_options.get("model_name") or DEFAULT_MODEL_NAME + logger.info(f"Loading model {model_name}") + self.update_state(state=STATES["loading_model"], meta={"progress": {"units": "models", "total": 1, "current": 0}}) + load_model(model_name) + + logger.info(f"Loading audio from {audio_file_path}") + self.update_state(state=STATES["encoding"], meta={"progress": {"units": "files", "total": 1, "current": 0}}) + audio_data = load_audio(audio_file, asr_options.get("encode", False)) + + logger.info(f"Transcribing audio") + self.update_state(state=STATES["transcribing"], meta={"progress": {"units": "files", "total": 1, "current": 0}}) + result = whisper_transcribe(audio_data, asr_options, output_format) + finally: + _TQDM.set_progress_function(None) + + logger.info(f"Transcription complete") + + os.remove(audio_file_path) + + filename = f"{original_filename.encode('latin-1', 'ignore').decode()}.{output_format}" + output_directory = get_output_path(self.request.id) + output_path = f"{output_directory}/{filename}" + + logger.info(f"Writing result to {output_path}") + + if not os.path.exists(output_directory): + os.makedirs(output_directory) + + with open(output_path, "w") as f: + f.write(result.read()) + + url_path = f"{get_output_url_path(transcribe.request.id)}/{filename}" + + return { + "output_filename": filename, + "output_path": output_path, + "url_path": url_path, + } + +def get_output_path(job_id: str): + return os.environ.get("OUTPUT_DIRECTORY", os.getcwd() + "/app/output") + "/" + job_id + +def get_output_url_path(job_id: str): + return os.environ.get("OUTPUT_URL_PREFIX", "/output") + "/" + job_id + +def update_progress(context): + def do_update(units, total, current): + logger.info(f"Updating progress with units={units}, total={total}, current={current}") + context.update_state( + state=STATES["transcribing"], + meta={"progress": {"units": units, "total": total, "current": current}} + ) + return do_update diff --git a/docs/assets/img/swagger-ui.png b/docs/assets/img/swagger-ui.png new file mode 100644 index 0000000000000000000000000000000000000000..1718c824e7d18f3c768e7613509cc787b72547a9 Binary files /dev/null and b/docs/assets/img/swagger-ui.png differ diff --git a/poetry.lock b/poetry.lock new file mode 100644 index 0000000000000000000000000000000000000000..c228e9b6416f58fe3dcc1e903b5934983beca47d --- /dev/null +++ b/poetry.lock @@ -0,0 +1,2112 @@ +# This file is automatically @generated by Poetry 1.6.1 and should not be changed by hand. + +[[package]] +name = "aiofiles" +version = "23.2.1" +description = "File support for asyncio." +optional = false +python-versions = ">=3.7" +files = [ + {file = "aiofiles-23.2.1-py3-none-any.whl", hash = "sha256:19297512c647d4b27a2cf7c34caa7e405c0d60b5560618a29a9fe027b18b0107"}, + {file = "aiofiles-23.2.1.tar.gz", hash = "sha256:84ec2218d8419404abcb9f0c02df3f34c6e0a68ed41072acfb1cef5cbc29051a"}, +] + +[[package]] +name = "amqp" +version = "5.2.0" +description = "Low-level AMQP client for Python (fork of amqplib)." +optional = false +python-versions = ">=3.6" +files = [ + {file = "amqp-5.2.0-py3-none-any.whl", hash = "sha256:827cb12fb0baa892aad844fd95258143bce4027fdac4fccddbc43330fd281637"}, + {file = "amqp-5.2.0.tar.gz", hash = "sha256:a1ecff425ad063ad42a486c902807d1482311481c8ad95a72694b2975e75f7fd"}, +] + +[package.dependencies] +vine = ">=5.0.0,<6.0.0" + +[[package]] +name = "anyio" +version = "4.3.0" +description = "High level compatibility layer for multiple asynchronous event loop implementations" +optional = false +python-versions = ">=3.8" +files = [ + {file = "anyio-4.3.0-py3-none-any.whl", hash = "sha256:048e05d0f6caeed70d731f3db756d35dcc1f35747c8c403364a8332c630441b8"}, + {file = "anyio-4.3.0.tar.gz", hash = "sha256:f75253795a87df48568485fd18cdd2a3fa5c4f7c5be8e5e36637733fce06fed6"}, +] + +[package.dependencies] +exceptiongroup = {version = ">=1.0.2", markers = "python_version < \"3.11\""} +idna = ">=2.8" +sniffio = ">=1.1" +typing-extensions = {version = ">=4.1", markers = "python_version < \"3.11\""} + +[package.extras] +doc = ["Sphinx (>=7)", "packaging", "sphinx-autodoc-typehints (>=1.2.0)", "sphinx-rtd-theme"] +test = ["anyio[trio]", "coverage[toml] (>=7)", "exceptiongroup (>=1.2.0)", "hypothesis (>=4.0)", "psutil (>=5.9)", "pytest (>=7.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (>=0.17)"] +trio = ["trio (>=0.23)"] + +[[package]] +name = "async-timeout" +version = "4.0.3" +description = "Timeout context manager for asyncio programs" +optional = false +python-versions = ">=3.7" +files = [ + {file = "async-timeout-4.0.3.tar.gz", hash = "sha256:4640d96be84d82d02ed59ea2b7105a0f7b33abe8703703cd0ab0bf87c427522f"}, + {file = "async_timeout-4.0.3-py3-none-any.whl", hash = "sha256:7405140ff1230c310e51dc27b3145b9092d659ce68ff733fb0cefe3ee42be028"}, +] + +[[package]] +name = "atomicwrites" +version = "1.4.1" +description = "Atomic file writes." +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +files = [ + {file = "atomicwrites-1.4.1.tar.gz", hash = "sha256:81b2c9071a49367a7f770170e5eec8cb66567cfbbc8c73d20ce5ca4a8d71cf11"}, +] + +[[package]] +name = "attrs" +version = "23.2.0" +description = "Classes Without Boilerplate" +optional = false +python-versions = ">=3.7" +files = [ + {file = "attrs-23.2.0-py3-none-any.whl", hash = "sha256:99b87a485a5820b23b879f04c2305b44b951b502fd64be915879d77a7e8fc6f1"}, + {file = "attrs-23.2.0.tar.gz", hash = "sha256:935dc3b529c262f6cf76e50877d35a4bd3c1de194fd41f47a2b7ae8f19971f30"}, +] + +[package.extras] +cov = ["attrs[tests]", "coverage[toml] (>=5.3)"] +dev = ["attrs[tests]", "pre-commit"] +docs = ["furo", "myst-parser", "sphinx", "sphinx-notfound-page", "sphinxcontrib-towncrier", "towncrier", "zope-interface"] +tests = ["attrs[tests-no-zope]", "zope-interface"] +tests-mypy = ["mypy (>=1.6)", "pytest-mypy-plugins"] +tests-no-zope = ["attrs[tests-mypy]", "cloudpickle", "hypothesis", "pympler", "pytest (>=4.3.0)", "pytest-xdist[psutil]"] + +[[package]] +name = "av" +version = "12.0.0" +description = "Pythonic bindings for FFmpeg's libraries." +optional = false +python-versions = ">=3.8" +files = [ + {file = "av-12.0.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:b9d0890553951f76c479a9f2bb952aebae902b1c7d52feea614d37e1cd728a44"}, + {file = "av-12.0.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:5d7f229a253c2e3fea9682c09c5ae179bd6d5d2da38d89eb7f29ef7bed10cb2f"}, + {file = "av-12.0.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:61b3555d143aacf02e0446f6030319403538eba4dc713c18dfa653a2a23e7f9c"}, + {file = "av-12.0.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:607e13b2c2b26159a37525d7b6f647a32ce78711fccff23d146d3e255ffa115f"}, + {file = "av-12.0.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:39f0b4cfb89f4f06b339c766f92648e798a96747d4163f2fa78660d1ab1f1b5e"}, + {file = "av-12.0.0-cp310-cp310-win_amd64.whl", hash = "sha256:41dcb8c269fa58a56edf3a3c814c32a0c69586827f132b4e395a951b0ce14fad"}, + {file = "av-12.0.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4fa78fbe0e4469226512380180063116105048c66cb12e18ab4b518466c57e6c"}, + {file = "av-12.0.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:60a869be1d6af916e65ea461cb93922f5db0698655ed7a7eae7c3ecd4af4debb"}, + {file = "av-12.0.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:df61811cc551c186f0a0e530d97b8b139453534d0f92c1790a923f666522ceda"}, + {file = "av-12.0.0-cp311-cp311-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:99cd2fc53091ebfb9a2fa9dd3580267f5bd1c040d0efd99fbc1a162576b271cb"}, + {file = "av-12.0.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7a6d4f1e261df48932128e6495772faa4cc23f5dd1512eec73daab82ad9f3240"}, + {file = "av-12.0.0-cp311-cp311-win_amd64.whl", hash = "sha256:6aec88e41a498b1e01e2dce5371557e20f9a51aae0c16decc5924ec0be2e22b6"}, + {file = "av-12.0.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:90eb8f2d548e96cbc6f78e89c911cdb15a3d80fd944f31111660ce45939cd037"}, + {file = "av-12.0.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:d7f3a02910e77d750dbd516256a16db15030e5371530ff5a5ae902dc03d9005d"}, + {file = "av-12.0.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e2477cc51526aa50575313d66e5e8ad7ab944588469be5e557b360ed572ae536"}, + {file = "av-12.0.0-cp312-cp312-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a2f47149d3ca6deb79f3e515b8bef50e27ebdb160813e6d67dba77278d2a7883"}, + {file = "av-12.0.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3306e4a3ce8b5bfcc3075793d4ed3a2df69179d8fba22cb944a6164dc235dfb6"}, + {file = "av-12.0.0-cp312-cp312-win_amd64.whl", hash = "sha256:dc1b742e7f6df1b499fb960bd6697d1dd8e7ada7484a041a8c20e70a87225f53"}, + {file = "av-12.0.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0183be6889e835e1b074b4037bfce4fd44671c606cf1c4ab92ea2f271b544aec"}, + {file = "av-12.0.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:57337f20b208292ec8d3b11e4d289d8688a43d728174850a81b865d3253fff2c"}, + {file = "av-12.0.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0ec915e8f6521545a38566eefc281042ee504ea3cee0618d8558e4920588b3b2"}, + {file = "av-12.0.0-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:33ad5c0a23c45b72bd6bd47f3b2c1adcd2935ee3d0b6178ed66bba62b964ff31"}, + {file = "av-12.0.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bfc3a652b12c93120514d56cf025da47442c5ba51530cdf7ba3660257dbb0de1"}, + {file = "av-12.0.0-cp38-cp38-win_amd64.whl", hash = "sha256:037f793dd1ef4a1f57f090191a7f803ad10ec82da0d04ea26bbe0b8a145fe927"}, + {file = "av-12.0.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:fc532376aa264722fae55063abd1871d17a563dc895978e142c8ecfcdeb3a2e8"}, + {file = "av-12.0.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:abf0c4bc40a0af8a30f4cd96f3be6f19fbce0f21222d7fcec148e085127153f7"}, + {file = "av-12.0.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:81cedd1c072fbebf606724c406b1a1b00adc711f1dfd2bc04c633ce39d8439d8"}, + {file = "av-12.0.0-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:02d60f48be9f15dcda37d50f3ce8d7249d9a455643d4322dd3449986bacfc628"}, + {file = "av-12.0.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5d2619e4c26d661eecfc404f7d739d8b35f0dcef353fabe61512e030254b7031"}, + {file = "av-12.0.0-cp39-cp39-win_amd64.whl", hash = "sha256:1892cc91c888d101777d5432d54e0554c11d1c3a2c65d02a2cae0a2256a8fbb9"}, + {file = "av-12.0.0-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:4819e3ef6c3a44ef6f75907229133a1ee7f688245b2cf49b6b8e969a81ca72c9"}, + {file = "av-12.0.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bb16bb314cf1503b0250fc46b2c455ee196584231101be0123f4f78638227b62"}, + {file = "av-12.0.0-pp310-pypy310_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f3e6a62bda9a1e144feeb59bbee046d7a2d98399634a30f57e4990197313c158"}, + {file = "av-12.0.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e08175ffbafa3a70c7b2f81083e160e34122a208cdf70f150b8f5d02c2de6965"}, + {file = "av-12.0.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:e1d255be317b7c1ebdc4dae98935b9f3869161112dc829c625e54f90d8bdd7ab"}, + {file = "av-12.0.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:17964b36e08435910aabd5b3f7dca12f99536902529767d276026bc08f94ced7"}, + {file = "av-12.0.0-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d2d5f78de29edee06ddcdd4c2b759914575492d6a0cd4de2ce31ee63a4953eff"}, + {file = "av-12.0.0-pp38-pypy38_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:309b32bc97158d0f0c19e273b8e17a855a86806b7194aebc23bd497326cff11f"}, + {file = "av-12.0.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c409c71bd9c7c2f8d018c822f36b1447cfa96eca158381a96f3319bb0ff6e79e"}, + {file = "av-12.0.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:08fc5eaef60a257d622998626e233bf3ff90d2f817f6695d6a27e0ffcfe9dcff"}, + {file = "av-12.0.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:746ab0eff8a7a21a6c6d16e6b6e61709527eba2ad1a524d92a01bb60d02a3df7"}, + {file = "av-12.0.0-pp39-pypy39_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:013b3ac3de3aa1c137af0cedafd364fd1c7524ab3e1cd53e04564fd1632ac04d"}, + {file = "av-12.0.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0fa55923527648f51ac005e44fe2797ebc67f53ad4850e0194d3753761ee33a2"}, + {file = "av-12.0.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:35d514f4dee0cf67e9e6b2a65fb4a28f98da88e71e8c7f7960bd04625d9fe965"}, + {file = "av-12.0.0.tar.gz", hash = "sha256:bcf21ebb722d4538b4099e5a78f730d78814dd70003511c185941dba5651b14d"}, +] + +[[package]] +name = "billiard" +version = "4.2.0" +description = "Python multiprocessing fork with improvements and bugfixes" +optional = false +python-versions = ">=3.7" +files = [ + {file = "billiard-4.2.0-py3-none-any.whl", hash = "sha256:07aa978b308f334ff8282bd4a746e681b3513db5c9a514cbdd810cbbdc19714d"}, + {file = "billiard-4.2.0.tar.gz", hash = "sha256:9a3c3184cb275aa17a732f93f65b20c525d3d9f253722d26a82194803ade5a2c"}, +] + +[[package]] +name = "celery" +version = "5.4.0" +description = "Distributed Task Queue." +optional = false +python-versions = ">=3.8" +files = [ + {file = "celery-5.4.0-py3-none-any.whl", hash = "sha256:369631eb580cf8c51a82721ec538684994f8277637edde2dfc0dacd73ed97f64"}, + {file = "celery-5.4.0.tar.gz", hash = "sha256:504a19140e8d3029d5acad88330c541d4c3f64c789d85f94756762d8bca7e706"}, +] + +[package.dependencies] +billiard = ">=4.2.0,<5.0" +click = ">=8.1.2,<9.0" +click-didyoumean = ">=0.3.0" +click-plugins = ">=1.1.1" +click-repl = ">=0.2.0" +kombu = ">=5.3.4,<6.0" +python-dateutil = ">=2.8.2" +tzdata = ">=2022.7" +vine = ">=5.1.0,<6.0" + +[package.extras] +arangodb = ["pyArango (>=2.0.2)"] +auth = ["cryptography (==42.0.5)"] +azureblockblob = ["azure-storage-blob (>=12.15.0)"] +brotli = ["brotli (>=1.0.0)", "brotlipy (>=0.7.0)"] +cassandra = ["cassandra-driver (>=3.25.0,<4)"] +consul = ["python-consul2 (==0.1.5)"] +cosmosdbsql = ["pydocumentdb (==2.3.5)"] +couchbase = ["couchbase (>=3.0.0)"] +couchdb = ["pycouchdb (==1.14.2)"] +django = ["Django (>=2.2.28)"] +dynamodb = ["boto3 (>=1.26.143)"] +elasticsearch = ["elastic-transport (<=8.13.0)", "elasticsearch (<=8.13.0)"] +eventlet = ["eventlet (>=0.32.0)"] +gcs = ["google-cloud-storage (>=2.10.0)"] +gevent = ["gevent (>=1.5.0)"] +librabbitmq = ["librabbitmq (>=2.0.0)"] +memcache = ["pylibmc (==1.6.3)"] +mongodb = ["pymongo[srv] (>=4.0.2)"] +msgpack = ["msgpack (==1.0.8)"] +pymemcache = ["python-memcached (>=1.61)"] +pyro = ["pyro4 (==4.82)"] +pytest = ["pytest-celery[all] (>=1.0.0)"] +redis = ["redis (>=4.5.2,!=4.5.5,<6.0.0)"] +s3 = ["boto3 (>=1.26.143)"] +slmq = ["softlayer-messaging (>=1.0.3)"] +solar = ["ephem (==4.1.5)"] +sqlalchemy = ["sqlalchemy (>=1.4.48,<2.1)"] +sqs = ["boto3 (>=1.26.143)", "kombu[sqs] (>=5.3.4)", "pycurl (>=7.43.0.5)", "urllib3 (>=1.26.16)"] +tblib = ["tblib (>=1.3.0)", "tblib (>=1.5.0)"] +yaml = ["PyYAML (>=3.10)"] +zookeeper = ["kazoo (>=1.3.1)"] +zstd = ["zstandard (==0.22.0)"] + +[[package]] +name = "certifi" +version = "2024.2.2" +description = "Python package for providing Mozilla's CA Bundle." +optional = false +python-versions = ">=3.6" +files = [ + {file = "certifi-2024.2.2-py3-none-any.whl", hash = "sha256:dc383c07b76109f368f6106eee2b593b04a011ea4d55f652c6ca24a754d1cdd1"}, + {file = "certifi-2024.2.2.tar.gz", hash = "sha256:0569859f95fc761b18b45ef421b1290a0f65f147e92a1e5eb3e635f9a5e4e66f"}, +] + +[[package]] +name = "charset-normalizer" +version = "3.3.2" +description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "charset-normalizer-3.3.2.tar.gz", hash = "sha256:f30c3cb33b24454a82faecaf01b19c18562b1e89558fb6c56de4d9118a032fd5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:25baf083bf6f6b341f4121c2f3c548875ee6f5339300e08be3f2b2ba1721cdd3"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:06435b539f889b1f6f4ac1758871aae42dc3a8c0e24ac9e60c2384973ad73027"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9063e24fdb1e498ab71cb7419e24622516c4a04476b17a2dab57e8baa30d6e03"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6897af51655e3691ff853668779c7bad41579facacf5fd7253b0133308cf000d"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1d3193f4a680c64b4b6a9115943538edb896edc190f0b222e73761716519268e"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cd70574b12bb8a4d2aaa0094515df2463cb429d8536cfb6c7ce983246983e5a6"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8465322196c8b4d7ab6d1e049e4c5cb460d0394da4a27d23cc242fbf0034b6b5"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a9a8e9031d613fd2009c182b69c7b2c1ef8239a0efb1df3f7c8da66d5dd3d537"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:beb58fe5cdb101e3a055192ac291b7a21e3b7ef4f67fa1d74e331a7f2124341c"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:e06ed3eb3218bc64786f7db41917d4e686cc4856944f53d5bdf83a6884432e12"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:2e81c7b9c8979ce92ed306c249d46894776a909505d8f5a4ba55b14206e3222f"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:572c3763a264ba47b3cf708a44ce965d98555f618ca42c926a9c1616d8f34269"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fd1abc0d89e30cc4e02e4064dc67fcc51bd941eb395c502aac3ec19fab46b519"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win32.whl", hash = "sha256:3d47fa203a7bd9c5b6cee4736ee84ca03b8ef23193c0d1ca99b5089f72645c73"}, + {file = "charset_normalizer-3.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:10955842570876604d404661fbccbc9c7e684caf432c09c715ec38fbae45ae09"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:802fe99cca7457642125a8a88a084cef28ff0cf9407060f7b93dca5aa25480db"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:573f6eac48f4769d667c4442081b1794f52919e7edada77495aaed9236d13a96"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:549a3a73da901d5bc3ce8d24e0600d1fa85524c10287f6004fbab87672bf3e1e"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f27273b60488abe721a075bcca6d7f3964f9f6f067c8c4c605743023d7d3944f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1ceae2f17a9c33cb48e3263960dc5fc8005351ee19db217e9b1bb15d28c02574"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:65f6f63034100ead094b8744b3b97965785388f308a64cf8d7c34f2f2e5be0c4"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:753f10e867343b4511128c6ed8c82f7bec3bd026875576dfd88483c5c73b2fd8"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4a78b2b446bd7c934f5dcedc588903fb2f5eec172f3d29e52a9096a43722adfc"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:e537484df0d8f426ce2afb2d0f8e1c3d0b114b83f8850e5f2fbea0e797bd82ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:eb6904c354526e758fda7167b33005998fb68c46fbc10e013ca97f21ca5c8887"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:deb6be0ac38ece9ba87dea880e438f25ca3eddfac8b002a2ec3d9183a454e8ae"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:4ab2fe47fae9e0f9dee8c04187ce5d09f48eabe611be8259444906793ab7cbce"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:80402cd6ee291dcb72644d6eac93785fe2c8b9cb30893c1af5b8fdd753b9d40f"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win32.whl", hash = "sha256:7cd13a2e3ddeed6913a65e66e94b51d80a041145a026c27e6bb76c31a853c6ab"}, + {file = "charset_normalizer-3.3.2-cp311-cp311-win_amd64.whl", hash = "sha256:663946639d296df6a2bb2aa51b60a2454ca1cb29835324c640dafb5ff2131a77"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0b2b64d2bb6d3fb9112bafa732def486049e63de9618b5843bcdd081d8144cd8"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:ddbb2551d7e0102e7252db79ba445cdab71b26640817ab1e3e3648dad515003b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:55086ee1064215781fff39a1af09518bc9255b50d6333f2e4c74ca09fac6a8f6"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f4a014bc36d3c57402e2977dada34f9c12300af536839dc38c0beab8878f38a"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a10af20b82360ab00827f916a6058451b723b4e65030c5a18577c8b2de5b3389"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:8d756e44e94489e49571086ef83b2bb8ce311e730092d2c34ca8f7d925cb20aa"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:90d558489962fd4918143277a773316e56c72da56ec7aa3dc3dbbe20fdfed15b"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6ac7ffc7ad6d040517be39eb591cac5ff87416c2537df6ba3cba3bae290c0fed"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:7ed9e526742851e8d5cc9e6cf41427dfc6068d4f5a3bb03659444b4cabf6bc26"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:8bdb58ff7ba23002a4c5808d608e4e6c687175724f54a5dade5fa8c67b604e4d"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:6b3251890fff30ee142c44144871185dbe13b11bab478a88887a639655be1068"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:b4a23f61ce87adf89be746c8a8974fe1c823c891d8f86eb218bb957c924bb143"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:efcb3f6676480691518c177e3b465bcddf57cea040302f9f4e6e191af91174d4"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win32.whl", hash = "sha256:d965bba47ddeec8cd560687584e88cf699fd28f192ceb452d1d7ee807c5597b7"}, + {file = "charset_normalizer-3.3.2-cp312-cp312-win_amd64.whl", hash = "sha256:96b02a3dc4381e5494fad39be677abcb5e6634bf7b4fa83a6dd3112607547001"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:95f2a5796329323b8f0512e09dbb7a1860c46a39da62ecb2324f116fa8fdc85c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c002b4ffc0be611f0d9da932eb0f704fe2602a9a949d1f738e4c34c75b0863d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a981a536974bbc7a512cf44ed14938cf01030a99e9b3a06dd59578882f06f985"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3287761bc4ee9e33561a7e058c72ac0938c4f57fe49a09eae428fd88aafe7bb6"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:42cb296636fcc8b0644486d15c12376cb9fa75443e00fb25de0b8602e64c1714"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0a55554a2fa0d408816b3b5cedf0045f4b8e1a6065aec45849de2d6f3f8e9786"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:c083af607d2515612056a31f0a8d9e0fcb5876b7bfc0abad3ecd275bc4ebc2d5"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:87d1351268731db79e0f8e745d92493ee2841c974128ef629dc518b937d9194c"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:bd8f7df7d12c2db9fab40bdd87a7c09b1530128315d047a086fa3ae3435cb3a8"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:c180f51afb394e165eafe4ac2936a14bee3eb10debc9d9e4db8958fe36afe711"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8c622a5fe39a48f78944a87d4fb8a53ee07344641b0562c540d840748571b811"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win32.whl", hash = "sha256:db364eca23f876da6f9e16c9da0df51aa4f104a972735574842618b8c6d999d4"}, + {file = "charset_normalizer-3.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:86216b5cee4b06df986d214f664305142d9c76df9b6512be2738aa72a2048f99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:6463effa3186ea09411d50efc7d85360b38d5f09b870c48e4600f63af490e56a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:6c4caeef8fa63d06bd437cd4bdcf3ffefe6738fb1b25951440d80dc7df8c03ac"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:37e55c8e51c236f95b033f6fb391d7d7970ba5fe7ff453dad675e88cf303377a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb69256e180cb6c8a894fee62b3afebae785babc1ee98b81cdf68bbca1987f33"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ae5f4161f18c61806f411a13b0310bea87f987c7d2ecdbdaad0e94eb2e404238"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b2b0a0c0517616b6869869f8c581d4eb2dd83a4d79e0ebcb7d373ef9956aeb0a"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:45485e01ff4d3630ec0d9617310448a8702f70e9c01906b0d0118bdf9d124cf2"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eb00ed941194665c332bf8e078baf037d6c35d7c4f3102ea2d4f16ca94a26dc8"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:2127566c664442652f024c837091890cb1942c30937add288223dc895793f898"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:a50aebfa173e157099939b17f18600f72f84eed3049e743b68ad15bd69b6bf99"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:4d0d1650369165a14e14e1e47b372cfcb31d6ab44e6e33cb2d4e57265290044d"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:923c0c831b7cfcb071580d3f46c4baf50f174be571576556269530f4bbd79d04"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:06a81e93cd441c56a9b65d8e1d043daeb97a3d0856d177d5c90ba85acb3db087"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win32.whl", hash = "sha256:6ef1d82a3af9d3eecdba2321dc1b3c238245d890843e040e41e470ffa64c3e25"}, + {file = "charset_normalizer-3.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:eb8821e09e916165e160797a6c17edda0679379a4be5c716c260e836e122f54b"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:c235ebd9baae02f1b77bcea61bce332cb4331dc3617d254df3323aa01ab47bd4"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5b4c145409bef602a690e7cfad0a15a55c13320ff7a3ad7ca59c13bb8ba4d45d"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:68d1f8a9e9e37c1223b656399be5d6b448dea850bed7d0f87a8311f1ff3dabb0"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:22afcb9f253dac0696b5a4be4a1c0f8762f8239e21b99680099abd9b2b1b2269"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:e27ad930a842b4c5eb8ac0016b0a54f5aebbe679340c26101df33424142c143c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1f79682fbe303db92bc2b1136016a38a42e835d932bab5b3b1bfcfbf0640e519"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b261ccdec7821281dade748d088bb6e9b69e6d15b30652b74cbbac25e280b796"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:122c7fa62b130ed55f8f285bfd56d5f4b4a5b503609d181f9ad85e55c89f4185"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:d0eccceffcb53201b5bfebb52600a5fb483a20b61da9dbc885f8b103cbe7598c"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f96df6923e21816da7e0ad3fd47dd8f94b2a5ce594e00677c0013018b813458"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:7f04c839ed0b6b98b1a7501a002144b76c18fb1c1850c8b98d458ac269e26ed2"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:34d1c8da1e78d2e001f363791c98a272bb734000fcef47a491c1e3b0505657a8"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:ff8fa367d09b717b2a17a052544193ad76cd49979c805768879cb63d9ca50561"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win32.whl", hash = "sha256:aed38f6e4fb3f5d6bf81bfa990a07806be9d83cf7bacef998ab1a9bd660a581f"}, + {file = "charset_normalizer-3.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:b01b88d45a6fcb69667cd6d2f7a9aeb4bf53760d7fc536bf679ec94fe9f3ff3d"}, + {file = "charset_normalizer-3.3.2-py3-none-any.whl", hash = "sha256:3e4d1f6587322d2788836a99c69062fbb091331ec940e02d12d179c1d53e25fc"}, +] + +[[package]] +name = "click" +version = "8.1.7" +description = "Composable command line interface toolkit" +optional = false +python-versions = ">=3.7" +files = [ + {file = "click-8.1.7-py3-none-any.whl", hash = "sha256:ae74fb96c20a0277a1d615f1e4d73c8414f5a98db8b799a7931d1582f3390c28"}, + {file = "click-8.1.7.tar.gz", hash = "sha256:ca9853ad459e787e2192211578cc907e7594e294c7ccc834310722b41b9ca6de"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + +[[package]] +name = "click-didyoumean" +version = "0.3.1" +description = "Enables git-like *did-you-mean* feature in click" +optional = false +python-versions = ">=3.6.2" +files = [ + {file = "click_didyoumean-0.3.1-py3-none-any.whl", hash = "sha256:5c4bb6007cfea5f2fd6583a2fb6701a22a41eb98957e63d0fac41c10e7c3117c"}, + {file = "click_didyoumean-0.3.1.tar.gz", hash = "sha256:4f82fdff0dbe64ef8ab2279bd6aa3f6a99c3b28c05aa09cbfc07c9d7fbb5a463"}, +] + +[package.dependencies] +click = ">=7" + +[[package]] +name = "click-plugins" +version = "1.1.1" +description = "An extension module for click to enable registering CLI commands via setuptools entry-points." +optional = false +python-versions = "*" +files = [ + {file = "click-plugins-1.1.1.tar.gz", hash = "sha256:46ab999744a9d831159c3411bb0c79346d94a444df9a3a3742e9ed63645f264b"}, + {file = "click_plugins-1.1.1-py2.py3-none-any.whl", hash = "sha256:5d262006d3222f5057fd81e1623d4443e41dcda5dc815c06b442aa3c02889fc8"}, +] + +[package.dependencies] +click = ">=4.0" + +[package.extras] +dev = ["coveralls", "pytest (>=3.6)", "pytest-cov", "wheel"] + +[[package]] +name = "click-repl" +version = "0.3.0" +description = "REPL plugin for Click" +optional = false +python-versions = ">=3.6" +files = [ + {file = "click-repl-0.3.0.tar.gz", hash = "sha256:17849c23dba3d667247dc4defe1757fff98694e90fe37474f3feebb69ced26a9"}, + {file = "click_repl-0.3.0-py3-none-any.whl", hash = "sha256:fb7e06deb8da8de86180a33a9da97ac316751c094c6899382da7feeeeb51b812"}, +] + +[package.dependencies] +click = ">=7.0" +prompt-toolkit = ">=3.0.36" + +[package.extras] +testing = ["pytest (>=7.2.1)", "pytest-cov (>=4.0.0)", "tox (>=4.4.3)"] + +[[package]] +name = "cmake" +version = "3.29.2" +description = "CMake is an open-source, cross-platform family of tools designed to build, test and package software" +optional = false +python-versions = ">=3.7" +files = [ + {file = "cmake-3.29.2-py3-none-macosx_10_10_universal2.macosx_10_10_x86_64.macosx_11_0_arm64.macosx_11_0_universal2.whl", hash = "sha256:1d40c5451d6467b20a0a6015a5a6b6dc86f61b83f71f935740485b259100a34e"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:ed3108e646cd65a4e23fa1cbe8123569a29334a3f2a8ce214d871406b161bedb"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:40aafe612b03a9fa140cca4024ba60b74cd92372f3f349d8062cba1f021e5001"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:027eebe9bb74c31759581a543f27bc1828fc76e6fc45b2b48b51f27847904669"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f1f087985fc2460476b0901716fbddb2fd69b7fe7bf1350e1ab5dc508d22600e"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:df2c63ce6d504aa4c91a42fd22d3887065ab029569691deb56ec19d0decd0ae9"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:6ea5ce007893d7d1363e13433dde1c0c7c344372213a90ff3c56e896a335301d"}, + {file = "cmake-3.29.2-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f9e941e73202cfa667ee488d1d88b8a758b516dcfa2a2728e73dbdcbfbdebf57"}, + {file = "cmake-3.29.2-py3-none-musllinux_1_1_aarch64.whl", hash = "sha256:37222e23485338c72b7ea51f865d8c6847d519f7e2222922fb70b4896ca6e897"}, + {file = "cmake-3.29.2-py3-none-musllinux_1_1_i686.whl", hash = "sha256:eeed08932c748647488280dc97ac00bcfeae5d760451105200cfe66c52ce6468"}, + {file = "cmake-3.29.2-py3-none-musllinux_1_1_ppc64le.whl", hash = "sha256:db7a05df020ba67bacd3070dd1645c76ca96fabd06d6aaa63288fd845706e47a"}, + {file = "cmake-3.29.2-py3-none-musllinux_1_1_s390x.whl", hash = "sha256:83b35de822ddabaaa184a7d8f9827381350c42d627689c775b214347f57c9e41"}, + {file = "cmake-3.29.2-py3-none-musllinux_1_1_x86_64.whl", hash = "sha256:cc0e36752e581430a93e58a268e515bb4ec1373b9e9911571f2cac1d2a6b5bec"}, + {file = "cmake-3.29.2-py3-none-win32.whl", hash = "sha256:a941e26fba81cf74832c8a0e17e007452e05b6ad4941b3d2d18c75faa4a677d8"}, + {file = "cmake-3.29.2-py3-none-win_amd64.whl", hash = "sha256:23336c8ca01205d18d92ed8de6c54e570c352a58e378b7f9adc02ef00f433960"}, + {file = "cmake-3.29.2-py3-none-win_arm64.whl", hash = "sha256:e722a949f7c91084dba61f8f17a9854787182ab711ed0b84b1507b24a8e12e25"}, + {file = "cmake-3.29.2.tar.gz", hash = "sha256:6a4c1185cb2eca7263190a5754d0c9edf738d9e50bff464f78f48d0c05318e7c"}, +] + +[package.extras] +test = ["coverage (>=4.2)", "pytest (>=3.0.3)", "pytest-cov (>=2.4.0)"] + +[[package]] +name = "colorama" +version = "0.4.6" +description = "Cross-platform colored terminal text." +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7" +files = [ + {file = "colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6"}, + {file = "colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44"}, +] + +[[package]] +name = "coloredlogs" +version = "15.0.1" +description = "Colored terminal output for Python's logging module" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +files = [ + {file = "coloredlogs-15.0.1-py2.py3-none-any.whl", hash = "sha256:612ee75c546f53e92e70049c9dbfcc18c935a2b9a53b66085ce9ef6a6e5c0934"}, + {file = "coloredlogs-15.0.1.tar.gz", hash = "sha256:7c991aa71a4577af2f82600d8f8f3a89f936baeaf9b50a9c197da014e5bf16b0"}, +] + +[package.dependencies] +humanfriendly = ">=9.1" + +[package.extras] +cron = ["capturer (>=2.4)"] + +[[package]] +name = "ctranslate2" +version = "4.2.1" +description = "Fast inference engine for Transformer models" +optional = false +python-versions = ">=3.8" +files = [ + {file = "ctranslate2-4.2.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:c8a98623342da8448043a32e3b20b7b6007d4893e7026a6f7ce6e5bed3ef07f5"}, + {file = "ctranslate2-4.2.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:550b8014f51b0290a67a37017fcce743c59e2be5a3b7a209db4b1c9e4fb5236b"}, + {file = "ctranslate2-4.2.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:48dd15c8f1d55d58fed9c3d8f7d4e995b17e7dbdfc3c9b8cc9e79eee40be0ab8"}, + {file = "ctranslate2-4.2.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1332436c4d5ebef57a5b6bec481bf89aa062aabc4a721e4b30a10016e044a28a"}, + {file = "ctranslate2-4.2.1-cp310-cp310-win_amd64.whl", hash = "sha256:f40e0552f096d2243d0286623ffed34f9fa7275cf0298ee44e1190078e2c8d7d"}, + {file = "ctranslate2-4.2.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:79112191709c802b9b66fdd1c4c4dc63d994e2facf9d1ab2561b9e291f0845ff"}, + {file = "ctranslate2-4.2.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:463391dcf8c1327c196af813d953399ca60274ef3cf38d9776b8f8b0d2138d23"}, + {file = "ctranslate2-4.2.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:549e4be62612271fcfe2cec9204b048777954dabb77af041ca0dc9cb6acfdbd8"}, + {file = "ctranslate2-4.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a364ac0ddbe1abb35136261632e4e8cc6a820afbfdde3d2f4a40e5e7ee1e3ebc"}, + {file = "ctranslate2-4.2.1-cp311-cp311-win_amd64.whl", hash = "sha256:a2332da680c1f6904d7c67ff539ed5e2d80532b8e531e24ca68ef4fdf5ba729e"}, + {file = "ctranslate2-4.2.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:75207688fc862457b578b5cf917e606283ab720822a3ba6d4ba405ddd53f33c3"}, + {file = "ctranslate2-4.2.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:41b5f401b7765c7d19d2b35e4d282b15f572d57cebdca5bd30a4deda076a05d2"}, + {file = "ctranslate2-4.2.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:174be98d48869053ac31d9c91c910f3860afff2a84139b6352bd9a11a7e99a5f"}, + {file = "ctranslate2-4.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1681fb06fa5714e3d440b4a9fce8f1fb71cfadfcc912e0346eaea4385269b347"}, + {file = "ctranslate2-4.2.1-cp312-cp312-win_amd64.whl", hash = "sha256:c5b604b64138c4cf49a9fac1ebc7023e3ea2fe95fe7b42a9c7076f789fd5ce65"}, + {file = "ctranslate2-4.2.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:bb9069476266a82ffaacff681fd8a2dd249bac18064f766910b55cfd025efab8"}, + {file = "ctranslate2-4.2.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:c6aff28115b00ec839c4f106feb2fb8de5cd75314a277fa72e1953efa3e251b2"}, + {file = "ctranslate2-4.2.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:998b1fc6da60863b4e9c48502cae5956520eeebeacef3838a443c538e846089c"}, + {file = "ctranslate2-4.2.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:47635ca9b58cb0946eefb187ad2963fea3f3c44093f8059ec37cf4d9ab5a9810"}, + {file = "ctranslate2-4.2.1-cp38-cp38-win_amd64.whl", hash = "sha256:ece6c255bad7f76ad92e6a2345685c85f8c0547a06861ec71672a6a82c519944"}, + {file = "ctranslate2-4.2.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9f4edde5d86466d85b6d6626f90cc0e40f1304c9ad7471e8363e06cf2c96ec2e"}, + {file = "ctranslate2-4.2.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:d478b00a9b2fc78aae815512dfea76d6e15c71f41c31e95b3d5129a1387cf1f5"}, + {file = "ctranslate2-4.2.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4f65820714a81df2ad707e38ebbffdaa4c2b76bef80063684921ab4378fdf659"}, + {file = "ctranslate2-4.2.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a1d6726c7ed4e528a18c48487a5f5f3dcc8d66d551069f319c044de0ae8c8015"}, + {file = "ctranslate2-4.2.1-cp39-cp39-win_amd64.whl", hash = "sha256:f889e37cb89e663be300a23e1fca4e8c8d3191615ec036c89e49b504b621ba90"}, +] + +[package.dependencies] +numpy = "*" +pyyaml = ">=5.3,<7" +setuptools = "*" + +[[package]] +name = "exceptiongroup" +version = "1.2.1" +description = "Backport of PEP 654 (exception groups)" +optional = false +python-versions = ">=3.7" +files = [ + {file = "exceptiongroup-1.2.1-py3-none-any.whl", hash = "sha256:5258b9ed329c5bbdd31a309f53cbfb0b155341807f6ff7606a1e801a891b29ad"}, + {file = "exceptiongroup-1.2.1.tar.gz", hash = "sha256:a4785e48b045528f5bfe627b6ad554ff32def154f42372786903b7abcfe1aa16"}, +] + +[package.extras] +test = ["pytest (>=6)"] + +[[package]] +name = "fastapi" +version = "0.95.2" +description = "FastAPI framework, high performance, easy to learn, fast to code, ready for production" +optional = false +python-versions = ">=3.7" +files = [ + {file = "fastapi-0.95.2-py3-none-any.whl", hash = "sha256:d374dbc4ef2ad9b803899bd3360d34c534adc574546e25314ab72c0c4411749f"}, + {file = "fastapi-0.95.2.tar.gz", hash = "sha256:4d9d3e8c71c73f11874bcf5e33626258d143252e329a01002f767306c64fb982"}, +] + +[package.dependencies] +pydantic = ">=1.6.2,<1.7 || >1.7,<1.7.1 || >1.7.1,<1.7.2 || >1.7.2,<1.7.3 || >1.7.3,<1.8 || >1.8,<1.8.1 || >1.8.1,<2.0.0" +starlette = ">=0.27.0,<0.28.0" + +[package.extras] +all = ["email-validator (>=1.1.1)", "httpx (>=0.23.0)", "itsdangerous (>=1.1.0)", "jinja2 (>=2.11.2)", "orjson (>=3.2.1)", "python-multipart (>=0.0.5)", "pyyaml (>=5.3.1)", "ujson (>=4.0.1,!=4.0.2,!=4.1.0,!=4.2.0,!=4.3.0,!=5.0.0,!=5.1.0)", "uvicorn[standard] (>=0.12.0)"] +dev = ["pre-commit (>=2.17.0,<3.0.0)", "ruff (==0.0.138)", "uvicorn[standard] (>=0.12.0,<0.21.0)"] +doc = ["mdx-include (>=1.4.1,<2.0.0)", "mkdocs (>=1.1.2,<2.0.0)", "mkdocs-markdownextradata-plugin (>=0.1.7,<0.3.0)", "mkdocs-material (>=8.1.4,<9.0.0)", "pyyaml (>=5.3.1,<7.0.0)", "typer-cli (>=0.0.13,<0.0.14)", "typer[all] (>=0.6.1,<0.8.0)"] +test = ["anyio[trio] (>=3.2.1,<4.0.0)", "black (==23.1.0)", "coverage[toml] (>=6.5.0,<8.0)", "databases[sqlite] (>=0.3.2,<0.7.0)", "email-validator (>=1.1.1,<2.0.0)", "flask (>=1.1.2,<3.0.0)", "httpx (>=0.23.0,<0.24.0)", "isort (>=5.0.6,<6.0.0)", "mypy (==0.982)", "orjson (>=3.2.1,<4.0.0)", "passlib[bcrypt] (>=1.7.2,<2.0.0)", "peewee (>=3.13.3,<4.0.0)", "pytest (>=7.1.3,<8.0.0)", "python-jose[cryptography] (>=3.3.0,<4.0.0)", "python-multipart (>=0.0.5,<0.0.7)", "pyyaml (>=5.3.1,<7.0.0)", "ruff (==0.0.138)", "sqlalchemy (>=1.3.18,<1.4.43)", "types-orjson (==3.6.2)", "types-ujson (==5.7.0.1)", "ujson (>=4.0.1,!=4.0.2,!=4.1.0,!=4.2.0,!=4.3.0,!=5.0.0,!=5.1.0,<6.0.0)"] + +[[package]] +name = "faster-whisper" +version = "1.0.2" +description = "Faster Whisper transcription with CTranslate2" +optional = false +python-versions = ">=3.8" +files = [ + {file = "faster-whisper-1.0.2.tar.gz", hash = "sha256:54d9fc698f7c665e00a0d5ed65d6e975b72a8862b8214f20a22e79b115c41511"}, + {file = "faster_whisper-1.0.2-py3-none-any.whl", hash = "sha256:d968c289222e766a49ed97eecec24e934bdef405183f57d6d434a364bb3569c1"}, +] + +[package.dependencies] +av = ">=11.0,<13" +ctranslate2 = ">=4.0,<5" +huggingface-hub = ">=0.13" +onnxruntime = ">=1.14,<2" +tokenizers = ">=0.13,<1" + +[package.extras] +conversion = ["transformers[torch] (>=4.23)"] +dev = ["black (==23.*)", "flake8 (==6.*)", "isort (==5.*)", "pytest (==7.*)"] + +[[package]] +name = "ffmpeg-python" +version = "0.2.0" +description = "Python bindings for FFmpeg - with complex filtering support" +optional = false +python-versions = "*" +files = [ + {file = "ffmpeg-python-0.2.0.tar.gz", hash = "sha256:65225db34627c578ef0e11c8b1eb528bb35e024752f6f10b78c011f6f64c4127"}, + {file = "ffmpeg_python-0.2.0-py3-none-any.whl", hash = "sha256:ac441a0404e053f8b6a1113a77c0f452f1cfc62f6344a769475ffdc0f56c23c5"}, +] + +[package.dependencies] +future = "*" + +[package.extras] +dev = ["Sphinx (==2.1.0)", "future (==0.17.1)", "numpy (==1.16.4)", "pytest (==4.6.1)", "pytest-mock (==1.10.4)", "tox (==3.12.1)"] + +[[package]] +name = "filelock" +version = "3.13.4" +description = "A platform independent file lock." +optional = false +python-versions = ">=3.8" +files = [ + {file = "filelock-3.13.4-py3-none-any.whl", hash = "sha256:404e5e9253aa60ad457cae1be07c0f0ca90a63931200a47d9b6a6af84fd7b45f"}, + {file = "filelock-3.13.4.tar.gz", hash = "sha256:d13f466618bfde72bd2c18255e269f72542c6e70e7bac83a0232d6b1cc5c8cf4"}, +] + +[package.extras] +docs = ["furo (>=2023.9.10)", "sphinx (>=7.2.6)", "sphinx-autodoc-typehints (>=1.25.2)"] +testing = ["covdefaults (>=2.3)", "coverage (>=7.3.2)", "diff-cover (>=8.0.1)", "pytest (>=7.4.3)", "pytest-cov (>=4.1)", "pytest-mock (>=3.12)", "pytest-timeout (>=2.2)"] +typing = ["typing-extensions (>=4.8)"] + +[[package]] +name = "flatbuffers" +version = "24.3.25" +description = "The FlatBuffers serialization format for Python" +optional = false +python-versions = "*" +files = [ + {file = "flatbuffers-24.3.25-py2.py3-none-any.whl", hash = "sha256:8dbdec58f935f3765e4f7f3cf635ac3a77f83568138d6a2311f524ec96364812"}, + {file = "flatbuffers-24.3.25.tar.gz", hash = "sha256:de2ec5b203f21441716617f38443e0a8ebf3d25bf0d9c0bb0ce68fa00ad546a4"}, +] + +[[package]] +name = "fsspec" +version = "2024.3.1" +description = "File-system specification" +optional = false +python-versions = ">=3.8" +files = [ + {file = "fsspec-2024.3.1-py3-none-any.whl", hash = "sha256:918d18d41bf73f0e2b261824baeb1b124bcf771767e3a26425cd7dec3332f512"}, + {file = "fsspec-2024.3.1.tar.gz", hash = "sha256:f39780e282d7d117ffb42bb96992f8a90795e4d0fb0f661a70ca39fe9c43ded9"}, +] + +[package.extras] +abfs = ["adlfs"] +adl = ["adlfs"] +arrow = ["pyarrow (>=1)"] +dask = ["dask", "distributed"] +devel = ["pytest", "pytest-cov"] +dropbox = ["dropbox", "dropboxdrivefs", "requests"] +full = ["adlfs", "aiohttp (!=4.0.0a0,!=4.0.0a1)", "dask", "distributed", "dropbox", "dropboxdrivefs", "fusepy", "gcsfs", "libarchive-c", "ocifs", "panel", "paramiko", "pyarrow (>=1)", "pygit2", "requests", "s3fs", "smbprotocol", "tqdm"] +fuse = ["fusepy"] +gcs = ["gcsfs"] +git = ["pygit2"] +github = ["requests"] +gs = ["gcsfs"] +gui = ["panel"] +hdfs = ["pyarrow (>=1)"] +http = ["aiohttp (!=4.0.0a0,!=4.0.0a1)"] +libarchive = ["libarchive-c"] +oci = ["ocifs"] +s3 = ["s3fs"] +sftp = ["paramiko"] +smb = ["smbprotocol"] +ssh = ["paramiko"] +tqdm = ["tqdm"] + +[[package]] +name = "future" +version = "1.0.0" +description = "Clean single-source support for Python 3 and 2" +optional = false +python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" +files = [ + {file = "future-1.0.0-py3-none-any.whl", hash = "sha256:929292d34f5872e70396626ef385ec22355a1fae8ad29e1a734c3e43f9fbc216"}, + {file = "future-1.0.0.tar.gz", hash = "sha256:bd2968309307861edae1458a4f8a4f3598c03be43b97521076aebf5d94c07b05"}, +] + +[[package]] +name = "gunicorn" +version = "20.1.0" +description = "WSGI HTTP Server for UNIX" +optional = false +python-versions = ">=3.5" +files = [ + {file = "gunicorn-20.1.0-py3-none-any.whl", hash = "sha256:9dcc4547dbb1cb284accfb15ab5667a0e5d1881cc443e0677b4882a4067a807e"}, + {file = "gunicorn-20.1.0.tar.gz", hash = "sha256:e0a968b5ba15f8a328fdfd7ab1fcb5af4470c28aaf7e55df02a99bc13138e6e8"}, +] + +[package.dependencies] +setuptools = ">=3.0" + +[package.extras] +eventlet = ["eventlet (>=0.24.1)"] +gevent = ["gevent (>=1.4.0)"] +setproctitle = ["setproctitle"] +tornado = ["tornado (>=0.2)"] + +[[package]] +name = "h11" +version = "0.14.0" +description = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" +optional = false +python-versions = ">=3.7" +files = [ + {file = "h11-0.14.0-py3-none-any.whl", hash = "sha256:e3fe4ac4b851c468cc8363d500db52c2ead036020723024a109d37346efaa761"}, + {file = "h11-0.14.0.tar.gz", hash = "sha256:8f19fbbe99e72420ff35c00b27a34cb9937e902a8b810e2c88300c6f0a3b699d"}, +] + +[[package]] +name = "httptools" +version = "0.6.1" +description = "A collection of framework independent HTTP protocol utils." +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "httptools-0.6.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:d2f6c3c4cb1948d912538217838f6e9960bc4a521d7f9b323b3da579cd14532f"}, + {file = "httptools-0.6.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:00d5d4b68a717765b1fabfd9ca755bd12bf44105eeb806c03d1962acd9b8e563"}, + {file = "httptools-0.6.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:639dc4f381a870c9ec860ce5c45921db50205a37cc3334e756269736ff0aac58"}, + {file = "httptools-0.6.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e57997ac7fb7ee43140cc03664de5f268813a481dff6245e0075925adc6aa185"}, + {file = "httptools-0.6.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:0ac5a0ae3d9f4fe004318d64b8a854edd85ab76cffbf7ef5e32920faef62f142"}, + {file = "httptools-0.6.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:3f30d3ce413088a98b9db71c60a6ada2001a08945cb42dd65a9a9fe228627658"}, + {file = "httptools-0.6.1-cp310-cp310-win_amd64.whl", hash = "sha256:1ed99a373e327f0107cb513b61820102ee4f3675656a37a50083eda05dc9541b"}, + {file = "httptools-0.6.1-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:7a7ea483c1a4485c71cb5f38be9db078f8b0e8b4c4dc0210f531cdd2ddac1ef1"}, + {file = "httptools-0.6.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:85ed077c995e942b6f1b07583e4eb0a8d324d418954fc6af913d36db7c05a5a0"}, + {file = "httptools-0.6.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b0bb634338334385351a1600a73e558ce619af390c2b38386206ac6a27fecfc"}, + {file = "httptools-0.6.1-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7d9ceb2c957320def533671fc9c715a80c47025139c8d1f3797477decbc6edd2"}, + {file = "httptools-0.6.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:4f0f8271c0a4db459f9dc807acd0eadd4839934a4b9b892f6f160e94da309837"}, + {file = "httptools-0.6.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:6a4f5ccead6d18ec072ac0b84420e95d27c1cdf5c9f1bc8fbd8daf86bd94f43d"}, + {file = "httptools-0.6.1-cp311-cp311-win_amd64.whl", hash = "sha256:5cceac09f164bcba55c0500a18fe3c47df29b62353198e4f37bbcc5d591172c3"}, + {file = "httptools-0.6.1-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:75c8022dca7935cba14741a42744eee13ba05db00b27a4b940f0d646bd4d56d0"}, + {file = "httptools-0.6.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:48ed8129cd9a0d62cf4d1575fcf90fb37e3ff7d5654d3a5814eb3d55f36478c2"}, + {file = "httptools-0.6.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6f58e335a1402fb5a650e271e8c2d03cfa7cea46ae124649346d17bd30d59c90"}, + {file = "httptools-0.6.1-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:93ad80d7176aa5788902f207a4e79885f0576134695dfb0fefc15b7a4648d503"}, + {file = "httptools-0.6.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:9bb68d3a085c2174c2477eb3ffe84ae9fb4fde8792edb7bcd09a1d8467e30a84"}, + {file = "httptools-0.6.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:b512aa728bc02354e5ac086ce76c3ce635b62f5fbc32ab7082b5e582d27867bb"}, + {file = "httptools-0.6.1-cp312-cp312-win_amd64.whl", hash = "sha256:97662ce7fb196c785344d00d638fc9ad69e18ee4bfb4000b35a52efe5adcc949"}, + {file = "httptools-0.6.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:8e216a038d2d52ea13fdd9b9c9c7459fb80d78302b257828285eca1c773b99b3"}, + {file = "httptools-0.6.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:3e802e0b2378ade99cd666b5bffb8b2a7cc8f3d28988685dc300469ea8dd86cb"}, + {file = "httptools-0.6.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4bd3e488b447046e386a30f07af05f9b38d3d368d1f7b4d8f7e10af85393db97"}, + {file = "httptools-0.6.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fe467eb086d80217b7584e61313ebadc8d187a4d95bb62031b7bab4b205c3ba3"}, + {file = "httptools-0.6.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:3c3b214ce057c54675b00108ac42bacf2ab8f85c58e3f324a4e963bbc46424f4"}, + {file = "httptools-0.6.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8ae5b97f690badd2ca27cbf668494ee1b6d34cf1c464271ef7bfa9ca6b83ffaf"}, + {file = "httptools-0.6.1-cp38-cp38-win_amd64.whl", hash = "sha256:405784577ba6540fa7d6ff49e37daf104e04f4b4ff2d1ac0469eaa6a20fde084"}, + {file = "httptools-0.6.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:95fb92dd3649f9cb139e9c56604cc2d7c7bf0fc2e7c8d7fbd58f96e35eddd2a3"}, + {file = "httptools-0.6.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:dcbab042cc3ef272adc11220517278519adf8f53fd3056d0e68f0a6f891ba94e"}, + {file = "httptools-0.6.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0cf2372e98406efb42e93bfe10f2948e467edfd792b015f1b4ecd897903d3e8d"}, + {file = "httptools-0.6.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:678fcbae74477a17d103b7cae78b74800d795d702083867ce160fc202104d0da"}, + {file = "httptools-0.6.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:e0b281cf5a125c35f7f6722b65d8542d2e57331be573e9e88bc8b0115c4a7a81"}, + {file = "httptools-0.6.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:95658c342529bba4e1d3d2b1a874db16c7cca435e8827422154c9da76ac4e13a"}, + {file = "httptools-0.6.1-cp39-cp39-win_amd64.whl", hash = "sha256:7ebaec1bf683e4bf5e9fbb49b8cc36da482033596a415b3e4ebab5a4c0d7ec5e"}, + {file = "httptools-0.6.1.tar.gz", hash = "sha256:c6e26c30455600b95d94b1b836085138e82f177351454ee841c148f93a9bad5a"}, +] + +[package.extras] +test = ["Cython (>=0.29.24,<0.30.0)"] + +[[package]] +name = "huggingface-hub" +version = "0.22.2" +description = "Client library to download and publish models, datasets and other repos on the huggingface.co hub" +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "huggingface_hub-0.22.2-py3-none-any.whl", hash = "sha256:3429e25f38ccb834d310804a3b711e7e4953db5a9e420cc147a5e194ca90fd17"}, + {file = "huggingface_hub-0.22.2.tar.gz", hash = "sha256:32e9a9a6843c92f253ff9ca16b9985def4d80a93fb357af5353f770ef74a81be"}, +] + +[package.dependencies] +filelock = "*" +fsspec = ">=2023.5.0" +packaging = ">=20.9" +pyyaml = ">=5.1" +requests = "*" +tqdm = ">=4.42.1" +typing-extensions = ">=3.7.4.3" + +[package.extras] +all = ["InquirerPy (==0.3.4)", "Jinja2", "Pillow", "aiohttp", "gradio", "jedi", "minijinja (>=1.0)", "mypy (==1.5.1)", "numpy", "pytest", "pytest-asyncio", "pytest-cov", "pytest-env", "pytest-rerunfailures", "pytest-vcr", "pytest-xdist", "ruff (>=0.3.0)", "soundfile", "types-PyYAML", "types-requests", "types-simplejson", "types-toml", "types-tqdm", "types-urllib3", "typing-extensions (>=4.8.0)", "urllib3 (<2.0)"] +cli = ["InquirerPy (==0.3.4)"] +dev = ["InquirerPy (==0.3.4)", "Jinja2", "Pillow", "aiohttp", "gradio", "jedi", "minijinja (>=1.0)", "mypy (==1.5.1)", "numpy", "pytest", "pytest-asyncio", "pytest-cov", "pytest-env", "pytest-rerunfailures", "pytest-vcr", "pytest-xdist", "ruff (>=0.3.0)", "soundfile", "types-PyYAML", "types-requests", "types-simplejson", "types-toml", "types-tqdm", "types-urllib3", "typing-extensions (>=4.8.0)", "urllib3 (<2.0)"] +fastai = ["fastai (>=2.4)", "fastcore (>=1.3.27)", "toml"] +hf-transfer = ["hf-transfer (>=0.1.4)"] +inference = ["aiohttp", "minijinja (>=1.0)"] +quality = ["mypy (==1.5.1)", "ruff (>=0.3.0)"] +tensorflow = ["graphviz", "pydot", "tensorflow"] +tensorflow-testing = ["keras (<3.0)", "tensorflow"] +testing = ["InquirerPy (==0.3.4)", "Jinja2", "Pillow", "aiohttp", "gradio", "jedi", "minijinja (>=1.0)", "numpy", "pytest", "pytest-asyncio", "pytest-cov", "pytest-env", "pytest-rerunfailures", "pytest-vcr", "pytest-xdist", "soundfile", "urllib3 (<2.0)"] +torch = ["safetensors", "torch"] +typing = ["types-PyYAML", "types-requests", "types-simplejson", "types-toml", "types-tqdm", "types-urllib3", "typing-extensions (>=4.8.0)"] + +[[package]] +name = "humanfriendly" +version = "10.0" +description = "Human friendly output for text interfaces using Python" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +files = [ + {file = "humanfriendly-10.0-py2.py3-none-any.whl", hash = "sha256:1697e1a8a8f550fd43c2865cd84542fc175a61dcb779b6fee18cf6b6ccba1477"}, + {file = "humanfriendly-10.0.tar.gz", hash = "sha256:6b0b831ce8f15f7300721aa49829fc4e83921a9a301cc7f606be6686a2288ddc"}, +] + +[package.dependencies] +pyreadline3 = {version = "*", markers = "sys_platform == \"win32\" and python_version >= \"3.8\""} + +[[package]] +name = "idna" +version = "3.7" +description = "Internationalized Domain Names in Applications (IDNA)" +optional = false +python-versions = ">=3.5" +files = [ + {file = "idna-3.7-py3-none-any.whl", hash = "sha256:82fee1fc78add43492d3a1898bfa6d8a904cc97d8427f683ed8e798d07761aa0"}, + {file = "idna-3.7.tar.gz", hash = "sha256:028ff3aadf0609c1fd278d8ea3089299412a7a8b9bd005dd08b9f8285bcb5cfc"}, +] + +[[package]] +name = "iniconfig" +version = "2.0.0" +description = "brain-dead simple config-ini parsing" +optional = false +python-versions = ">=3.7" +files = [ + {file = "iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374"}, + {file = "iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3"}, +] + +[[package]] +name = "jinja2" +version = "3.1.3" +description = "A very fast and expressive template engine." +optional = false +python-versions = ">=3.7" +files = [ + {file = "Jinja2-3.1.3-py3-none-any.whl", hash = "sha256:7d6d50dd97d52cbc355597bd845fabfbac3f551e1f99619e39a35ce8c370b5fa"}, + {file = "Jinja2-3.1.3.tar.gz", hash = "sha256:ac8bd6544d4bb2c9792bf3a159e80bba8fda7f07e81bc3aed565432d5925ba90"}, +] + +[package.dependencies] +MarkupSafe = ">=2.0" + +[package.extras] +i18n = ["Babel (>=2.7)"] + +[[package]] +name = "kombu" +version = "5.3.7" +description = "Messaging library for Python." +optional = false +python-versions = ">=3.8" +files = [ + {file = "kombu-5.3.7-py3-none-any.whl", hash = "sha256:5634c511926309c7f9789f1433e9ed402616b56836ef9878f01bd59267b4c7a9"}, + {file = "kombu-5.3.7.tar.gz", hash = "sha256:011c4cd9a355c14a1de8d35d257314a1d2456d52b7140388561acac3cf1a97bf"}, +] + +[package.dependencies] +amqp = ">=5.1.1,<6.0.0" +vine = "*" + +[package.extras] +azureservicebus = ["azure-servicebus (>=7.10.0)"] +azurestoragequeues = ["azure-identity (>=1.12.0)", "azure-storage-queue (>=12.6.0)"] +confluentkafka = ["confluent-kafka (>=2.2.0)"] +consul = ["python-consul2"] +librabbitmq = ["librabbitmq (>=2.0.0)"] +mongodb = ["pymongo (>=4.1.1)"] +msgpack = ["msgpack"] +pyro = ["pyro4"] +qpid = ["qpid-python (>=0.26)", "qpid-tools (>=0.26)"] +redis = ["redis (>=4.5.2,!=4.5.5,!=5.0.2)"] +slmq = ["softlayer-messaging (>=1.0.3)"] +sqlalchemy = ["sqlalchemy (>=1.4.48,<2.1)"] +sqs = ["boto3 (>=1.26.143)", "pycurl (>=7.43.0.5)", "urllib3 (>=1.26.16)"] +yaml = ["PyYAML (>=3.10)"] +zookeeper = ["kazoo (>=2.8.0)"] + +[[package]] +name = "lit" +version = "18.1.4" +description = "A Software Testing Tool" +optional = false +python-versions = "*" +files = [ + {file = "lit-18.1.4-py3-none-any.whl", hash = "sha256:167a8beec8cc3628a38b31727b647953541df043168c84c4e99397515d489054"}, + {file = "lit-18.1.4.tar.gz", hash = "sha256:e6ca26eb0a86aef88cb674616100e32d1250d05cfec4ca57a74acabb0a26de78"}, +] + +[[package]] +name = "llvmlite" +version = "0.39.1" +description = "lightweight wrapper around basic LLVM functionality" +optional = false +python-versions = ">=3.7" +files = [ + {file = "llvmlite-0.39.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:6717c7a6e93c9d2c3d07c07113ec80ae24af45cde536b34363d4bcd9188091d9"}, + {file = "llvmlite-0.39.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ddab526c5a2c4ccb8c9ec4821fcea7606933dc53f510e2a6eebb45a418d3488a"}, + {file = "llvmlite-0.39.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a3f331a323d0f0ada6b10d60182ef06c20a2f01be21699999d204c5750ffd0b4"}, + {file = "llvmlite-0.39.1-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e2c00ff204afa721b0bb9835b5bf1ba7fba210eefcec5552a9e05a63219ba0dc"}, + {file = "llvmlite-0.39.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:16f56eb1eec3cda3a5c526bc3f63594fc24e0c8d219375afeb336f289764c6c7"}, + {file = "llvmlite-0.39.1-cp310-cp310-win32.whl", hash = "sha256:d0bfd18c324549c0fec2c5dc610fd024689de6f27c6cc67e4e24a07541d6e49b"}, + {file = "llvmlite-0.39.1-cp310-cp310-win_amd64.whl", hash = "sha256:7ebf1eb9badc2a397d4f6a6c8717447c81ac011db00064a00408bc83c923c0e4"}, + {file = "llvmlite-0.39.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:6546bed4e02a1c3d53a22a0bced254b3b6894693318b16c16c8e43e29d6befb6"}, + {file = "llvmlite-0.39.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1578f5000fdce513712e99543c50e93758a954297575610f48cb1fd71b27c08a"}, + {file = "llvmlite-0.39.1-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3803f11ad5f6f6c3d2b545a303d68d9fabb1d50e06a8d6418e6fcd2d0df00959"}, + {file = "llvmlite-0.39.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:50aea09a2b933dab7c9df92361b1844ad3145bfb8dd2deb9cd8b8917d59306fb"}, + {file = "llvmlite-0.39.1-cp37-cp37m-win32.whl", hash = "sha256:b1a0bbdb274fb683f993198775b957d29a6f07b45d184c571ef2a721ce4388cf"}, + {file = "llvmlite-0.39.1-cp37-cp37m-win_amd64.whl", hash = "sha256:e172c73fccf7d6db4bd6f7de963dedded900d1a5c6778733241d878ba613980e"}, + {file = "llvmlite-0.39.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:e31f4b799d530255aaf0566e3da2df5bfc35d3cd9d6d5a3dcc251663656c27b1"}, + {file = "llvmlite-0.39.1-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:62c0ea22e0b9dffb020601bb65cb11dd967a095a488be73f07d8867f4e327ca5"}, + {file = "llvmlite-0.39.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9ffc84ade195abd4abcf0bd3b827b9140ae9ef90999429b9ea84d5df69c9058c"}, + {file = "llvmlite-0.39.1-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c0f158e4708dda6367d21cf15afc58de4ebce979c7a1aa2f6b977aae737e2a54"}, + {file = "llvmlite-0.39.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:22d36591cd5d02038912321d9ab8e4668e53ae2211da5523f454e992b5e13c36"}, + {file = "llvmlite-0.39.1-cp38-cp38-win32.whl", hash = "sha256:4c6ebace910410daf0bebda09c1859504fc2f33d122e9a971c4c349c89cca630"}, + {file = "llvmlite-0.39.1-cp38-cp38-win_amd64.whl", hash = "sha256:fb62fc7016b592435d3e3a8f680e3ea8897c3c9e62e6e6cc58011e7a4801439e"}, + {file = "llvmlite-0.39.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:fa9b26939ae553bf30a9f5c4c754db0fb2d2677327f2511e674aa2f5df941789"}, + {file = "llvmlite-0.39.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e4f212c018db951da3e1dc25c2651abc688221934739721f2dad5ff1dd5f90e7"}, + {file = "llvmlite-0.39.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:39dc2160aed36e989610fc403487f11b8764b6650017ff367e45384dff88ffbf"}, + {file = "llvmlite-0.39.1-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1ec3d70b3e507515936e475d9811305f52d049281eaa6c8273448a61c9b5b7e2"}, + {file = "llvmlite-0.39.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:60f8dd1e76f47b3dbdee4b38d9189f3e020d22a173c00f930b52131001d801f9"}, + {file = "llvmlite-0.39.1-cp39-cp39-win32.whl", hash = "sha256:03aee0ccd81735696474dc4f8b6be60774892a2929d6c05d093d17392c237f32"}, + {file = "llvmlite-0.39.1-cp39-cp39-win_amd64.whl", hash = "sha256:3fc14e757bc07a919221f0cbaacb512704ce5774d7fcada793f1996d6bc75f2a"}, + {file = "llvmlite-0.39.1.tar.gz", hash = "sha256:b43abd7c82e805261c425d50335be9a6c4f84264e34d6d6e475207300005d572"}, +] + +[[package]] +name = "markupsafe" +version = "2.1.5" +description = "Safely add untrusted strings to HTML/XML markup." +optional = false +python-versions = ">=3.7" +files = [ + {file = "MarkupSafe-2.1.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:a17a92de5231666cfbe003f0e4b9b3a7ae3afb1ec2845aadc2bacc93ff85febc"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:72b6be590cc35924b02c78ef34b467da4ba07e4e0f0454a2c5907f473fc50ce5"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e61659ba32cf2cf1481e575d0462554625196a1f2fc06a1c777d3f48e8865d46"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2174c595a0d73a3080ca3257b40096db99799265e1c27cc5a610743acd86d62f"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ae2ad8ae6ebee9d2d94b17fb62763125f3f374c25618198f40cbb8b525411900"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:075202fa5b72c86ad32dc7d0b56024ebdbcf2048c0ba09f1cde31bfdd57bcfff"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:598e3276b64aff0e7b3451b72e94fa3c238d452e7ddcd893c3ab324717456bad"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:fce659a462a1be54d2ffcacea5e3ba2d74daa74f30f5f143fe0c58636e355fdd"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-win32.whl", hash = "sha256:d9fad5155d72433c921b782e58892377c44bd6252b5af2f67f16b194987338a4"}, + {file = "MarkupSafe-2.1.5-cp310-cp310-win_amd64.whl", hash = "sha256:bf50cd79a75d181c9181df03572cdce0fbb75cc353bc350712073108cba98de5"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:629ddd2ca402ae6dbedfceeba9c46d5f7b2a61d9749597d4307f943ef198fc1f"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:5b7b716f97b52c5a14bffdf688f971b2d5ef4029127f1ad7a513973cfd818df2"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6ec585f69cec0aa07d945b20805be741395e28ac1627333b1c5b0105962ffced"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b91c037585eba9095565a3556f611e3cbfaa42ca1e865f7b8015fe5c7336d5a5"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7502934a33b54030eaf1194c21c692a534196063db72176b0c4028e140f8f32c"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:0e397ac966fdf721b2c528cf028494e86172b4feba51d65f81ffd65c63798f3f"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:c061bb86a71b42465156a3ee7bd58c8c2ceacdbeb95d05a99893e08b8467359a"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:3a57fdd7ce31c7ff06cdfbf31dafa96cc533c21e443d57f5b1ecc6cdc668ec7f"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-win32.whl", hash = "sha256:397081c1a0bfb5124355710fe79478cdbeb39626492b15d399526ae53422b906"}, + {file = "MarkupSafe-2.1.5-cp311-cp311-win_amd64.whl", hash = "sha256:2b7c57a4dfc4f16f7142221afe5ba4e093e09e728ca65c51f5620c9aaeb9a617"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:8dec4936e9c3100156f8a2dc89c4b88d5c435175ff03413b443469c7c8c5f4d1"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:3c6b973f22eb18a789b1460b4b91bf04ae3f0c4234a0a6aa6b0a92f6f7b951d4"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ac07bad82163452a6884fe8fa0963fb98c2346ba78d779ec06bd7a6262132aee"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f5dfb42c4604dddc8e4305050aa6deb084540643ed5804d7455b5df8fe16f5e5"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ea3d8a3d18833cf4304cd2fc9cbb1efe188ca9b5efef2bdac7adc20594a0e46b"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d050b3361367a06d752db6ead6e7edeb0009be66bc3bae0ee9d97fb326badc2a"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:bec0a414d016ac1a18862a519e54b2fd0fc8bbfd6890376898a6c0891dd82e9f"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:58c98fee265677f63a4385256a6d7683ab1832f3ddd1e66fe948d5880c21a169"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-win32.whl", hash = "sha256:8590b4ae07a35970728874632fed7bd57b26b0102df2d2b233b6d9d82f6c62ad"}, + {file = "MarkupSafe-2.1.5-cp312-cp312-win_amd64.whl", hash = "sha256:823b65d8706e32ad2df51ed89496147a42a2a6e01c13cfb6ffb8b1e92bc910bb"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:c8b29db45f8fe46ad280a7294f5c3ec36dbac9491f2d1c17345be8e69cc5928f"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ec6a563cff360b50eed26f13adc43e61bc0c04d94b8be985e6fb24b81f6dcfdf"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a549b9c31bec33820e885335b451286e2969a2d9e24879f83fe904a5ce59d70a"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4f11aa001c540f62c6166c7726f71f7573b52c68c31f014c25cc7901deea0b52"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:7b2e5a267c855eea6b4283940daa6e88a285f5f2a67f2220203786dfa59b37e9"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:2d2d793e36e230fd32babe143b04cec8a8b3eb8a3122d2aceb4a371e6b09b8df"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:ce409136744f6521e39fd8e2a24c53fa18ad67aa5bc7c2cf83645cce5b5c4e50"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-win32.whl", hash = "sha256:4096e9de5c6fdf43fb4f04c26fb114f61ef0bf2e5604b6ee3019d51b69e8c371"}, + {file = "MarkupSafe-2.1.5-cp37-cp37m-win_amd64.whl", hash = "sha256:4275d846e41ecefa46e2015117a9f491e57a71ddd59bbead77e904dc02b1bed2"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:656f7526c69fac7f600bd1f400991cc282b417d17539a1b228617081106feb4a"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:97cafb1f3cbcd3fd2b6fbfb99ae11cdb14deea0736fc2b0952ee177f2b813a46"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1f3fbcb7ef1f16e48246f704ab79d79da8a46891e2da03f8783a5b6fa41a9532"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:fa9db3f79de01457b03d4f01b34cf91bc0048eb2c3846ff26f66687c2f6d16ab"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ffee1f21e5ef0d712f9033568f8344d5da8cc2869dbd08d87c84656e6a2d2f68"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:5dedb4db619ba5a2787a94d877bc8ffc0566f92a01c0ef214865e54ecc9ee5e0"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:30b600cf0a7ac9234b2638fbc0fb6158ba5bdcdf46aeb631ead21248b9affbc4"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:8dd717634f5a044f860435c1d8c16a270ddf0ef8588d4887037c5028b859b0c3"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-win32.whl", hash = "sha256:daa4ee5a243f0f20d528d939d06670a298dd39b1ad5f8a72a4275124a7819eff"}, + {file = "MarkupSafe-2.1.5-cp38-cp38-win_amd64.whl", hash = "sha256:619bc166c4f2de5caa5a633b8b7326fbe98e0ccbfacabd87268a2b15ff73a029"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:7a68b554d356a91cce1236aa7682dc01df0edba8d043fd1ce607c49dd3c1edcf"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:db0b55e0f3cc0be60c1f19efdde9a637c32740486004f20d1cff53c3c0ece4d2"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3e53af139f8579a6d5f7b76549125f0d94d7e630761a2111bc431fd820e163b8"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:17b950fccb810b3293638215058e432159d2b71005c74371d784862b7e4683f3"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:4c31f53cdae6ecfa91a77820e8b151dba54ab528ba65dfd235c80b086d68a465"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:bff1b4290a66b490a2f4719358c0cdcd9bafb6b8f061e45c7a2460866bf50c2e"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:bc1667f8b83f48511b94671e0e441401371dfd0f0a795c7daa4a3cd1dde55bea"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5049256f536511ee3f7e1b3f87d1d1209d327e818e6ae1365e8653d7e3abb6a6"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-win32.whl", hash = "sha256:00e046b6dd71aa03a41079792f8473dc494d564611a8f89bbbd7cb93295ebdcf"}, + {file = "MarkupSafe-2.1.5-cp39-cp39-win_amd64.whl", hash = "sha256:fa173ec60341d6bb97a89f5ea19c85c5643c1e7dedebc22f5181eb73573142c5"}, + {file = "MarkupSafe-2.1.5.tar.gz", hash = "sha256:d283d37a890ba4c1ae73ffadf8046435c76e7bc2247bbb63c00bd1a709c6544b"}, +] + +[[package]] +name = "more-itertools" +version = "10.2.0" +description = "More routines for operating on iterables, beyond itertools" +optional = false +python-versions = ">=3.8" +files = [ + {file = "more-itertools-10.2.0.tar.gz", hash = "sha256:8fccb480c43d3e99a00087634c06dd02b0d50fbf088b380de5a41a015ec239e1"}, + {file = "more_itertools-10.2.0-py3-none-any.whl", hash = "sha256:686b06abe565edfab151cb8fd385a05651e1fdf8f0a14191e4439283421f8684"}, +] + +[[package]] +name = "mpmath" +version = "1.3.0" +description = "Python library for arbitrary-precision floating-point arithmetic" +optional = false +python-versions = "*" +files = [ + {file = "mpmath-1.3.0-py3-none-any.whl", hash = "sha256:a0b2b9fe80bbcd81a6647ff13108738cfb482d481d826cc0e02f5b35e5c88d2c"}, + {file = "mpmath-1.3.0.tar.gz", hash = "sha256:7a28eb2a9774d00c7bc92411c19a89209d5da7c4c9a9e227be8330a23a25b91f"}, +] + +[package.extras] +develop = ["codecov", "pycodestyle", "pytest (>=4.6)", "pytest-cov", "wheel"] +docs = ["sphinx"] +gmpy = ["gmpy2 (>=2.1.0a4)"] +tests = ["pytest (>=4.6)"] + +[[package]] +name = "numba" +version = "0.56.4" +description = "compiling Python code using LLVM" +optional = false +python-versions = ">=3.7" +files = [ + {file = "numba-0.56.4-cp310-cp310-macosx_10_14_x86_64.whl", hash = "sha256:9f62672145f8669ec08762895fe85f4cf0ead08ce3164667f2b94b2f62ab23c3"}, + {file = "numba-0.56.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c602d015478b7958408d788ba00a50272649c5186ea8baa6cf71d4a1c761bba1"}, + {file = "numba-0.56.4-cp310-cp310-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:85dbaed7a05ff96492b69a8900c5ba605551afb9b27774f7f10511095451137c"}, + {file = "numba-0.56.4-cp310-cp310-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:f4cfc3a19d1e26448032049c79fc60331b104f694cf570a9e94f4e2c9d0932bb"}, + {file = "numba-0.56.4-cp310-cp310-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:4e08e203b163ace08bad500b0c16f6092b1eb34fd1fce4feaf31a67a3a5ecf3b"}, + {file = "numba-0.56.4-cp310-cp310-win32.whl", hash = "sha256:0611e6d3eebe4cb903f1a836ffdb2bda8d18482bcd0a0dcc56e79e2aa3fefef5"}, + {file = "numba-0.56.4-cp310-cp310-win_amd64.whl", hash = "sha256:fbfb45e7b297749029cb28694abf437a78695a100e7c2033983d69f0ba2698d4"}, + {file = "numba-0.56.4-cp37-cp37m-macosx_10_14_x86_64.whl", hash = "sha256:3cb1a07a082a61df80a468f232e452d818f5ae254b40c26390054e4e868556e0"}, + {file = "numba-0.56.4-cp37-cp37m-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:d69ad934e13c15684e7887100a8f5f0f61d7a8e57e0fd29d9993210089a5b531"}, + {file = "numba-0.56.4-cp37-cp37m-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:dbcc847bac2d225265d054993a7f910fda66e73d6662fe7156452cac0325b073"}, + {file = "numba-0.56.4-cp37-cp37m-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:8a95ca9cc77ea4571081f6594e08bd272b66060634b8324e99cd1843020364f9"}, + {file = "numba-0.56.4-cp37-cp37m-win32.whl", hash = "sha256:fcdf84ba3ed8124eb7234adfbb8792f311991cbf8aed1cad4b1b1a7ee08380c1"}, + {file = "numba-0.56.4-cp37-cp37m-win_amd64.whl", hash = "sha256:42f9e1be942b215df7e6cc9948cf9c15bb8170acc8286c063a9e57994ef82fd1"}, + {file = "numba-0.56.4-cp38-cp38-macosx_10_14_x86_64.whl", hash = "sha256:553da2ce74e8862e18a72a209ed3b6d2924403bdd0fb341fa891c6455545ba7c"}, + {file = "numba-0.56.4-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:4373da9757049db7c90591e9ec55a2e97b2b36ba7ae3bf9c956a513374077470"}, + {file = "numba-0.56.4-cp38-cp38-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:3a993349b90569518739009d8f4b523dfedd7e0049e6838c0e17435c3e70dcc4"}, + {file = "numba-0.56.4-cp38-cp38-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:720886b852a2d62619ae3900fe71f1852c62db4f287d0c275a60219e1643fc04"}, + {file = "numba-0.56.4-cp38-cp38-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:e64d338b504c9394a4a34942df4627e1e6cb07396ee3b49fe7b8d6420aa5104f"}, + {file = "numba-0.56.4-cp38-cp38-win32.whl", hash = "sha256:03fe94cd31e96185cce2fae005334a8cc712fc2ba7756e52dff8c9400718173f"}, + {file = "numba-0.56.4-cp38-cp38-win_amd64.whl", hash = "sha256:91f021145a8081f881996818474ef737800bcc613ffb1e618a655725a0f9e246"}, + {file = "numba-0.56.4-cp39-cp39-macosx_10_14_x86_64.whl", hash = "sha256:d0ae9270a7a5cc0ede63cd234b4ff1ce166c7a749b91dbbf45e0000c56d3eade"}, + {file = "numba-0.56.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:c75e8a5f810ce80a0cfad6e74ee94f9fde9b40c81312949bf356b7304ef20740"}, + {file = "numba-0.56.4-cp39-cp39-manylinux2014_aarch64.manylinux_2_17_aarch64.whl", hash = "sha256:a12ef323c0f2101529d455cfde7f4135eaa147bad17afe10b48634f796d96abd"}, + {file = "numba-0.56.4-cp39-cp39-manylinux2014_i686.manylinux_2_17_i686.whl", hash = "sha256:03634579d10a6129181129de293dd6b5eaabee86881369d24d63f8fe352dd6cb"}, + {file = "numba-0.56.4-cp39-cp39-manylinux2014_x86_64.manylinux_2_17_x86_64.whl", hash = "sha256:0240f9026b015e336069329839208ebd70ec34ae5bfbf402e4fcc8e06197528e"}, + {file = "numba-0.56.4-cp39-cp39-win32.whl", hash = "sha256:14dbbabf6ffcd96ee2ac827389afa59a70ffa9f089576500434c34abf9b054a4"}, + {file = "numba-0.56.4-cp39-cp39-win_amd64.whl", hash = "sha256:0da583c532cd72feefd8e551435747e0e0fbb3c0530357e6845fcc11e38d6aea"}, + {file = "numba-0.56.4.tar.gz", hash = "sha256:32d9fef412c81483d7efe0ceb6cf4d3310fde8b624a9cecca00f790573ac96ee"}, +] + +[package.dependencies] +llvmlite = "==0.39.*" +numpy = ">=1.18,<1.24" +setuptools = "*" + +[[package]] +name = "numpy" +version = "1.23.5" +description = "NumPy is the fundamental package for array computing with Python." +optional = false +python-versions = ">=3.8" +files = [ + {file = "numpy-1.23.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:9c88793f78fca17da0145455f0d7826bcb9f37da4764af27ac945488116efe63"}, + {file = "numpy-1.23.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:e9f4c4e51567b616be64e05d517c79a8a22f3606499941d97bb76f2ca59f982d"}, + {file = "numpy-1.23.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7903ba8ab592b82014713c491f6c5d3a1cde5b4a3bf116404e08f5b52f6daf43"}, + {file = "numpy-1.23.5-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5e05b1c973a9f858c74367553e236f287e749465f773328c8ef31abe18f691e1"}, + {file = "numpy-1.23.5-cp310-cp310-win32.whl", hash = "sha256:522e26bbf6377e4d76403826ed689c295b0b238f46c28a7251ab94716da0b280"}, + {file = "numpy-1.23.5-cp310-cp310-win_amd64.whl", hash = "sha256:dbee87b469018961d1ad79b1a5d50c0ae850000b639bcb1b694e9981083243b6"}, + {file = "numpy-1.23.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:ce571367b6dfe60af04e04a1834ca2dc5f46004ac1cc756fb95319f64c095a96"}, + {file = "numpy-1.23.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:56e454c7833e94ec9769fa0f86e6ff8e42ee38ce0ce1fa4cbb747ea7e06d56aa"}, + {file = "numpy-1.23.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5039f55555e1eab31124a5768898c9e22c25a65c1e0037f4d7c495a45778c9f2"}, + {file = "numpy-1.23.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:58f545efd1108e647604a1b5aa809591ccd2540f468a880bedb97247e72db387"}, + {file = "numpy-1.23.5-cp311-cp311-win32.whl", hash = "sha256:b2a9ab7c279c91974f756c84c365a669a887efa287365a8e2c418f8b3ba73fb0"}, + {file = "numpy-1.23.5-cp311-cp311-win_amd64.whl", hash = "sha256:0cbe9848fad08baf71de1a39e12d1b6310f1d5b2d0ea4de051058e6e1076852d"}, + {file = "numpy-1.23.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:f063b69b090c9d918f9df0a12116029e274daf0181df392839661c4c7ec9018a"}, + {file = "numpy-1.23.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:0aaee12d8883552fadfc41e96b4c82ee7d794949e2a7c3b3a7201e968c7ecab9"}, + {file = "numpy-1.23.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:92c8c1e89a1f5028a4c6d9e3ccbe311b6ba53694811269b992c0b224269e2398"}, + {file = "numpy-1.23.5-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d208a0f8729f3fb790ed18a003f3a57895b989b40ea4dce4717e9cf4af62c6bb"}, + {file = "numpy-1.23.5-cp38-cp38-win32.whl", hash = "sha256:06005a2ef6014e9956c09ba07654f9837d9e26696a0470e42beedadb78c11b07"}, + {file = "numpy-1.23.5-cp38-cp38-win_amd64.whl", hash = "sha256:ca51fcfcc5f9354c45f400059e88bc09215fb71a48d3768fb80e357f3b457e1e"}, + {file = "numpy-1.23.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8969bfd28e85c81f3f94eb4a66bc2cf1dbdc5c18efc320af34bffc54d6b1e38f"}, + {file = "numpy-1.23.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:a7ac231a08bb37f852849bbb387a20a57574a97cfc7b6cabb488a4fc8be176de"}, + {file = "numpy-1.23.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bf837dc63ba5c06dc8797c398db1e223a466c7ece27a1f7b5232ba3466aafe3d"}, + {file = "numpy-1.23.5-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:33161613d2269025873025b33e879825ec7b1d831317e68f4f2f0f84ed14c719"}, + {file = "numpy-1.23.5-cp39-cp39-win32.whl", hash = "sha256:af1da88f6bc3d2338ebbf0e22fe487821ea4d8e89053e25fa59d1d79786e7481"}, + {file = "numpy-1.23.5-cp39-cp39-win_amd64.whl", hash = "sha256:09b7847f7e83ca37c6e627682f145856de331049013853f344f37b0c9690e3df"}, + {file = "numpy-1.23.5-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:abdde9f795cf292fb9651ed48185503a2ff29be87770c3b8e2a14b0cd7aa16f8"}, + {file = "numpy-1.23.5-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f9a909a8bae284d46bbfdefbdd4a262ba19d3bc9921b1e76126b1d21c3c34135"}, + {file = "numpy-1.23.5-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:01dd17cbb340bf0fc23981e52e1d18a9d4050792e8fb8363cecbf066a84b827d"}, + {file = "numpy-1.23.5.tar.gz", hash = "sha256:1b1766d6f397c18153d40015ddfc79ddb715cabadc04d2d228d4e5a8bc4ded1a"}, +] + +[[package]] +name = "onnxruntime" +version = "1.17.3" +description = "ONNX Runtime is a runtime accelerator for Machine Learning models" +optional = false +python-versions = "*" +files = [ + {file = "onnxruntime-1.17.3-cp310-cp310-macosx_11_0_universal2.whl", hash = "sha256:d86dde9c0bb435d709e51bd25991c9fe5b9a5b168df45ce119769edc4d198b15"}, + {file = "onnxruntime-1.17.3-cp310-cp310-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:9d87b68bf931ac527b2d3c094ead66bb4381bac4298b65f46c54fe4d1e255865"}, + {file = "onnxruntime-1.17.3-cp310-cp310-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:26e950cf0333cf114a155f9142e71da344d2b08dfe202763a403ae81cc02ebd1"}, + {file = "onnxruntime-1.17.3-cp310-cp310-win32.whl", hash = "sha256:0962a4d0f5acebf62e1f0bf69b6e0adf16649115d8de854c1460e79972324d68"}, + {file = "onnxruntime-1.17.3-cp310-cp310-win_amd64.whl", hash = "sha256:468ccb8a0faa25c681a41787b1594bf4448b0252d3efc8b62fd8b2411754340f"}, + {file = "onnxruntime-1.17.3-cp311-cp311-macosx_11_0_universal2.whl", hash = "sha256:e8cd90c1c17d13d47b89ab076471e07fb85467c01dcd87a8b8b5cdfbcb40aa51"}, + {file = "onnxruntime-1.17.3-cp311-cp311-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:a058b39801baefe454eeb8acf3ada298c55a06a4896fafc224c02d79e9037f60"}, + {file = "onnxruntime-1.17.3-cp311-cp311-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2f823d5eb4807007f3da7b27ca972263df6a1836e6f327384eb266274c53d05d"}, + {file = "onnxruntime-1.17.3-cp311-cp311-win32.whl", hash = "sha256:b66b23f9109e78ff2791628627a26f65cd335dcc5fbd67ff60162733a2f7aded"}, + {file = "onnxruntime-1.17.3-cp311-cp311-win_amd64.whl", hash = "sha256:570760ca53a74cdd751ee49f13de70d1384dcf73d9888b8deac0917023ccda6d"}, + {file = "onnxruntime-1.17.3-cp312-cp312-macosx_11_0_universal2.whl", hash = "sha256:77c318178d9c16e9beadd9a4070d8aaa9f57382c3f509b01709f0f010e583b99"}, + {file = "onnxruntime-1.17.3-cp312-cp312-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:23da8469049b9759082e22c41a444f44a520a9c874b084711b6343672879f50b"}, + {file = "onnxruntime-1.17.3-cp312-cp312-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:2949730215af3f9289008b2e31e9bbef952012a77035b911c4977edea06f3f9e"}, + {file = "onnxruntime-1.17.3-cp312-cp312-win32.whl", hash = "sha256:6c7555a49008f403fb3b19204671efb94187c5085976ae526cb625f6ede317bc"}, + {file = "onnxruntime-1.17.3-cp312-cp312-win_amd64.whl", hash = "sha256:58672cf20293a1b8a277a5c6c55383359fcdf6119b2f14df6ce3b140f5001c39"}, + {file = "onnxruntime-1.17.3-cp38-cp38-macosx_11_0_universal2.whl", hash = "sha256:4395ba86e3c1e93c794a00619ef1aec597ab78f5a5039f3c6d2e9d0695c0a734"}, + {file = "onnxruntime-1.17.3-cp38-cp38-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:bdf354c04344ec38564fc22394e1fe08aa6d70d790df00159205a0055c4a4d3f"}, + {file = "onnxruntime-1.17.3-cp38-cp38-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:a94b600b7af50e922d44b95a57981e3e35103c6e3693241a03d3ca204740bbda"}, + {file = "onnxruntime-1.17.3-cp38-cp38-win32.whl", hash = "sha256:5a335c76f9c002a8586c7f38bc20fe4b3725ced21f8ead835c3e4e507e42b2ab"}, + {file = "onnxruntime-1.17.3-cp38-cp38-win_amd64.whl", hash = "sha256:8f56a86fbd0ddc8f22696ddeda0677b041381f4168a2ca06f712ef6ec6050d6d"}, + {file = "onnxruntime-1.17.3-cp39-cp39-macosx_11_0_universal2.whl", hash = "sha256:e0ae39f5452278cd349520c296e7de3e90d62dc5b0157c6868e2748d7f28b871"}, + {file = "onnxruntime-1.17.3-cp39-cp39-manylinux_2_27_aarch64.manylinux_2_28_aarch64.whl", hash = "sha256:3ff2dc012bd930578aff5232afd2905bf16620815f36783a941aafabf94b3702"}, + {file = "onnxruntime-1.17.3-cp39-cp39-manylinux_2_27_x86_64.manylinux_2_28_x86_64.whl", hash = "sha256:cf6c37483782e4785019b56e26224a25e9b9a35b849d0169ce69189867a22bb1"}, + {file = "onnxruntime-1.17.3-cp39-cp39-win32.whl", hash = "sha256:351bf5a1140dcc43bfb8d3d1a230928ee61fcd54b0ea664c8e9a889a8e3aa515"}, + {file = "onnxruntime-1.17.3-cp39-cp39-win_amd64.whl", hash = "sha256:57a3de15778da8d6cc43fbf6cf038e1e746146300b5f0b1fbf01f6f795dc6440"}, +] + +[package.dependencies] +coloredlogs = "*" +flatbuffers = "*" +numpy = ">=1.21.6" +packaging = "*" +protobuf = "*" +sympy = "*" + +[[package]] +name = "openai-whisper" +version = "20230918" +description = "Robust Speech Recognition via Large-Scale Weak Supervision" +optional = false +python-versions = ">=3.8" +files = [ + {file = "openai-whisper-20230918.tar.gz", hash = "sha256:32a1ee39c3faaf6c719e3a83f1aacc8e164aad87976350371e26845271287c30"}, +] + +[package.dependencies] +more-itertools = "*" +numba = "*" +numpy = "*" +tiktoken = "0.3.3" +torch = "*" +tqdm = "*" + +[package.extras] +dev = ["black", "flake8", "isort", "pytest", "scipy"] + +[[package]] +name = "packaging" +version = "24.0" +description = "Core utilities for Python packages" +optional = false +python-versions = ">=3.7" +files = [ + {file = "packaging-24.0-py3-none-any.whl", hash = "sha256:2ddfb553fdf02fb784c234c7ba6ccc288296ceabec964ad2eae3777778130bc5"}, + {file = "packaging-24.0.tar.gz", hash = "sha256:eb82c5e3e56209074766e6885bb04b8c38a0c015d0a30036ebe7ece34c9989e9"}, +] + +[[package]] +name = "pluggy" +version = "1.5.0" +description = "plugin and hook calling mechanisms for python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pluggy-1.5.0-py3-none-any.whl", hash = "sha256:44e1ad92c8ca002de6377e165f3e0f1be63266ab4d554740532335b9d75ea669"}, + {file = "pluggy-1.5.0.tar.gz", hash = "sha256:2cffa88e94fdc978c4c574f15f9e59b7f4201d439195c3715ca9e2486f1d0cf1"}, +] + +[package.extras] +dev = ["pre-commit", "tox"] +testing = ["pytest", "pytest-benchmark"] + +[[package]] +name = "prompt-toolkit" +version = "3.0.43" +description = "Library for building powerful interactive command lines in Python" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "prompt_toolkit-3.0.43-py3-none-any.whl", hash = "sha256:a11a29cb3bf0a28a387fe5122cdb649816a957cd9261dcedf8c9f1fef33eacf6"}, + {file = "prompt_toolkit-3.0.43.tar.gz", hash = "sha256:3527b7af26106cbc65a040bcc84839a3566ec1b051bb0bfe953631e704b0ff7d"}, +] + +[package.dependencies] +wcwidth = "*" + +[[package]] +name = "protobuf" +version = "5.26.1" +description = "" +optional = false +python-versions = ">=3.8" +files = [ + {file = "protobuf-5.26.1-cp310-abi3-win32.whl", hash = "sha256:3c388ea6ddfe735f8cf69e3f7dc7611e73107b60bdfcf5d0f024c3ccd3794e23"}, + {file = "protobuf-5.26.1-cp310-abi3-win_amd64.whl", hash = "sha256:e6039957449cb918f331d32ffafa8eb9255769c96aa0560d9a5bf0b4e00a2a33"}, + {file = "protobuf-5.26.1-cp37-abi3-macosx_10_9_universal2.whl", hash = "sha256:38aa5f535721d5bb99861166c445c4105c4e285c765fbb2ac10f116e32dcd46d"}, + {file = "protobuf-5.26.1-cp37-abi3-manylinux2014_aarch64.whl", hash = "sha256:fbfe61e7ee8c1860855696e3ac6cfd1b01af5498facc6834fcc345c9684fb2ca"}, + {file = "protobuf-5.26.1-cp37-abi3-manylinux2014_x86_64.whl", hash = "sha256:f7417703f841167e5a27d48be13389d52ad705ec09eade63dfc3180a959215d7"}, + {file = "protobuf-5.26.1-cp38-cp38-win32.whl", hash = "sha256:d693d2504ca96750d92d9de8a103102dd648fda04540495535f0fec7577ed8fc"}, + {file = "protobuf-5.26.1-cp38-cp38-win_amd64.whl", hash = "sha256:9b557c317ebe6836835ec4ef74ec3e994ad0894ea424314ad3552bc6e8835b4e"}, + {file = "protobuf-5.26.1-cp39-cp39-win32.whl", hash = "sha256:b9ba3ca83c2e31219ffbeb9d76b63aad35a3eb1544170c55336993d7a18ae72c"}, + {file = "protobuf-5.26.1-cp39-cp39-win_amd64.whl", hash = "sha256:7ee014c2c87582e101d6b54260af03b6596728505c79f17c8586e7523aaa8f8c"}, + {file = "protobuf-5.26.1-py3-none-any.whl", hash = "sha256:da612f2720c0183417194eeaa2523215c4fcc1a1949772dc65f05047e08d5932"}, + {file = "protobuf-5.26.1.tar.gz", hash = "sha256:8ca2a1d97c290ec7b16e4e5dff2e5ae150cc1582f55b5ab300d45cb0dfa90e51"}, +] + +[[package]] +name = "py" +version = "1.11.0" +description = "library with cross-python path, ini-parsing, io, code, log facilities" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +files = [ + {file = "py-1.11.0-py2.py3-none-any.whl", hash = "sha256:607c53218732647dff4acdfcd50cb62615cedf612e72d1724fb1a0cc6405b378"}, + {file = "py-1.11.0.tar.gz", hash = "sha256:51c75c4126074b472f746a24399ad32f6053d1b34b68d2fa41e558e6f4a98719"}, +] + +[[package]] +name = "pydantic" +version = "1.10.15" +description = "Data validation and settings management using python type hints" +optional = false +python-versions = ">=3.7" +files = [ + {file = "pydantic-1.10.15-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:22ed12ee588b1df028a2aa5d66f07bf8f8b4c8579c2e96d5a9c1f96b77f3bb55"}, + {file = "pydantic-1.10.15-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:75279d3cac98186b6ebc2597b06bcbc7244744f6b0b44a23e4ef01e5683cc0d2"}, + {file = "pydantic-1.10.15-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:50f1666a9940d3d68683c9d96e39640f709d7a72ff8702987dab1761036206bb"}, + {file = "pydantic-1.10.15-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:82790d4753ee5d00739d6cb5cf56bceb186d9d6ce134aca3ba7befb1eedbc2c8"}, + {file = "pydantic-1.10.15-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:d207d5b87f6cbefbdb1198154292faee8017d7495a54ae58db06762004500d00"}, + {file = "pydantic-1.10.15-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:e49db944fad339b2ccb80128ffd3f8af076f9f287197a480bf1e4ca053a866f0"}, + {file = "pydantic-1.10.15-cp310-cp310-win_amd64.whl", hash = "sha256:d3b5c4cbd0c9cb61bbbb19ce335e1f8ab87a811f6d589ed52b0254cf585d709c"}, + {file = "pydantic-1.10.15-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:c3d5731a120752248844676bf92f25a12f6e45425e63ce22e0849297a093b5b0"}, + {file = "pydantic-1.10.15-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:c365ad9c394f9eeffcb30a82f4246c0006417f03a7c0f8315d6211f25f7cb654"}, + {file = "pydantic-1.10.15-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3287e1614393119c67bd4404f46e33ae3be3ed4cd10360b48d0a4459f420c6a3"}, + {file = "pydantic-1.10.15-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:be51dd2c8596b25fe43c0a4a59c2bee4f18d88efb8031188f9e7ddc6b469cf44"}, + {file = "pydantic-1.10.15-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:6a51a1dd4aa7b3f1317f65493a182d3cff708385327c1c82c81e4a9d6d65b2e4"}, + {file = "pydantic-1.10.15-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:4e316e54b5775d1eb59187f9290aeb38acf620e10f7fd2f776d97bb788199e53"}, + {file = "pydantic-1.10.15-cp311-cp311-win_amd64.whl", hash = "sha256:0d142fa1b8f2f0ae11ddd5e3e317dcac060b951d605fda26ca9b234b92214986"}, + {file = "pydantic-1.10.15-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:7ea210336b891f5ea334f8fc9f8f862b87acd5d4a0cbc9e3e208e7aa1775dabf"}, + {file = "pydantic-1.10.15-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3453685ccd7140715e05f2193d64030101eaad26076fad4e246c1cc97e1bb30d"}, + {file = "pydantic-1.10.15-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9bea1f03b8d4e8e86702c918ccfd5d947ac268f0f0cc6ed71782e4b09353b26f"}, + {file = "pydantic-1.10.15-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:005655cabc29081de8243126e036f2065bd7ea5b9dff95fde6d2c642d39755de"}, + {file = "pydantic-1.10.15-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:af9850d98fc21e5bc24ea9e35dd80a29faf6462c608728a110c0a30b595e58b7"}, + {file = "pydantic-1.10.15-cp37-cp37m-win_amd64.whl", hash = "sha256:d31ee5b14a82c9afe2bd26aaa405293d4237d0591527d9129ce36e58f19f95c1"}, + {file = "pydantic-1.10.15-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:5e09c19df304b8123938dc3c53d3d3be6ec74b9d7d0d80f4f4b5432ae16c2022"}, + {file = "pydantic-1.10.15-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:7ac9237cd62947db00a0d16acf2f3e00d1ae9d3bd602b9c415f93e7a9fc10528"}, + {file = "pydantic-1.10.15-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:584f2d4c98ffec420e02305cf675857bae03c9d617fcfdc34946b1160213a948"}, + {file = "pydantic-1.10.15-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bbc6989fad0c030bd70a0b6f626f98a862224bc2b1e36bfc531ea2facc0a340c"}, + {file = "pydantic-1.10.15-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:d573082c6ef99336f2cb5b667b781d2f776d4af311574fb53d908517ba523c22"}, + {file = "pydantic-1.10.15-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:6bd7030c9abc80134087d8b6e7aa957e43d35714daa116aced57269a445b8f7b"}, + {file = "pydantic-1.10.15-cp38-cp38-win_amd64.whl", hash = "sha256:3350f527bb04138f8aff932dc828f154847fbdc7a1a44c240fbfff1b57f49a12"}, + {file = "pydantic-1.10.15-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:51d405b42f1b86703555797270e4970a9f9bd7953f3990142e69d1037f9d9e51"}, + {file = "pydantic-1.10.15-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:a980a77c52723b0dc56640ced396b73a024d4b74f02bcb2d21dbbac1debbe9d0"}, + {file = "pydantic-1.10.15-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:67f1a1fb467d3f49e1708a3f632b11c69fccb4e748a325d5a491ddc7b5d22383"}, + {file = "pydantic-1.10.15-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:676ed48f2c5bbad835f1a8ed8a6d44c1cd5a21121116d2ac40bd1cd3619746ed"}, + {file = "pydantic-1.10.15-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:92229f73400b80c13afcd050687f4d7e88de9234d74b27e6728aa689abcf58cc"}, + {file = "pydantic-1.10.15-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:2746189100c646682eff0bce95efa7d2e203420d8e1c613dc0c6b4c1d9c1fde4"}, + {file = "pydantic-1.10.15-cp39-cp39-win_amd64.whl", hash = "sha256:394f08750bd8eaad714718812e7fab615f873b3cdd0b9d84e76e51ef3b50b6b7"}, + {file = "pydantic-1.10.15-py3-none-any.whl", hash = "sha256:28e552a060ba2740d0d2aabe35162652c1459a0b9069fe0db7f4ee0e18e74d58"}, + {file = "pydantic-1.10.15.tar.gz", hash = "sha256:ca832e124eda231a60a041da4f013e3ff24949d94a01154b137fc2f2a43c3ffb"}, +] + +[package.dependencies] +typing-extensions = ">=4.2.0" + +[package.extras] +dotenv = ["python-dotenv (>=0.10.4)"] +email = ["email-validator (>=1.0.3)"] + +[[package]] +name = "pyreadline3" +version = "3.4.1" +description = "A python implementation of GNU readline." +optional = false +python-versions = "*" +files = [ + {file = "pyreadline3-3.4.1-py3-none-any.whl", hash = "sha256:b0efb6516fd4fb07b45949053826a62fa4cb353db5be2bbb4a7aa1fdd1e345fb"}, + {file = "pyreadline3-3.4.1.tar.gz", hash = "sha256:6f3d1f7b8a31ba32b73917cefc1f28cc660562f39aea8646d30bd6eff21f7bae"}, +] + +[[package]] +name = "pytest" +version = "6.2.5" +description = "pytest: simple powerful testing with Python" +optional = false +python-versions = ">=3.6" +files = [ + {file = "pytest-6.2.5-py3-none-any.whl", hash = "sha256:7310f8d27bc79ced999e760ca304d69f6ba6c6649c0b60fb0e04a4a77cacc134"}, + {file = "pytest-6.2.5.tar.gz", hash = "sha256:131b36680866a76e6781d13f101efb86cf674ebb9762eb70d3082b6f29889e89"}, +] + +[package.dependencies] +atomicwrites = {version = ">=1.0", markers = "sys_platform == \"win32\""} +attrs = ">=19.2.0" +colorama = {version = "*", markers = "sys_platform == \"win32\""} +iniconfig = "*" +packaging = "*" +pluggy = ">=0.12,<2.0" +py = ">=1.8.2" +toml = "*" + +[package.extras] +testing = ["argcomplete", "hypothesis (>=3.56)", "mock", "nose", "requests", "xmlschema"] + +[[package]] +name = "python-dateutil" +version = "2.9.0.post0" +description = "Extensions to the standard Python datetime module" +optional = false +python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" +files = [ + {file = "python-dateutil-2.9.0.post0.tar.gz", hash = "sha256:37dd54208da7e1cd875388217d5e00ebd4179249f90fb72437e91a35459a0ad3"}, + {file = "python_dateutil-2.9.0.post0-py2.py3-none-any.whl", hash = "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427"}, +] + +[package.dependencies] +six = ">=1.5" + +[[package]] +name = "python-dotenv" +version = "1.0.1" +description = "Read key-value pairs from a .env file and set them as environment variables" +optional = false +python-versions = ">=3.8" +files = [ + {file = "python-dotenv-1.0.1.tar.gz", hash = "sha256:e324ee90a023d808f1959c46bcbc04446a10ced277783dc6ee09987c37ec10ca"}, + {file = "python_dotenv-1.0.1-py3-none-any.whl", hash = "sha256:f7b63ef50f1b690dddf550d03497b66d609393b40b564ed0d674909a68ebf16a"}, +] + +[package.extras] +cli = ["click (>=5.0)"] + +[[package]] +name = "python-multipart" +version = "0.0.5" +description = "A streaming multipart parser for Python" +optional = false +python-versions = "*" +files = [ + {file = "python-multipart-0.0.5.tar.gz", hash = "sha256:f7bb5f611fc600d15fa47b3974c8aa16e93724513b49b5f95c81e6624c83fa43"}, +] + +[package.dependencies] +six = ">=1.4.0" + +[[package]] +name = "pyyaml" +version = "6.0.1" +description = "YAML parser and emitter for Python" +optional = false +python-versions = ">=3.6" +files = [ + {file = "PyYAML-6.0.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d858aa552c999bc8a8d57426ed01e40bef403cd8ccdd0fc5f6f04a00414cac2a"}, + {file = "PyYAML-6.0.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:fd66fc5d0da6d9815ba2cebeb4205f95818ff4b79c3ebe268e75d961704af52f"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:69b023b2b4daa7548bcfbd4aa3da05b3a74b772db9e23b982788168117739938"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:81e0b275a9ecc9c0c0c07b4b90ba548307583c125f54d5b6946cfee6360c733d"}, + {file = "PyYAML-6.0.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba336e390cd8e4d1739f42dfe9bb83a3cc2e80f567d8805e11b46f4a943f5515"}, + {file = "PyYAML-6.0.1-cp310-cp310-win32.whl", hash = "sha256:bd4af7373a854424dabd882decdc5579653d7868b8fb26dc7d0e99f823aa5924"}, + {file = "PyYAML-6.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:fd1592b3fdf65fff2ad0004b5e363300ef59ced41c2e6b3a99d4089fa8c5435d"}, + {file = "PyYAML-6.0.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6965a7bc3cf88e5a1c3bd2e0b5c22f8d677dc88a455344035f03399034eb3007"}, + {file = "PyYAML-6.0.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:f003ed9ad21d6a4713f0a9b5a7a0a79e08dd0f221aff4525a2be4c346ee60aab"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:42f8152b8dbc4fe7d96729ec2b99c7097d656dc1213a3229ca5383f973a5ed6d"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:062582fca9fabdd2c8b54a3ef1c978d786e0f6b3a1510e0ac93ef59e0ddae2bc"}, + {file = "PyYAML-6.0.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d2b04aac4d386b172d5b9692e2d2da8de7bfb6c387fa4f801fbf6fb2e6ba4673"}, + {file = "PyYAML-6.0.1-cp311-cp311-win32.whl", hash = "sha256:1635fd110e8d85d55237ab316b5b011de701ea0f29d07611174a1b42f1444741"}, + {file = "PyYAML-6.0.1-cp311-cp311-win_amd64.whl", hash = "sha256:bf07ee2fef7014951eeb99f56f39c9bb4af143d8aa3c21b1677805985307da34"}, + {file = "PyYAML-6.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:50550eb667afee136e9a77d6dc71ae76a44df8b3e51e41b77f6de2932bfe0f47"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1fe35611261b29bd1de0070f0b2f47cb6ff71fa6595c077e42bd0c419fa27b98"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:704219a11b772aea0d8ecd7058d0082713c3562b4e271b849ad7dc4a5c90c13c"}, + {file = "PyYAML-6.0.1-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:afd7e57eddb1a54f0f1a974bc4391af8bcce0b444685d936840f125cf046d5bd"}, + {file = "PyYAML-6.0.1-cp36-cp36m-win32.whl", hash = "sha256:fca0e3a251908a499833aa292323f32437106001d436eca0e6e7833256674585"}, + {file = "PyYAML-6.0.1-cp36-cp36m-win_amd64.whl", hash = "sha256:f22ac1c3cac4dbc50079e965eba2c1058622631e526bd9afd45fedd49ba781fa"}, + {file = "PyYAML-6.0.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:b1275ad35a5d18c62a7220633c913e1b42d44b46ee12554e5fd39c70a243d6a3"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:18aeb1bf9a78867dc38b259769503436b7c72f7a1f1f4c93ff9a17de54319b27"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:596106435fa6ad000c2991a98fa58eeb8656ef2325d7e158344fb33864ed87e3"}, + {file = "PyYAML-6.0.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:baa90d3f661d43131ca170712d903e6295d1f7a0f595074f151c0aed377c9b9c"}, + {file = "PyYAML-6.0.1-cp37-cp37m-win32.whl", hash = "sha256:9046c58c4395dff28dd494285c82ba00b546adfc7ef001486fbf0324bc174fba"}, + {file = "PyYAML-6.0.1-cp37-cp37m-win_amd64.whl", hash = "sha256:4fb147e7a67ef577a588a0e2c17b6db51dda102c71de36f8549b6816a96e1867"}, + {file = "PyYAML-6.0.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:1d4c7e777c441b20e32f52bd377e0c409713e8bb1386e1099c2415f26e479595"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a0cd17c15d3bb3fa06978b4e8958dcdc6e0174ccea823003a106c7d4d7899ac5"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:28c119d996beec18c05208a8bd78cbe4007878c6dd15091efb73a30e90539696"}, + {file = "PyYAML-6.0.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7e07cbde391ba96ab58e532ff4803f79c4129397514e1413a7dc761ccd755735"}, + {file = "PyYAML-6.0.1-cp38-cp38-win32.whl", hash = "sha256:184c5108a2aca3c5b3d3bf9395d50893a7ab82a38004c8f61c258d4428e80206"}, + {file = "PyYAML-6.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:1e2722cc9fbb45d9b87631ac70924c11d3a401b2d7f410cc0e3bbf249f2dca62"}, + {file = "PyYAML-6.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:9eb6caa9a297fc2c2fb8862bc5370d0303ddba53ba97e71f08023b6cd73d16a8"}, + {file = "PyYAML-6.0.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:c8098ddcc2a85b61647b2590f825f3db38891662cfc2fc776415143f599bb859"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5773183b6446b2c99bb77e77595dd486303b4faab2b086e7b17bc6bef28865f6"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b786eecbdf8499b9ca1d697215862083bd6d2a99965554781d0d8d1ad31e13a0"}, + {file = "PyYAML-6.0.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bc1bf2925a1ecd43da378f4db9e4f799775d6367bdb94671027b73b393a7c42c"}, + {file = "PyYAML-6.0.1-cp39-cp39-win32.whl", hash = "sha256:faca3bdcf85b2fc05d06ff3fbc1f83e1391b3e724afa3feba7d13eeab355484c"}, + {file = "PyYAML-6.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:510c9deebc5c0225e8c96813043e62b680ba2f9c50a08d3724c7f28a747d1486"}, + {file = "PyYAML-6.0.1.tar.gz", hash = "sha256:bfdf460b1736c775f2ba9f6a92bca30bc2095067b8a9d77876d1fad6cc3b4a43"}, +] + +[[package]] +name = "redis" +version = "4.6.0" +description = "Python client for Redis database and key-value store" +optional = false +python-versions = ">=3.7" +files = [ + {file = "redis-4.6.0-py3-none-any.whl", hash = "sha256:e2b03db868160ee4591de3cb90d40ebb50a90dd302138775937f6a42b7ed183c"}, + {file = "redis-4.6.0.tar.gz", hash = "sha256:585dc516b9eb042a619ef0a39c3d7d55fe81bdb4df09a52c9cdde0d07bf1aa7d"}, +] + +[package.dependencies] +async-timeout = {version = ">=4.0.2", markers = "python_full_version <= \"3.11.2\""} + +[package.extras] +hiredis = ["hiredis (>=1.0.0)"] +ocsp = ["cryptography (>=36.0.1)", "pyopenssl (==20.0.1)", "requests (>=2.26.0)"] + +[[package]] +name = "regex" +version = "2024.4.16" +description = "Alternative regular expression module, to replace re." +optional = false +python-versions = ">=3.7" +files = [ + {file = "regex-2024.4.16-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:fb83cc090eac63c006871fd24db5e30a1f282faa46328572661c0a24a2323a08"}, + {file = "regex-2024.4.16-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8c91e1763696c0eb66340c4df98623c2d4e77d0746b8f8f2bee2c6883fd1fe18"}, + {file = "regex-2024.4.16-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:10188fe732dec829c7acca7422cdd1bf57d853c7199d5a9e96bb4d40db239c73"}, + {file = "regex-2024.4.16-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:956b58d692f235cfbf5b4f3abd6d99bf102f161ccfe20d2fd0904f51c72c4c66"}, + {file = "regex-2024.4.16-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a70b51f55fd954d1f194271695821dd62054d949efd6368d8be64edd37f55c86"}, + {file = "regex-2024.4.16-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5c02fcd2bf45162280613d2e4a1ca3ac558ff921ae4e308ecb307650d3a6ee51"}, + {file = "regex-2024.4.16-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c4ed75ea6892a56896d78f11006161eea52c45a14994794bcfa1654430984b22"}, + {file = "regex-2024.4.16-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:bd727ad276bb91928879f3aa6396c9a1d34e5e180dce40578421a691eeb77f47"}, + {file = "regex-2024.4.16-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:7cbc5d9e8a1781e7be17da67b92580d6ce4dcef5819c1b1b89f49d9678cc278c"}, + {file = "regex-2024.4.16-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:78fddb22b9ef810b63ef341c9fcf6455232d97cfe03938cbc29e2672c436670e"}, + {file = "regex-2024.4.16-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:445ca8d3c5a01309633a0c9db57150312a181146315693273e35d936472df912"}, + {file = "regex-2024.4.16-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:95399831a206211d6bc40224af1c635cb8790ddd5c7493e0bd03b85711076a53"}, + {file = "regex-2024.4.16-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:7731728b6568fc286d86745f27f07266de49603a6fdc4d19c87e8c247be452af"}, + {file = "regex-2024.4.16-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:4facc913e10bdba42ec0aee76d029aedda628161a7ce4116b16680a0413f658a"}, + {file = "regex-2024.4.16-cp310-cp310-win32.whl", hash = "sha256:911742856ce98d879acbea33fcc03c1d8dc1106234c5e7d068932c945db209c0"}, + {file = "regex-2024.4.16-cp310-cp310-win_amd64.whl", hash = "sha256:e0a2df336d1135a0b3a67f3bbf78a75f69562c1199ed9935372b82215cddd6e2"}, + {file = "regex-2024.4.16-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:1210365faba7c2150451eb78ec5687871c796b0f1fa701bfd2a4a25420482d26"}, + {file = "regex-2024.4.16-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:9ab40412f8cd6f615bfedea40c8bf0407d41bf83b96f6fc9ff34976d6b7037fd"}, + {file = "regex-2024.4.16-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:fd80d1280d473500d8086d104962a82d77bfbf2b118053824b7be28cd5a79ea5"}, + {file = "regex-2024.4.16-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7bb966fdd9217e53abf824f437a5a2d643a38d4fd5fd0ca711b9da683d452969"}, + {file = "regex-2024.4.16-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:20b7a68444f536365af42a75ccecb7ab41a896a04acf58432db9e206f4e525d6"}, + {file = "regex-2024.4.16-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b74586dd0b039c62416034f811d7ee62810174bb70dffcca6439f5236249eb09"}, + {file = "regex-2024.4.16-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0c8290b44d8b0af4e77048646c10c6e3aa583c1ca67f3b5ffb6e06cf0c6f0f89"}, + {file = "regex-2024.4.16-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f2d80a6749724b37853ece57988b39c4e79d2b5fe2869a86e8aeae3bbeef9eb0"}, + {file = "regex-2024.4.16-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:3a1018e97aeb24e4f939afcd88211ace472ba566efc5bdf53fd8fd7f41fa7170"}, + {file = "regex-2024.4.16-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:8d015604ee6204e76569d2f44e5a210728fa917115bef0d102f4107e622b08d5"}, + {file = "regex-2024.4.16-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:3d5ac5234fb5053850d79dd8eb1015cb0d7d9ed951fa37aa9e6249a19aa4f336"}, + {file = "regex-2024.4.16-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:0a38d151e2cdd66d16dab550c22f9521ba79761423b87c01dae0a6e9add79c0d"}, + {file = "regex-2024.4.16-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:159dc4e59a159cb8e4e8f8961eb1fa5d58f93cb1acd1701d8aff38d45e1a84a6"}, + {file = "regex-2024.4.16-cp311-cp311-win32.whl", hash = "sha256:ba2336d6548dee3117520545cfe44dc28a250aa091f8281d28804aa8d707d93d"}, + {file = "regex-2024.4.16-cp311-cp311-win_amd64.whl", hash = "sha256:8f83b6fd3dc3ba94d2b22717f9c8b8512354fd95221ac661784df2769ea9bba9"}, + {file = "regex-2024.4.16-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:80b696e8972b81edf0af2a259e1b2a4a661f818fae22e5fa4fa1a995fb4a40fd"}, + {file = "regex-2024.4.16-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:d61ae114d2a2311f61d90c2ef1358518e8f05eafda76eaf9c772a077e0b465ec"}, + {file = "regex-2024.4.16-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:8ba6745440b9a27336443b0c285d705ce73adb9ec90e2f2004c64d95ab5a7598"}, + {file = "regex-2024.4.16-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6295004b2dd37b0835ea5c14a33e00e8cfa3c4add4d587b77287825f3418d310"}, + {file = "regex-2024.4.16-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4aba818dcc7263852aabb172ec27b71d2abca02a593b95fa79351b2774eb1d2b"}, + {file = "regex-2024.4.16-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d0800631e565c47520aaa04ae38b96abc5196fe8b4aa9bd864445bd2b5848a7a"}, + {file = "regex-2024.4.16-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:08dea89f859c3df48a440dbdcd7b7155bc675f2fa2ec8c521d02dc69e877db70"}, + {file = "regex-2024.4.16-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:eeaa0b5328b785abc344acc6241cffde50dc394a0644a968add75fcefe15b9d4"}, + {file = "regex-2024.4.16-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:4e819a806420bc010489f4e741b3036071aba209f2e0989d4750b08b12a9343f"}, + {file = "regex-2024.4.16-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:c2d0e7cbb6341e830adcbfa2479fdeebbfbb328f11edd6b5675674e7a1e37730"}, + {file = "regex-2024.4.16-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:91797b98f5e34b6a49f54be33f72e2fb658018ae532be2f79f7c63b4ae225145"}, + {file = "regex-2024.4.16-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:d2da13568eff02b30fd54fccd1e042a70fe920d816616fda4bf54ec705668d81"}, + {file = "regex-2024.4.16-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:370c68dc5570b394cbaadff50e64d705f64debed30573e5c313c360689b6aadc"}, + {file = "regex-2024.4.16-cp312-cp312-win32.whl", hash = "sha256:904c883cf10a975b02ab3478bce652f0f5346a2c28d0a8521d97bb23c323cc8b"}, + {file = "regex-2024.4.16-cp312-cp312-win_amd64.whl", hash = "sha256:785c071c982dce54d44ea0b79cd6dfafddeccdd98cfa5f7b86ef69b381b457d9"}, + {file = "regex-2024.4.16-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:e2f142b45c6fed48166faeb4303b4b58c9fcd827da63f4cf0a123c3480ae11fb"}, + {file = "regex-2024.4.16-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e87ab229332ceb127a165612d839ab87795972102cb9830e5f12b8c9a5c1b508"}, + {file = "regex-2024.4.16-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:81500ed5af2090b4a9157a59dbc89873a25c33db1bb9a8cf123837dcc9765047"}, + {file = "regex-2024.4.16-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b340cccad138ecb363324aa26893963dcabb02bb25e440ebdf42e30963f1a4e0"}, + {file = "regex-2024.4.16-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2c72608e70f053643437bd2be0608f7f1c46d4022e4104d76826f0839199347a"}, + {file = "regex-2024.4.16-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a01fe2305e6232ef3e8f40bfc0f0f3a04def9aab514910fa4203bafbc0bb4682"}, + {file = "regex-2024.4.16-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:03576e3a423d19dda13e55598f0fd507b5d660d42c51b02df4e0d97824fdcae3"}, + {file = "regex-2024.4.16-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:549c3584993772e25f02d0656ac48abdda73169fe347263948cf2b1cead622f3"}, + {file = "regex-2024.4.16-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:34422d5a69a60b7e9a07a690094e824b66f5ddc662a5fc600d65b7c174a05f04"}, + {file = "regex-2024.4.16-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:5f580c651a72b75c39e311343fe6875d6f58cf51c471a97f15a938d9fe4e0d37"}, + {file = "regex-2024.4.16-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:3399dd8a7495bbb2bacd59b84840eef9057826c664472e86c91d675d007137f5"}, + {file = "regex-2024.4.16-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:8d1f86f3f4e2388aa3310b50694ac44daefbd1681def26b4519bd050a398dc5a"}, + {file = "regex-2024.4.16-cp37-cp37m-win32.whl", hash = "sha256:dd5acc0a7d38fdc7a3a6fd3ad14c880819008ecb3379626e56b163165162cc46"}, + {file = "regex-2024.4.16-cp37-cp37m-win_amd64.whl", hash = "sha256:ba8122e3bb94ecda29a8de4cf889f600171424ea586847aa92c334772d200331"}, + {file = "regex-2024.4.16-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:743deffdf3b3481da32e8a96887e2aa945ec6685af1cfe2bcc292638c9ba2f48"}, + {file = "regex-2024.4.16-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:7571f19f4a3fd00af9341c7801d1ad1967fc9c3f5e62402683047e7166b9f2b4"}, + {file = "regex-2024.4.16-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:df79012ebf6f4efb8d307b1328226aef24ca446b3ff8d0e30202d7ebcb977a8c"}, + {file = "regex-2024.4.16-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e757d475953269fbf4b441207bb7dbdd1c43180711b6208e129b637792ac0b93"}, + {file = "regex-2024.4.16-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4313ab9bf6a81206c8ac28fdfcddc0435299dc88cad12cc6305fd0e78b81f9e4"}, + {file = "regex-2024.4.16-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:d83c2bc678453646f1a18f8db1e927a2d3f4935031b9ad8a76e56760461105dd"}, + {file = "regex-2024.4.16-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9df1bfef97db938469ef0a7354b2d591a2d438bc497b2c489471bec0e6baf7c4"}, + {file = "regex-2024.4.16-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:62120ed0de69b3649cc68e2965376048793f466c5a6c4370fb27c16c1beac22d"}, + {file = "regex-2024.4.16-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:c2ef6f7990b6e8758fe48ad08f7e2f66c8f11dc66e24093304b87cae9037bb4a"}, + {file = "regex-2024.4.16-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:8fc6976a3395fe4d1fbeb984adaa8ec652a1e12f36b56ec8c236e5117b585427"}, + {file = "regex-2024.4.16-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:03e68f44340528111067cecf12721c3df4811c67268b897fbe695c95f860ac42"}, + {file = "regex-2024.4.16-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:ec7e0043b91115f427998febaa2beb82c82df708168b35ece3accb610b91fac1"}, + {file = "regex-2024.4.16-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:c21fc21a4c7480479d12fd8e679b699f744f76bb05f53a1d14182b31f55aac76"}, + {file = "regex-2024.4.16-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:12f6a3f2f58bb7344751919a1876ee1b976fe08b9ffccb4bbea66f26af6017b9"}, + {file = "regex-2024.4.16-cp38-cp38-win32.whl", hash = "sha256:479595a4fbe9ed8f8f72c59717e8cf222da2e4c07b6ae5b65411e6302af9708e"}, + {file = "regex-2024.4.16-cp38-cp38-win_amd64.whl", hash = "sha256:0534b034fba6101611968fae8e856c1698da97ce2efb5c2b895fc8b9e23a5834"}, + {file = "regex-2024.4.16-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:a7ccdd1c4a3472a7533b0a7aa9ee34c9a2bef859ba86deec07aff2ad7e0c3b94"}, + {file = "regex-2024.4.16-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6f2f017c5be19984fbbf55f8af6caba25e62c71293213f044da3ada7091a4455"}, + {file = "regex-2024.4.16-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:803b8905b52de78b173d3c1e83df0efb929621e7b7c5766c0843704d5332682f"}, + {file = "regex-2024.4.16-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:684008ec44ad275832a5a152f6e764bbe1914bea10968017b6feaecdad5736e0"}, + {file = "regex-2024.4.16-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:65436dce9fdc0aeeb0a0effe0839cb3d6a05f45aa45a4d9f9c60989beca78b9c"}, + {file = "regex-2024.4.16-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ea355eb43b11764cf799dda62c658c4d2fdb16af41f59bb1ccfec517b60bcb07"}, + {file = "regex-2024.4.16-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:98c1165f3809ce7774f05cb74e5408cd3aa93ee8573ae959a97a53db3ca3180d"}, + {file = "regex-2024.4.16-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:cccc79a9be9b64c881f18305a7c715ba199e471a3973faeb7ba84172abb3f317"}, + {file = "regex-2024.4.16-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:00169caa125f35d1bca6045d65a662af0202704489fada95346cfa092ec23f39"}, + {file = "regex-2024.4.16-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:6cc38067209354e16c5609b66285af17a2863a47585bcf75285cab33d4c3b8df"}, + {file = "regex-2024.4.16-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:23cff1b267038501b179ccbbd74a821ac4a7192a1852d1d558e562b507d46013"}, + {file = "regex-2024.4.16-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:b9d320b3bf82a39f248769fc7f188e00f93526cc0fe739cfa197868633d44701"}, + {file = "regex-2024.4.16-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:89ec7f2c08937421bbbb8b48c54096fa4f88347946d4747021ad85f1b3021b3c"}, + {file = "regex-2024.4.16-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:4918fd5f8b43aa7ec031e0fef1ee02deb80b6afd49c85f0790be1dc4ce34cb50"}, + {file = "regex-2024.4.16-cp39-cp39-win32.whl", hash = "sha256:684e52023aec43bdf0250e843e1fdd6febbe831bd9d52da72333fa201aaa2335"}, + {file = "regex-2024.4.16-cp39-cp39-win_amd64.whl", hash = "sha256:e697e1c0238133589e00c244a8b676bc2cfc3ab4961318d902040d099fec7483"}, + {file = "regex-2024.4.16.tar.gz", hash = "sha256:fa454d26f2e87ad661c4f0c5a5fe4cf6aab1e307d1b94f16ffdfcb089ba685c0"}, +] + +[[package]] +name = "requests" +version = "2.31.0" +description = "Python HTTP for Humans." +optional = false +python-versions = ">=3.7" +files = [ + {file = "requests-2.31.0-py3-none-any.whl", hash = "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f"}, + {file = "requests-2.31.0.tar.gz", hash = "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1"}, +] + +[package.dependencies] +certifi = ">=2017.4.17" +charset-normalizer = ">=2,<4" +idna = ">=2.5,<4" +urllib3 = ">=1.21.1,<3" + +[package.extras] +socks = ["PySocks (>=1.5.6,!=1.5.7)"] +use-chardet-on-py3 = ["chardet (>=3.0.2,<6)"] + +[[package]] +name = "setuptools" +version = "69.5.1" +description = "Easily download, build, install, upgrade, and uninstall Python packages" +optional = false +python-versions = ">=3.8" +files = [ + {file = "setuptools-69.5.1-py3-none-any.whl", hash = "sha256:c636ac361bc47580504644275c9ad802c50415c7522212252c033bd15f301f32"}, + {file = "setuptools-69.5.1.tar.gz", hash = "sha256:6c1fccdac05a97e598fb0ae3bbed5904ccb317337a51139dcd51453611bbb987"}, +] + +[package.extras] +docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"] +testing = ["build[virtualenv]", "filelock (>=3.4.0)", "importlib-metadata", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "mypy (==1.9)", "packaging (>=23.2)", "pip (>=19.1)", "pytest (>=6,!=8.1.1)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-timeout", "pytest-xdist (>=3)", "tomli", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"] +testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.2)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"] + +[[package]] +name = "six" +version = "1.16.0" +description = "Python 2 and 3 compatibility utilities" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" +files = [ + {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"}, + {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, +] + +[[package]] +name = "sniffio" +version = "1.3.1" +description = "Sniff out which async library your code is running under" +optional = false +python-versions = ">=3.7" +files = [ + {file = "sniffio-1.3.1-py3-none-any.whl", hash = "sha256:2f6da418d1f1e0fddd844478f41680e794e6051915791a034ff65e5f100525a2"}, + {file = "sniffio-1.3.1.tar.gz", hash = "sha256:f4324edc670a0f49750a81b895f35c3adb843cca46f0530f79fc1babb23789dc"}, +] + +[[package]] +name = "starlette" +version = "0.27.0" +description = "The little ASGI library that shines." +optional = false +python-versions = ">=3.7" +files = [ + {file = "starlette-0.27.0-py3-none-any.whl", hash = "sha256:918416370e846586541235ccd38a474c08b80443ed31c578a418e2209b3eef91"}, + {file = "starlette-0.27.0.tar.gz", hash = "sha256:6a6b0d042acb8d469a01eba54e9cda6cbd24ac602c4cd016723117d6a7e73b75"}, +] + +[package.dependencies] +anyio = ">=3.4.0,<5" + +[package.extras] +full = ["httpx (>=0.22.0)", "itsdangerous", "jinja2", "python-multipart", "pyyaml"] + +[[package]] +name = "sympy" +version = "1.12" +description = "Computer algebra system (CAS) in Python" +optional = false +python-versions = ">=3.8" +files = [ + {file = "sympy-1.12-py3-none-any.whl", hash = "sha256:c3588cd4295d0c0f603d0f2ae780587e64e2efeedb3521e46b9bb1d08d184fa5"}, + {file = "sympy-1.12.tar.gz", hash = "sha256:ebf595c8dac3e0fdc4152c51878b498396ec7f30e7a914d6071e674d49420fb8"}, +] + +[package.dependencies] +mpmath = ">=0.19" + +[[package]] +name = "tiktoken" +version = "0.3.3" +description = "tiktoken is a fast BPE tokeniser for use with OpenAI's models" +optional = false +python-versions = ">=3.8" +files = [ + {file = "tiktoken-0.3.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d1f37fa75ba70c1bc7806641e8ccea1fba667d23e6341a1591ea333914c226a9"}, + {file = "tiktoken-0.3.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3d7296c38392a943c2ccc0b61323086b8550cef08dcf6855de9949890dbc1fd3"}, + {file = "tiktoken-0.3.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3c84491965e139a905280ac28b74baaa13445b3678e07f96767089ad1ef5ee7b"}, + {file = "tiktoken-0.3.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:65970d77ea85ce6c7fce45131da9258cd58a802ffb29ead8f5552e331c025b2b"}, + {file = "tiktoken-0.3.3-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:bd3f72d0ba7312c25c1652292121a24c8f1711207b63c6d8dab21afe4be0bf04"}, + {file = "tiktoken-0.3.3-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:719c9e13432602dc496b24f13e3c3ad3ec0d2fbdb9aace84abfb95e9c3a425a4"}, + {file = "tiktoken-0.3.3-cp310-cp310-win_amd64.whl", hash = "sha256:dc00772284c94e65045b984ed7e9f95d000034f6b2411df252011b069bd36217"}, + {file = "tiktoken-0.3.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4db2c40f79f8f7a21a9fdbf1c6dee32dea77b0d7402355dc584a3083251d2e15"}, + {file = "tiktoken-0.3.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e3c0f2231aa3829a1a431a882201dc27858634fd9989898e0f7d991dbc6bcc9d"}, + {file = "tiktoken-0.3.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:48c13186a479de16cfa2c72bb0631fa9c518350a5b7569e4d77590f7fee96be9"}, + {file = "tiktoken-0.3.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6674e4e37ab225020135cd66a392589623d5164c6456ba28cc27505abed10d9e"}, + {file = "tiktoken-0.3.3-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:4a0c1357f6191211c544f935d5aa3cb9d7abd118c8f3c7124196d5ecd029b4af"}, + {file = "tiktoken-0.3.3-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:2e948d167fc3b04483cbc33426766fd742e7cefe5346cd62b0cbd7279ef59539"}, + {file = "tiktoken-0.3.3-cp311-cp311-win_amd64.whl", hash = "sha256:5dca434c8680b987eacde2dbc449e9ea4526574dbf9f3d8938665f638095be82"}, + {file = "tiktoken-0.3.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:984758ebc07cd8c557345697c234f1f221bd730b388f4340dd08dffa50213a01"}, + {file = "tiktoken-0.3.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:891012f29e159a989541ae47259234fb29ff88c22e1097567316e27ad33a3734"}, + {file = "tiktoken-0.3.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:210f8602228e4c5d706deeb389da5a152b214966a5aa558eec87b57a1969ced5"}, + {file = "tiktoken-0.3.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd783564f80d4dc44ff0a64b13756ded8390ed2548549aefadbe156af9188307"}, + {file = "tiktoken-0.3.3-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:03f64bde9b4eb8338bf49c8532bfb4c3578f6a9a6979fc176d939f9e6f68b408"}, + {file = "tiktoken-0.3.3-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:1ac369367b6f5e5bd80e8f9a7766ac2a9c65eda2aa856d5f3c556d924ff82986"}, + {file = "tiktoken-0.3.3-cp38-cp38-win_amd64.whl", hash = "sha256:94600798891f78db780e5aa9321456cf355e54a4719fbd554147a628de1f163f"}, + {file = "tiktoken-0.3.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:e59db6fca8d5ccea302fe2888917364446d6f4201a25272a1a1c44975c65406a"}, + {file = "tiktoken-0.3.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:19340d8ba4d6fd729b2e3a096a547ded85f71012843008f97475f9db484869ee"}, + {file = "tiktoken-0.3.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:542686cbc9225540e3a10f472f82fa2e1bebafce2233a211dee8459e95821cfd"}, + {file = "tiktoken-0.3.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3a43612b2a09f4787c050163a216bf51123851859e9ab128ad03d2729826cde9"}, + {file = "tiktoken-0.3.3-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:a11674f0275fa75fb59941b703650998bd4acb295adbd16fc8af17051aaed19d"}, + {file = "tiktoken-0.3.3-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:65fc0a449630bab28c30b4adec257442a4706d79cffc2337c1d9df3e91825cdd"}, + {file = "tiktoken-0.3.3-cp39-cp39-win_amd64.whl", hash = "sha256:0b9a7a9a8b781a50ee9289e85e28771d7e113cc0c656eadfb6fc6d3a106ff9bb"}, + {file = "tiktoken-0.3.3.tar.gz", hash = "sha256:97b58b7bfda945791ec855e53d166e8ec20c6378942b93851a6c919ddf9d0496"}, +] + +[package.dependencies] +regex = ">=2022.1.18" +requests = ">=2.26.0" + +[package.extras] +blobfile = ["blobfile (>=2)"] + +[[package]] +name = "tokenizers" +version = "0.13.3" +description = "Fast and Customizable Tokenizers" +optional = false +python-versions = "*" +files = [ + {file = "tokenizers-0.13.3-cp310-cp310-macosx_10_11_x86_64.whl", hash = "sha256:f3835c5be51de8c0a092058a4d4380cb9244fb34681fd0a295fbf0a52a5fdf33"}, + {file = "tokenizers-0.13.3-cp310-cp310-macosx_12_0_arm64.whl", hash = "sha256:4ef4c3e821730f2692489e926b184321e887f34fb8a6b80b8096b966ba663d07"}, + {file = "tokenizers-0.13.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c5fd1a6a25353e9aa762e2aae5a1e63883cad9f4e997c447ec39d071020459bc"}, + {file = "tokenizers-0.13.3-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ee0b1b311d65beab83d7a41c56a1e46ab732a9eed4460648e8eb0bd69fc2d059"}, + {file = "tokenizers-0.13.3-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5ef4215284df1277dadbcc5e17d4882bda19f770d02348e73523f7e7d8b8d396"}, + {file = "tokenizers-0.13.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a4d53976079cff8a033f778fb9adca2d9d69d009c02fa2d71a878b5f3963ed30"}, + {file = "tokenizers-0.13.3-cp310-cp310-win32.whl", hash = "sha256:1f0e3b4c2ea2cd13238ce43548959c118069db7579e5d40ec270ad77da5833ce"}, + {file = "tokenizers-0.13.3-cp310-cp310-win_amd64.whl", hash = "sha256:89649c00d0d7211e8186f7a75dfa1db6996f65edce4b84821817eadcc2d3c79e"}, + {file = "tokenizers-0.13.3-cp311-cp311-macosx_10_11_universal2.whl", hash = "sha256:56b726e0d2bbc9243872b0144515ba684af5b8d8cd112fb83ee1365e26ec74c8"}, + {file = "tokenizers-0.13.3-cp311-cp311-macosx_12_0_arm64.whl", hash = "sha256:cc5c022ce692e1f499d745af293ab9ee6f5d92538ed2faf73f9708c89ee59ce6"}, + {file = "tokenizers-0.13.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f55c981ac44ba87c93e847c333e58c12abcbb377a0c2f2ef96e1a266e4184ff2"}, + {file = "tokenizers-0.13.3-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f247eae99800ef821a91f47c5280e9e9afaeed9980fc444208d5aa6ba69ff148"}, + {file = "tokenizers-0.13.3-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:4b3e3215d048e94f40f1c95802e45dcc37c5b05eb46280fc2ccc8cd351bff839"}, + {file = "tokenizers-0.13.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9ba2b0bf01777c9b9bc94b53764d6684554ce98551fec496f71bc5be3a03e98b"}, + {file = "tokenizers-0.13.3-cp311-cp311-win32.whl", hash = "sha256:cc78d77f597d1c458bf0ea7c2a64b6aa06941c7a99cb135b5969b0278824d808"}, + {file = "tokenizers-0.13.3-cp311-cp311-win_amd64.whl", hash = "sha256:ecf182bf59bd541a8876deccf0360f5ae60496fd50b58510048020751cf1724c"}, + {file = "tokenizers-0.13.3-cp37-cp37m-macosx_10_11_x86_64.whl", hash = "sha256:0527dc5436a1f6bf2c0327da3145687d3bcfbeab91fed8458920093de3901b44"}, + {file = "tokenizers-0.13.3-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:07cbb2c307627dc99b44b22ef05ff4473aa7c7cc1fec8f0a8b37d8a64b1a16d2"}, + {file = "tokenizers-0.13.3-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4560dbdeaae5b7ee0d4e493027e3de6d53c991b5002d7ff95083c99e11dd5ac0"}, + {file = "tokenizers-0.13.3-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:64064bd0322405c9374305ab9b4c07152a1474370327499911937fd4a76d004b"}, + {file = "tokenizers-0.13.3-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8c6e2ab0f2e3d939ca66aa1d596602105fe33b505cd2854a4c1717f704c51de"}, + {file = "tokenizers-0.13.3-cp37-cp37m-win32.whl", hash = "sha256:6cc29d410768f960db8677221e497226e545eaaea01aa3613fa0fdf2cc96cff4"}, + {file = "tokenizers-0.13.3-cp37-cp37m-win_amd64.whl", hash = "sha256:fc2a7fdf864554a0dacf09d32e17c0caa9afe72baf9dd7ddedc61973bae352d8"}, + {file = "tokenizers-0.13.3-cp38-cp38-macosx_10_11_x86_64.whl", hash = "sha256:8791dedba834c1fc55e5f1521be325ea3dafb381964be20684b92fdac95d79b7"}, + {file = "tokenizers-0.13.3-cp38-cp38-macosx_12_0_arm64.whl", hash = "sha256:d607a6a13718aeb20507bdf2b96162ead5145bbbfa26788d6b833f98b31b26e1"}, + {file = "tokenizers-0.13.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3791338f809cd1bf8e4fee6b540b36822434d0c6c6bc47162448deee3f77d425"}, + {file = "tokenizers-0.13.3-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:c2f35f30e39e6aab8716f07790f646bdc6e4a853816cc49a95ef2a9016bf9ce6"}, + {file = "tokenizers-0.13.3-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:310204dfed5aa797128b65d63538a9837cbdd15da2a29a77d67eefa489edda26"}, + {file = "tokenizers-0.13.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a0f9b92ea052305166559f38498b3b0cae159caea712646648aaa272f7160963"}, + {file = "tokenizers-0.13.3-cp38-cp38-win32.whl", hash = "sha256:9a3fa134896c3c1f0da6e762d15141fbff30d094067c8f1157b9fdca593b5806"}, + {file = "tokenizers-0.13.3-cp38-cp38-win_amd64.whl", hash = "sha256:8e7b0cdeace87fa9e760e6a605e0ae8fc14b7d72e9fc19c578116f7287bb873d"}, + {file = "tokenizers-0.13.3-cp39-cp39-macosx_10_11_x86_64.whl", hash = "sha256:00cee1e0859d55507e693a48fa4aef07060c4bb6bd93d80120e18fea9371c66d"}, + {file = "tokenizers-0.13.3-cp39-cp39-macosx_12_0_arm64.whl", hash = "sha256:a23ff602d0797cea1d0506ce69b27523b07e70f6dda982ab8cf82402de839088"}, + {file = "tokenizers-0.13.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:70ce07445050b537d2696022dafb115307abdffd2a5c106f029490f84501ef97"}, + {file = "tokenizers-0.13.3-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:280ffe95f50eaaf655b3a1dc7ff1d9cf4777029dbbc3e63a74e65a056594abc3"}, + {file = "tokenizers-0.13.3-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:97acfcec592f7e9de8cadcdcda50a7134423ac8455c0166b28c9ff04d227b371"}, + {file = "tokenizers-0.13.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd7730c98a3010cd4f523465867ff95cd9d6430db46676ce79358f65ae39797b"}, + {file = "tokenizers-0.13.3-cp39-cp39-win32.whl", hash = "sha256:48625a108029cb1ddf42e17a81b5a3230ba6888a70c9dc14e81bc319e812652d"}, + {file = "tokenizers-0.13.3-cp39-cp39-win_amd64.whl", hash = "sha256:bc0a6f1ba036e482db6453571c9e3e60ecd5489980ffd95d11dc9f960483d783"}, + {file = "tokenizers-0.13.3.tar.gz", hash = "sha256:2e546dbb68b623008a5442353137fbb0123d311a6d7ba52f2667c8862a75af2e"}, +] + +[package.extras] +dev = ["black (==22.3)", "datasets", "numpy", "pytest", "requests"] +docs = ["setuptools-rust", "sphinx", "sphinx-rtd-theme"] +testing = ["black (==22.3)", "datasets", "numpy", "pytest", "requests"] + +[[package]] +name = "toml" +version = "0.10.2" +description = "Python Library for Tom's Obvious, Minimal Language" +optional = false +python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" +files = [ + {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"}, + {file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"}, +] + +[[package]] +name = "torch" +version = "1.13.1" +description = "Tensors and Dynamic neural networks in Python with strong GPU acceleration" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "torch-1.13.1-cp310-none-macosx_10_9_x86_64.whl", hash = "sha256:393a6273c832e047581063fb74335ff50b4c566217019cc6ace318cd79eb0566"}, +] + +[package.dependencies] +typing-extensions = "*" + +[package.extras] +opt-einsum = ["opt-einsum (>=3.3)"] + +[package.source] +type = "url" +url = "https://download.pytorch.org/whl/cpu/torch-1.13.1-cp310-none-macosx_10_9_x86_64.whl" + +[[package]] +name = "torch" +version = "1.13.1" +description = "Tensors and Dynamic neural networks in Python with strong GPU acceleration" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "torch-1.13.1-cp310-none-macosx_11_0_arm64.whl", hash = "sha256:0122806b111b949d21fa1a5f9764d1fd2fcc4a47cb7f8ff914204fd4fc752ed5"}, +] + +[package.dependencies] +typing-extensions = "*" + +[package.extras] +opt-einsum = ["opt-einsum (>=3.3)"] + +[package.source] +type = "url" +url = "https://download.pytorch.org/whl/cpu/torch-1.13.1-cp310-none-macosx_11_0_arm64.whl" + +[[package]] +name = "torch" +version = "1.13.1" +description = "Tensors and Dynamic neural networks in Python with strong GPU acceleration" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "torch-1.13.1-cp310-cp310-manylinux2014_aarch64.whl", hash = "sha256:d9fe785d375f2e26a5d5eba5de91f89e6a3be5d11efb497e76705fdf93fa3c2e"}, +] + +[package.dependencies] +typing-extensions = "*" + +[package.extras] +opt-einsum = ["opt-einsum (>=3.3)"] + +[package.source] +type = "url" +url = "https://download.pytorch.org/whl/torch-1.13.1-cp310-cp310-manylinux2014_aarch64.whl" + +[[package]] +name = "torch" +version = "1.13.1+cpu" +description = "Tensors and Dynamic neural networks in Python with strong GPU acceleration" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "torch-1.13.1+cpu-cp310-cp310-linux_x86_64.whl", hash = "sha256:11692523b87c45b79ddfb5148b12a713d85235d399915490d94e079521f7e014"}, +] + +[package.dependencies] +typing-extensions = "*" + +[package.extras] +opt-einsum = ["opt-einsum (>=3.3)"] + +[package.source] +type = "url" +url = "https://download.pytorch.org/whl/cpu/torch-1.13.1%2Bcpu-cp310-cp310-linux_x86_64.whl" + +[[package]] +name = "torch" +version = "1.13.1+cpu" +description = "Tensors and Dynamic neural networks in Python with strong GPU acceleration" +optional = false +python-versions = ">=3.7.0" +files = [ + {file = "torch-1.13.1+cpu-cp310-cp310-win_amd64.whl", hash = "sha256:207ab3700cd9c4349f4fd1892597eb3d385eb78221c0f2974ec54b8ea903aa00"}, +] + +[package.dependencies] +typing-extensions = "*" + +[package.extras] +opt-einsum = ["opt-einsum (>=3.3)"] + +[package.source] +type = "url" +url = "https://download.pytorch.org/whl/cpu/torch-1.13.1%2Bcpu-cp310-cp310-win_amd64.whl" + +[[package]] +name = "tqdm" +version = "4.66.2" +description = "Fast, Extensible Progress Meter" +optional = false +python-versions = ">=3.7" +files = [ + {file = "tqdm-4.66.2-py3-none-any.whl", hash = "sha256:1ee4f8a893eb9bef51c6e35730cebf234d5d0b6bd112b0271e10ed7c24a02bd9"}, + {file = "tqdm-4.66.2.tar.gz", hash = "sha256:6cd52cdf0fef0e0f543299cfc96fec90d7b8a7e88745f411ec33eb44d5ed3531"}, +] + +[package.dependencies] +colorama = {version = "*", markers = "platform_system == \"Windows\""} + +[package.extras] +dev = ["pytest (>=6)", "pytest-cov", "pytest-timeout", "pytest-xdist"] +notebook = ["ipywidgets (>=6)"] +slack = ["slack-sdk"] +telegram = ["requests"] + +[[package]] +name = "typing-extensions" +version = "4.11.0" +description = "Backported and Experimental Type Hints for Python 3.8+" +optional = false +python-versions = ">=3.8" +files = [ + {file = "typing_extensions-4.11.0-py3-none-any.whl", hash = "sha256:c1f94d72897edaf4ce775bb7558d5b79d8126906a14ea5ed1635921406c0387a"}, + {file = "typing_extensions-4.11.0.tar.gz", hash = "sha256:83f085bd5ca59c80295fc2a82ab5dac679cbe02b9f33f7d83af68e241bea51b0"}, +] + +[[package]] +name = "tzdata" +version = "2024.1" +description = "Provider of IANA time zone data" +optional = false +python-versions = ">=2" +files = [ + {file = "tzdata-2024.1-py2.py3-none-any.whl", hash = "sha256:9068bc196136463f5245e51efda838afa15aaeca9903f49050dfa2679db4d252"}, + {file = "tzdata-2024.1.tar.gz", hash = "sha256:2674120f8d891909751c38abcdfd386ac0a5a1127954fbc332af6b5ceae07efd"}, +] + +[[package]] +name = "unidecode" +version = "1.3.8" +description = "ASCII transliterations of Unicode text" +optional = false +python-versions = ">=3.5" +files = [ + {file = "Unidecode-1.3.8-py3-none-any.whl", hash = "sha256:d130a61ce6696f8148a3bd8fe779c99adeb4b870584eeb9526584e9aa091fd39"}, + {file = "Unidecode-1.3.8.tar.gz", hash = "sha256:cfdb349d46ed3873ece4586b96aa75258726e2fa8ec21d6f00a591d98806c2f4"}, +] + +[[package]] +name = "urllib3" +version = "2.2.1" +description = "HTTP library with thread-safe connection pooling, file post, and more." +optional = false +python-versions = ">=3.8" +files = [ + {file = "urllib3-2.2.1-py3-none-any.whl", hash = "sha256:450b20ec296a467077128bff42b73080516e71b56ff59a60a02bef2232c4fa9d"}, + {file = "urllib3-2.2.1.tar.gz", hash = "sha256:d0570876c61ab9e520d776c38acbbb5b05a776d3f9ff98a5c8fd5162a444cf19"}, +] + +[package.extras] +brotli = ["brotli (>=1.0.9)", "brotlicffi (>=0.8.0)"] +h2 = ["h2 (>=4,<5)"] +socks = ["pysocks (>=1.5.6,!=1.5.7,<2.0)"] +zstd = ["zstandard (>=0.18.0)"] + +[[package]] +name = "uvicorn" +version = "0.18.3" +description = "The lightning-fast ASGI server." +optional = false +python-versions = ">=3.7" +files = [ + {file = "uvicorn-0.18.3-py3-none-any.whl", hash = "sha256:0abd429ebb41e604ed8d2be6c60530de3408f250e8d2d84967d85ba9e86fe3af"}, + {file = "uvicorn-0.18.3.tar.gz", hash = "sha256:9a66e7c42a2a95222f76ec24a4b754c158261c4696e683b9dadc72b590e0311b"}, +] + +[package.dependencies] +click = ">=7.0" +colorama = {version = ">=0.4", optional = true, markers = "sys_platform == \"win32\" and extra == \"standard\""} +h11 = ">=0.8" +httptools = {version = ">=0.4.0", optional = true, markers = "extra == \"standard\""} +python-dotenv = {version = ">=0.13", optional = true, markers = "extra == \"standard\""} +pyyaml = {version = ">=5.1", optional = true, markers = "extra == \"standard\""} +uvloop = {version = ">=0.14.0,<0.15.0 || >0.15.0,<0.15.1 || >0.15.1", optional = true, markers = "(sys_platform != \"win32\" and sys_platform != \"cygwin\") and platform_python_implementation != \"PyPy\" and extra == \"standard\""} +watchfiles = {version = ">=0.13", optional = true, markers = "extra == \"standard\""} +websockets = {version = ">=10.0", optional = true, markers = "extra == \"standard\""} + +[package.extras] +standard = ["colorama (>=0.4)", "httptools (>=0.4.0)", "python-dotenv (>=0.13)", "pyyaml (>=5.1)", "uvloop (>=0.14.0,!=0.15.0,!=0.15.1)", "watchfiles (>=0.13)", "websockets (>=10.0)"] + +[[package]] +name = "uvloop" +version = "0.19.0" +description = "Fast implementation of asyncio event loop on top of libuv" +optional = false +python-versions = ">=3.8.0" +files = [ + {file = "uvloop-0.19.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:de4313d7f575474c8f5a12e163f6d89c0a878bc49219641d49e6f1444369a90e"}, + {file = "uvloop-0.19.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:5588bd21cf1fcf06bded085f37e43ce0e00424197e7c10e77afd4bbefffef428"}, + {file = "uvloop-0.19.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7b1fd71c3843327f3bbc3237bedcdb6504fd50368ab3e04d0410e52ec293f5b8"}, + {file = "uvloop-0.19.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5a05128d315e2912791de6088c34136bfcdd0c7cbc1cf85fd6fd1bb321b7c849"}, + {file = "uvloop-0.19.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:cd81bdc2b8219cb4b2556eea39d2e36bfa375a2dd021404f90a62e44efaaf957"}, + {file = "uvloop-0.19.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:5f17766fb6da94135526273080f3455a112f82570b2ee5daa64d682387fe0dcd"}, + {file = "uvloop-0.19.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:4ce6b0af8f2729a02a5d1575feacb2a94fc7b2e983868b009d51c9a9d2149bef"}, + {file = "uvloop-0.19.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:31e672bb38b45abc4f26e273be83b72a0d28d074d5b370fc4dcf4c4eb15417d2"}, + {file = "uvloop-0.19.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:570fc0ed613883d8d30ee40397b79207eedd2624891692471808a95069a007c1"}, + {file = "uvloop-0.19.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5138821e40b0c3e6c9478643b4660bd44372ae1e16a322b8fc07478f92684e24"}, + {file = "uvloop-0.19.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:91ab01c6cd00e39cde50173ba4ec68a1e578fee9279ba64f5221810a9e786533"}, + {file = "uvloop-0.19.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:47bf3e9312f63684efe283f7342afb414eea4d3011542155c7e625cd799c3b12"}, + {file = "uvloop-0.19.0-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:da8435a3bd498419ee8c13c34b89b5005130a476bda1d6ca8cfdde3de35cd650"}, + {file = "uvloop-0.19.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:02506dc23a5d90e04d4f65c7791e65cf44bd91b37f24cfc3ef6cf2aff05dc7ec"}, + {file = "uvloop-0.19.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2693049be9d36fef81741fddb3f441673ba12a34a704e7b4361efb75cf30befc"}, + {file = "uvloop-0.19.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:7010271303961c6f0fe37731004335401eb9075a12680738731e9c92ddd96ad6"}, + {file = "uvloop-0.19.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:5daa304d2161d2918fa9a17d5635099a2f78ae5b5960e742b2fcfbb7aefaa593"}, + {file = "uvloop-0.19.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:7207272c9520203fea9b93843bb775d03e1cf88a80a936ce760f60bb5add92f3"}, + {file = "uvloop-0.19.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:78ab247f0b5671cc887c31d33f9b3abfb88d2614b84e4303f1a63b46c046c8bd"}, + {file = "uvloop-0.19.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:472d61143059c84947aa8bb74eabbace30d577a03a1805b77933d6bd13ddebbd"}, + {file = "uvloop-0.19.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:45bf4c24c19fb8a50902ae37c5de50da81de4922af65baf760f7c0c42e1088be"}, + {file = "uvloop-0.19.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:271718e26b3e17906b28b67314c45d19106112067205119dddbd834c2b7ce797"}, + {file = "uvloop-0.19.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:34175c9fd2a4bc3adc1380e1261f60306344e3407c20a4d684fd5f3be010fa3d"}, + {file = "uvloop-0.19.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:e27f100e1ff17f6feeb1f33968bc185bf8ce41ca557deee9d9bbbffeb72030b7"}, + {file = "uvloop-0.19.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:13dfdf492af0aa0a0edf66807d2b465607d11c4fa48f4a1fd41cbea5b18e8e8b"}, + {file = "uvloop-0.19.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:6e3d4e85ac060e2342ff85e90d0c04157acb210b9ce508e784a944f852a40e67"}, + {file = "uvloop-0.19.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8ca4956c9ab567d87d59d49fa3704cf29e37109ad348f2d5223c9bf761a332e7"}, + {file = "uvloop-0.19.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f467a5fd23b4fc43ed86342641f3936a68ded707f4627622fa3f82a120e18256"}, + {file = "uvloop-0.19.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:492e2c32c2af3f971473bc22f086513cedfc66a130756145a931a90c3958cb17"}, + {file = "uvloop-0.19.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:2df95fca285a9f5bfe730e51945ffe2fa71ccbfdde3b0da5772b4ee4f2e770d5"}, + {file = "uvloop-0.19.0.tar.gz", hash = "sha256:0246f4fd1bf2bf702e06b0d45ee91677ee5c31242f39aab4ea6fe0c51aedd0fd"}, +] + +[package.extras] +docs = ["Sphinx (>=4.1.2,<4.2.0)", "sphinx-rtd-theme (>=0.5.2,<0.6.0)", "sphinxcontrib-asyncio (>=0.3.0,<0.4.0)"] +test = ["Cython (>=0.29.36,<0.30.0)", "aiohttp (==3.9.0b0)", "aiohttp (>=3.8.1)", "flake8 (>=5.0,<6.0)", "mypy (>=0.800)", "psutil", "pyOpenSSL (>=23.0.0,<23.1.0)", "pycodestyle (>=2.9.0,<2.10.0)"] + +[[package]] +name = "vine" +version = "5.1.0" +description = "Python promises." +optional = false +python-versions = ">=3.6" +files = [ + {file = "vine-5.1.0-py3-none-any.whl", hash = "sha256:40fdf3c48b2cfe1c38a49e9ae2da6fda88e4794c810050a728bd7413811fb1dc"}, + {file = "vine-5.1.0.tar.gz", hash = "sha256:8b62e981d35c41049211cf62a0a1242d8c1ee9bd15bb196ce38aefd6799e61e0"}, +] + +[[package]] +name = "watchfiles" +version = "0.21.0" +description = "Simple, modern and high performance file watching and code reload in python." +optional = false +python-versions = ">=3.8" +files = [ + {file = "watchfiles-0.21.0-cp310-cp310-macosx_10_7_x86_64.whl", hash = "sha256:27b4035013f1ea49c6c0b42d983133b136637a527e48c132d368eb19bf1ac6aa"}, + {file = "watchfiles-0.21.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c81818595eff6e92535ff32825f31c116f867f64ff8cdf6562cd1d6b2e1e8f3e"}, + {file = "watchfiles-0.21.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6c107ea3cf2bd07199d66f156e3ea756d1b84dfd43b542b2d870b77868c98c03"}, + {file = "watchfiles-0.21.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0d9ac347653ebd95839a7c607608703b20bc07e577e870d824fa4801bc1cb124"}, + {file = "watchfiles-0.21.0-cp310-cp310-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:5eb86c6acb498208e7663ca22dbe68ca2cf42ab5bf1c776670a50919a56e64ab"}, + {file = "watchfiles-0.21.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:f564bf68404144ea6b87a78a3f910cc8de216c6b12a4cf0b27718bf4ec38d303"}, + {file = "watchfiles-0.21.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:3d0f32ebfaa9c6011f8454994f86108c2eb9c79b8b7de00b36d558cadcedaa3d"}, + {file = "watchfiles-0.21.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b6d45d9b699ecbac6c7bd8e0a2609767491540403610962968d258fd6405c17c"}, + {file = "watchfiles-0.21.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:aff06b2cac3ef4616e26ba17a9c250c1fe9dd8a5d907d0193f84c499b1b6e6a9"}, + {file = "watchfiles-0.21.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:d9792dff410f266051025ecfaa927078b94cc7478954b06796a9756ccc7e14a9"}, + {file = "watchfiles-0.21.0-cp310-none-win32.whl", hash = "sha256:214cee7f9e09150d4fb42e24919a1e74d8c9b8a9306ed1474ecaddcd5479c293"}, + {file = "watchfiles-0.21.0-cp310-none-win_amd64.whl", hash = "sha256:1ad7247d79f9f55bb25ab1778fd47f32d70cf36053941f07de0b7c4e96b5d235"}, + {file = "watchfiles-0.21.0-cp311-cp311-macosx_10_7_x86_64.whl", hash = "sha256:668c265d90de8ae914f860d3eeb164534ba2e836811f91fecc7050416ee70aa7"}, + {file = "watchfiles-0.21.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:3a23092a992e61c3a6a70f350a56db7197242f3490da9c87b500f389b2d01eef"}, + {file = "watchfiles-0.21.0-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:e7941bbcfdded9c26b0bf720cb7e6fd803d95a55d2c14b4bd1f6a2772230c586"}, + {file = "watchfiles-0.21.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:11cd0c3100e2233e9c53106265da31d574355c288e15259c0d40a4405cbae317"}, + {file = "watchfiles-0.21.0-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:d78f30cbe8b2ce770160d3c08cff01b2ae9306fe66ce899b73f0409dc1846c1b"}, + {file = "watchfiles-0.21.0-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6674b00b9756b0af620aa2a3346b01f8e2a3dc729d25617e1b89cf6af4a54eb1"}, + {file = "watchfiles-0.21.0-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fd7ac678b92b29ba630d8c842d8ad6c555abda1b9ef044d6cc092dacbfc9719d"}, + {file = "watchfiles-0.21.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9c873345680c1b87f1e09e0eaf8cf6c891b9851d8b4d3645e7efe2ec20a20cc7"}, + {file = "watchfiles-0.21.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:49f56e6ecc2503e7dbe233fa328b2be1a7797d31548e7a193237dcdf1ad0eee0"}, + {file = "watchfiles-0.21.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:02d91cbac553a3ad141db016e3350b03184deaafeba09b9d6439826ee594b365"}, + {file = "watchfiles-0.21.0-cp311-none-win32.whl", hash = "sha256:ebe684d7d26239e23d102a2bad2a358dedf18e462e8808778703427d1f584400"}, + {file = "watchfiles-0.21.0-cp311-none-win_amd64.whl", hash = "sha256:4566006aa44cb0d21b8ab53baf4b9c667a0ed23efe4aaad8c227bfba0bf15cbe"}, + {file = "watchfiles-0.21.0-cp311-none-win_arm64.whl", hash = "sha256:c550a56bf209a3d987d5a975cdf2063b3389a5d16caf29db4bdddeae49f22078"}, + {file = "watchfiles-0.21.0-cp312-cp312-macosx_10_7_x86_64.whl", hash = "sha256:51ddac60b96a42c15d24fbdc7a4bfcd02b5a29c047b7f8bf63d3f6f5a860949a"}, + {file = "watchfiles-0.21.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:511f0b034120cd1989932bf1e9081aa9fb00f1f949fbd2d9cab6264916ae89b1"}, + {file = "watchfiles-0.21.0-cp312-cp312-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:cfb92d49dbb95ec7a07511bc9efb0faff8fe24ef3805662b8d6808ba8409a71a"}, + {file = "watchfiles-0.21.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3f92944efc564867bbf841c823c8b71bb0be75e06b8ce45c084b46411475a915"}, + {file = "watchfiles-0.21.0-cp312-cp312-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:642d66b75eda909fd1112d35c53816d59789a4b38c141a96d62f50a3ef9b3360"}, + {file = "watchfiles-0.21.0-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d23bcd6c8eaa6324fe109d8cac01b41fe9a54b8c498af9ce464c1aeeb99903d6"}, + {file = "watchfiles-0.21.0-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:18d5b4da8cf3e41895b34e8c37d13c9ed294954907929aacd95153508d5d89d7"}, + {file = "watchfiles-0.21.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1b8d1eae0f65441963d805f766c7e9cd092f91e0c600c820c764a4ff71a0764c"}, + {file = "watchfiles-0.21.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:1fd9a5205139f3c6bb60d11f6072e0552f0a20b712c85f43d42342d162be1235"}, + {file = "watchfiles-0.21.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:a1e3014a625bcf107fbf38eece0e47fa0190e52e45dc6eee5a8265ddc6dc5ea7"}, + {file = "watchfiles-0.21.0-cp312-none-win32.whl", hash = "sha256:9d09869f2c5a6f2d9df50ce3064b3391d3ecb6dced708ad64467b9e4f2c9bef3"}, + {file = "watchfiles-0.21.0-cp312-none-win_amd64.whl", hash = "sha256:18722b50783b5e30a18a8a5db3006bab146d2b705c92eb9a94f78c72beb94094"}, + {file = "watchfiles-0.21.0-cp312-none-win_arm64.whl", hash = "sha256:a3b9bec9579a15fb3ca2d9878deae789df72f2b0fdaf90ad49ee389cad5edab6"}, + {file = "watchfiles-0.21.0-cp38-cp38-macosx_10_7_x86_64.whl", hash = "sha256:4ea10a29aa5de67de02256a28d1bf53d21322295cb00bd2d57fcd19b850ebd99"}, + {file = "watchfiles-0.21.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:40bca549fdc929b470dd1dbfcb47b3295cb46a6d2c90e50588b0a1b3bd98f429"}, + {file = "watchfiles-0.21.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:9b37a7ba223b2f26122c148bb8d09a9ff312afca998c48c725ff5a0a632145f7"}, + {file = "watchfiles-0.21.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ec8c8900dc5c83650a63dd48c4d1d245343f904c4b64b48798c67a3767d7e165"}, + {file = "watchfiles-0.21.0-cp38-cp38-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:8ad3fe0a3567c2f0f629d800409cd528cb6251da12e81a1f765e5c5345fd0137"}, + {file = "watchfiles-0.21.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:9d353c4cfda586db2a176ce42c88f2fc31ec25e50212650c89fdd0f560ee507b"}, + {file = "watchfiles-0.21.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:83a696da8922314ff2aec02987eefb03784f473281d740bf9170181829133765"}, + {file = "watchfiles-0.21.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5a03651352fc20975ee2a707cd2d74a386cd303cc688f407296064ad1e6d1562"}, + {file = "watchfiles-0.21.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:3ad692bc7792be8c32918c699638b660c0de078a6cbe464c46e1340dadb94c19"}, + {file = "watchfiles-0.21.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:06247538e8253975bdb328e7683f8515ff5ff041f43be6c40bff62d989b7d0b0"}, + {file = "watchfiles-0.21.0-cp38-none-win32.whl", hash = "sha256:9a0aa47f94ea9a0b39dd30850b0adf2e1cd32a8b4f9c7aa443d852aacf9ca214"}, + {file = "watchfiles-0.21.0-cp38-none-win_amd64.whl", hash = "sha256:8d5f400326840934e3507701f9f7269247f7c026d1b6cfd49477d2be0933cfca"}, + {file = "watchfiles-0.21.0-cp39-cp39-macosx_10_7_x86_64.whl", hash = "sha256:7f762a1a85a12cc3484f77eee7be87b10f8c50b0b787bb02f4e357403cad0c0e"}, + {file = "watchfiles-0.21.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:6e9be3ef84e2bb9710f3f777accce25556f4a71e15d2b73223788d528fcc2052"}, + {file = "watchfiles-0.21.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:4c48a10d17571d1275701e14a601e36959ffada3add8cdbc9e5061a6e3579a5d"}, + {file = "watchfiles-0.21.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6c889025f59884423428c261f212e04d438de865beda0b1e1babab85ef4c0f01"}, + {file = "watchfiles-0.21.0-cp39-cp39-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:66fac0c238ab9a2e72d026b5fb91cb902c146202bbd29a9a1a44e8db7b710b6f"}, + {file = "watchfiles-0.21.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:b4a21f71885aa2744719459951819e7bf5a906a6448a6b2bbce8e9cc9f2c8128"}, + {file = "watchfiles-0.21.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:1c9198c989f47898b2c22201756f73249de3748e0fc9de44adaf54a8b259cc0c"}, + {file = "watchfiles-0.21.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d8f57c4461cd24fda22493109c45b3980863c58a25b8bec885ca8bea6b8d4b28"}, + {file = "watchfiles-0.21.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:853853cbf7bf9408b404754b92512ebe3e3a83587503d766d23e6bf83d092ee6"}, + {file = "watchfiles-0.21.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d5b1dc0e708fad9f92c296ab2f948af403bf201db8fb2eb4c8179db143732e49"}, + {file = "watchfiles-0.21.0-cp39-none-win32.whl", hash = "sha256:59137c0c6826bd56c710d1d2bda81553b5e6b7c84d5a676747d80caf0409ad94"}, + {file = "watchfiles-0.21.0-cp39-none-win_amd64.whl", hash = "sha256:6cb8fdc044909e2078c248986f2fc76f911f72b51ea4a4fbbf472e01d14faa58"}, + {file = "watchfiles-0.21.0-pp310-pypy310_pp73-macosx_10_7_x86_64.whl", hash = "sha256:ab03a90b305d2588e8352168e8c5a1520b721d2d367f31e9332c4235b30b8994"}, + {file = "watchfiles-0.21.0-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:927c589500f9f41e370b0125c12ac9e7d3a2fd166b89e9ee2828b3dda20bfe6f"}, + {file = "watchfiles-0.21.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1bd467213195e76f838caf2c28cd65e58302d0254e636e7c0fca81efa4a2e62c"}, + {file = "watchfiles-0.21.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:02b73130687bc3f6bb79d8a170959042eb56eb3a42df3671c79b428cd73f17cc"}, + {file = "watchfiles-0.21.0-pp38-pypy38_pp73-macosx_10_7_x86_64.whl", hash = "sha256:08dca260e85ffae975448e344834d765983237ad6dc308231aa16e7933db763e"}, + {file = "watchfiles-0.21.0-pp38-pypy38_pp73-macosx_11_0_arm64.whl", hash = "sha256:3ccceb50c611c433145502735e0370877cced72a6c70fd2410238bcbc7fe51d8"}, + {file = "watchfiles-0.21.0-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:57d430f5fb63fea141ab71ca9c064e80de3a20b427ca2febcbfcef70ff0ce895"}, + {file = "watchfiles-0.21.0-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0dd5fad9b9c0dd89904bbdea978ce89a2b692a7ee8a0ce19b940e538c88a809c"}, + {file = "watchfiles-0.21.0-pp39-pypy39_pp73-macosx_10_7_x86_64.whl", hash = "sha256:be6dd5d52b73018b21adc1c5d28ac0c68184a64769052dfeb0c5d9998e7f56a2"}, + {file = "watchfiles-0.21.0-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:b3cab0e06143768499384a8a5efb9c4dc53e19382952859e4802f294214f36ec"}, + {file = "watchfiles-0.21.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8c6ed10c2497e5fedadf61e465b3ca12a19f96004c15dcffe4bd442ebadc2d85"}, + {file = "watchfiles-0.21.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:43babacef21c519bc6631c5fce2a61eccdfc011b4bcb9047255e9620732c8097"}, + {file = "watchfiles-0.21.0.tar.gz", hash = "sha256:c76c635fabf542bb78524905718c39f736a98e5ab25b23ec6d4abede1a85a6a3"}, +] + +[package.dependencies] +anyio = ">=3.0.0" + +[[package]] +name = "wcwidth" +version = "0.2.13" +description = "Measures the displayed width of unicode strings in a terminal" +optional = false +python-versions = "*" +files = [ + {file = "wcwidth-0.2.13-py2.py3-none-any.whl", hash = "sha256:3da69048e4540d84af32131829ff948f1e022c1c6bdb8d6102117aac784f6859"}, + {file = "wcwidth-0.2.13.tar.gz", hash = "sha256:72ea0c06399eb286d978fdedb6923a9eb47e1c486ce63e9b4e64fc18303972b5"}, +] + +[[package]] +name = "websockets" +version = "12.0" +description = "An implementation of the WebSocket Protocol (RFC 6455 & 7692)" +optional = false +python-versions = ">=3.8" +files = [ + {file = "websockets-12.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:d554236b2a2006e0ce16315c16eaa0d628dab009c33b63ea03f41c6107958374"}, + {file = "websockets-12.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:2d225bb6886591b1746b17c0573e29804619c8f755b5598d875bb4235ea639be"}, + {file = "websockets-12.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:eb809e816916a3b210bed3c82fb88eaf16e8afcf9c115ebb2bacede1797d2547"}, + {file = "websockets-12.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c588f6abc13f78a67044c6b1273a99e1cf31038ad51815b3b016ce699f0d75c2"}, + {file = "websockets-12.0-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5aa9348186d79a5f232115ed3fa9020eab66d6c3437d72f9d2c8ac0c6858c558"}, + {file = "websockets-12.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6350b14a40c95ddd53e775dbdbbbc59b124a5c8ecd6fbb09c2e52029f7a9f480"}, + {file = "websockets-12.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:70ec754cc2a769bcd218ed8d7209055667b30860ffecb8633a834dde27d6307c"}, + {file = "websockets-12.0-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:6e96f5ed1b83a8ddb07909b45bd94833b0710f738115751cdaa9da1fb0cb66e8"}, + {file = "websockets-12.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:4d87be612cbef86f994178d5186add3d94e9f31cc3cb499a0482b866ec477603"}, + {file = "websockets-12.0-cp310-cp310-win32.whl", hash = "sha256:befe90632d66caaf72e8b2ed4d7f02b348913813c8b0a32fae1cc5fe3730902f"}, + {file = "websockets-12.0-cp310-cp310-win_amd64.whl", hash = "sha256:363f57ca8bc8576195d0540c648aa58ac18cf85b76ad5202b9f976918f4219cf"}, + {file = "websockets-12.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:5d873c7de42dea355d73f170be0f23788cf3fa9f7bed718fd2830eefedce01b4"}, + {file = "websockets-12.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3f61726cae9f65b872502ff3c1496abc93ffbe31b278455c418492016e2afc8f"}, + {file = "websockets-12.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ed2fcf7a07334c77fc8a230755c2209223a7cc44fc27597729b8ef5425aa61a3"}, + {file = "websockets-12.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8e332c210b14b57904869ca9f9bf4ca32f5427a03eeb625da9b616c85a3a506c"}, + {file = "websockets-12.0-cp311-cp311-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:5693ef74233122f8ebab026817b1b37fe25c411ecfca084b29bc7d6efc548f45"}, + {file = "websockets-12.0-cp311-cp311-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6e9e7db18b4539a29cc5ad8c8b252738a30e2b13f033c2d6e9d0549b45841c04"}, + {file = "websockets-12.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:6e2df67b8014767d0f785baa98393725739287684b9f8d8a1001eb2839031447"}, + {file = "websockets-12.0-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:bea88d71630c5900690fcb03161ab18f8f244805c59e2e0dc4ffadae0a7ee0ca"}, + {file = "websockets-12.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:dff6cdf35e31d1315790149fee351f9e52978130cef6c87c4b6c9b3baf78bc53"}, + {file = "websockets-12.0-cp311-cp311-win32.whl", hash = "sha256:3e3aa8c468af01d70332a382350ee95f6986db479ce7af14d5e81ec52aa2b402"}, + {file = "websockets-12.0-cp311-cp311-win_amd64.whl", hash = "sha256:25eb766c8ad27da0f79420b2af4b85d29914ba0edf69f547cc4f06ca6f1d403b"}, + {file = "websockets-12.0-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:0e6e2711d5a8e6e482cacb927a49a3d432345dfe7dea8ace7b5790df5932e4df"}, + {file = "websockets-12.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:dbcf72a37f0b3316e993e13ecf32f10c0e1259c28ffd0a85cee26e8549595fbc"}, + {file = "websockets-12.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:12743ab88ab2af1d17dd4acb4645677cb7063ef4db93abffbf164218a5d54c6b"}, + {file = "websockets-12.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7b645f491f3c48d3f8a00d1fce07445fab7347fec54a3e65f0725d730d5b99cb"}, + {file = "websockets-12.0-cp312-cp312-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9893d1aa45a7f8b3bc4510f6ccf8db8c3b62120917af15e3de247f0780294b92"}, + {file = "websockets-12.0-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1f38a7b376117ef7aff996e737583172bdf535932c9ca021746573bce40165ed"}, + {file = "websockets-12.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:f764ba54e33daf20e167915edc443b6f88956f37fb606449b4a5b10ba42235a5"}, + {file = "websockets-12.0-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:1e4b3f8ea6a9cfa8be8484c9221ec0257508e3a1ec43c36acdefb2a9c3b00aa2"}, + {file = "websockets-12.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:9fdf06fd06c32205a07e47328ab49c40fc1407cdec801d698a7c41167ea45113"}, + {file = "websockets-12.0-cp312-cp312-win32.whl", hash = "sha256:baa386875b70cbd81798fa9f71be689c1bf484f65fd6fb08d051a0ee4e79924d"}, + {file = "websockets-12.0-cp312-cp312-win_amd64.whl", hash = "sha256:ae0a5da8f35a5be197f328d4727dbcfafa53d1824fac3d96cdd3a642fe09394f"}, + {file = "websockets-12.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5f6ffe2c6598f7f7207eef9a1228b6f5c818f9f4d53ee920aacd35cec8110438"}, + {file = "websockets-12.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:9edf3fc590cc2ec20dc9d7a45108b5bbaf21c0d89f9fd3fd1685e223771dc0b2"}, + {file = "websockets-12.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:8572132c7be52632201a35f5e08348137f658e5ffd21f51f94572ca6c05ea81d"}, + {file = "websockets-12.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:604428d1b87edbf02b233e2c207d7d528460fa978f9e391bd8aaf9c8311de137"}, + {file = "websockets-12.0-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1a9d160fd080c6285e202327aba140fc9a0d910b09e423afff4ae5cbbf1c7205"}, + {file = "websockets-12.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:87b4aafed34653e465eb77b7c93ef058516cb5acf3eb21e42f33928616172def"}, + {file = "websockets-12.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:b2ee7288b85959797970114deae81ab41b731f19ebcd3bd499ae9ca0e3f1d2c8"}, + {file = "websockets-12.0-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:7fa3d25e81bfe6a89718e9791128398a50dec6d57faf23770787ff441d851967"}, + {file = "websockets-12.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:a571f035a47212288e3b3519944f6bf4ac7bc7553243e41eac50dd48552b6df7"}, + {file = "websockets-12.0-cp38-cp38-win32.whl", hash = "sha256:3c6cc1360c10c17463aadd29dd3af332d4a1adaa8796f6b0e9f9df1fdb0bad62"}, + {file = "websockets-12.0-cp38-cp38-win_amd64.whl", hash = "sha256:1bf386089178ea69d720f8db6199a0504a406209a0fc23e603b27b300fdd6892"}, + {file = "websockets-12.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:ab3d732ad50a4fbd04a4490ef08acd0517b6ae6b77eb967251f4c263011a990d"}, + {file = "websockets-12.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:a1d9697f3337a89691e3bd8dc56dea45a6f6d975f92e7d5f773bc715c15dde28"}, + {file = "websockets-12.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1df2fbd2c8a98d38a66f5238484405b8d1d16f929bb7a33ed73e4801222a6f53"}, + {file = "websockets-12.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:23509452b3bc38e3a057382c2e941d5ac2e01e251acce7adc74011d7d8de434c"}, + {file = "websockets-12.0-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:2e5fc14ec6ea568200ea4ef46545073da81900a2b67b3e666f04adf53ad452ec"}, + {file = "websockets-12.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:46e71dbbd12850224243f5d2aeec90f0aaa0f2dde5aeeb8fc8df21e04d99eff9"}, + {file = "websockets-12.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:b81f90dcc6c85a9b7f29873beb56c94c85d6f0dac2ea8b60d995bd18bf3e2aae"}, + {file = "websockets-12.0-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:a02413bc474feda2849c59ed2dfb2cddb4cd3d2f03a2fedec51d6e959d9b608b"}, + {file = "websockets-12.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:bbe6013f9f791944ed31ca08b077e26249309639313fff132bfbf3ba105673b9"}, + {file = "websockets-12.0-cp39-cp39-win32.whl", hash = "sha256:cbe83a6bbdf207ff0541de01e11904827540aa069293696dd528a6640bd6a5f6"}, + {file = "websockets-12.0-cp39-cp39-win_amd64.whl", hash = "sha256:fc4e7fa5414512b481a2483775a8e8be7803a35b30ca805afa4998a84f9fd9e8"}, + {file = "websockets-12.0-pp310-pypy310_pp73-macosx_10_9_x86_64.whl", hash = "sha256:248d8e2446e13c1d4326e0a6a4e9629cb13a11195051a73acf414812700badbd"}, + {file = "websockets-12.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f44069528d45a933997a6fef143030d8ca8042f0dfaad753e2906398290e2870"}, + {file = "websockets-12.0-pp310-pypy310_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c4e37d36f0d19f0a4413d3e18c0d03d0c268ada2061868c1e6f5ab1a6d575077"}, + {file = "websockets-12.0-pp310-pypy310_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3d829f975fc2e527a3ef2f9c8f25e553eb7bc779c6665e8e1d52aa22800bb38b"}, + {file = "websockets-12.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:2c71bd45a777433dd9113847af751aae36e448bc6b8c361a566cb043eda6ec30"}, + {file = "websockets-12.0-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:0bee75f400895aef54157b36ed6d3b308fcab62e5260703add87f44cee9c82a6"}, + {file = "websockets-12.0-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:423fc1ed29f7512fceb727e2d2aecb952c46aa34895e9ed96071821309951123"}, + {file = "websockets-12.0-pp38-pypy38_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:27a5e9964ef509016759f2ef3f2c1e13f403725a5e6a1775555994966a66e931"}, + {file = "websockets-12.0-pp38-pypy38_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c3181df4583c4d3994d31fb235dc681d2aaad744fbdbf94c4802485ececdecf2"}, + {file = "websockets-12.0-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:b067cb952ce8bf40115f6c19f478dc71c5e719b7fbaa511359795dfd9d1a6468"}, + {file = "websockets-12.0-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:00700340c6c7ab788f176d118775202aadea7602c5cc6be6ae127761c16d6b0b"}, + {file = "websockets-12.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e469d01137942849cff40517c97a30a93ae79917752b34029f0ec72df6b46399"}, + {file = "websockets-12.0-pp39-pypy39_pp73-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:ffefa1374cd508d633646d51a8e9277763a9b78ae71324183693959cf94635a7"}, + {file = "websockets-12.0-pp39-pypy39_pp73-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ba0cab91b3956dfa9f512147860783a1829a8d905ee218a9837c18f683239611"}, + {file = "websockets-12.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:2cb388a5bfb56df4d9a406783b7f9dbefb888c09b71629351cc6b036e9259370"}, + {file = "websockets-12.0-py3-none-any.whl", hash = "sha256:dc284bbc8d7c78a6c69e0c7325ab46ee5e40bb4d50e494d8131a07ef47500e9e"}, + {file = "websockets-12.0.tar.gz", hash = "sha256:81df9cbcbb6c260de1e007e58c011bfebe2dafc8435107b0537f393dd38c8b1b"}, +] + +[metadata] +lock-version = "2.0" +python-versions = "^3.10" +content-hash = "9c11764f0a3a9df174c0caec1f13eafb88ea099a2174a574f2de5ba49ae37b14" diff --git a/pyproject.toml b/pyproject.toml new file mode 100644 index 0000000000000000000000000000000000000000..3ffd3e2fb7a215e0187a063921754b33d4801541 --- /dev/null +++ b/pyproject.toml @@ -0,0 +1,52 @@ +[tool.poetry] +name = "reaspeech" +version = "1.0.0" +description = "Speech recognition for REAPER" +homepage = "https://github.com/TeamAudio/reaspeech/" +license = "https://github.com/TeamAudio/reaspeech/blob/main/LICENSE" +authors = [ + "Dave Benjamin", + "Mike DeFreitas", + "Roel Sanchez", +] +readme = "README.md" +packages = [{ include = "app" }] + +[[tool.poetry.source]] +name = "pytorch" +url = "https://download.pytorch.org/whl/cpu" +priority = "explicit" + +[tool.poetry.dependencies] +python = "^3.10" +unidecode = "^1.3.4" +uvicorn = { extras = ["standard"], version = "^0.18.2" } +gunicorn = "^20.1.0" +tqdm = "^4.64.1" +python-multipart = "^0.0.5" +ffmpeg-python = "^0.2.0" +fastapi = "^0.95.1" +llvmlite = "^0.39.1" +numba = "^0.56.4" +openai-whisper = "20230918" +faster-whisper = "^1.0.2" +torch = [ + {markers = "sys_platform == 'darwin' and platform_machine == 'arm64'", url = "https://download.pytorch.org/whl/cpu/torch-1.13.1-cp310-none-macosx_11_0_arm64.whl"}, + {markers = "sys_platform == 'linux' and platform_machine == 'arm64'", url="https://download.pytorch.org/whl/cpu/torch-1.13.1-cp310-none-macosx_11_0_arm64.whl"}, + {markers = "sys_platform == 'darwin' and platform_machine == 'x86_64'", url = "https://download.pytorch.org/whl/cpu/torch-1.13.1-cp310-none-macosx_10_9_x86_64.whl"}, + {markers = "sys_platform == 'linux' and platform_machine == 'aarch64'", url="https://download.pytorch.org/whl/torch-1.13.1-cp310-cp310-manylinux2014_aarch64.whl"}, + {markers = "sys_platform == 'linux' and platform_machine == 'x86_64'", url="https://download.pytorch.org/whl/cpu/torch-1.13.1%2Bcpu-cp310-cp310-linux_x86_64.whl"}, + {markers = "sys_platform == 'win' and platform_machine == 'amd64'", url="https://download.pytorch.org/whl/cpu/torch-1.13.1%2Bcpu-cp310-cp310-win_amd64.whl"}, +] +jinja2 = "^3.1.2" +celery = "^5.2.7" +redis = "^4.5.4" +ctranslate2 = "4.2.1" +aiofiles = "^23.2.1" + +[tool.poetry.dev-dependencies] +pytest = "^6.2.5" + +[build-system] +requires = ["poetry-core>=1.0.0"] +build-backend = "poetry.core.masonry.api" diff --git a/reascripts/LICENSE b/reascripts/LICENSE new file mode 100644 index 0000000000000000000000000000000000000000..bc0a21eef23de542233268d2348a09e57e7d0a91 --- /dev/null +++ b/reascripts/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2024 TeamAudio + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/reascripts/ReaSpeech/.gitignore b/reascripts/ReaSpeech/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..bea575523049584375fc6633238255015d3a4c84 --- /dev/null +++ b/reascripts/ReaSpeech/.gitignore @@ -0,0 +1 @@ +TAGS diff --git a/reascripts/ReaSpeech/.luacheckrc b/reascripts/ReaSpeech/.luacheckrc new file mode 100644 index 0000000000000000000000000000000000000000..43c2e3bdfcb92fb940a034f125cf35de95330822 --- /dev/null +++ b/reascripts/ReaSpeech/.luacheckrc @@ -0,0 +1,14 @@ +globals = { + "Script", + "json", + "reaper", + "url", + "ImGuiTheme", + "IMAGES", + "Polo", + "ReaUtil", +} + +allow_defined = true +color = false +ignore = {"212/self", "212/_.*", "432/self", "631"} diff --git a/reascripts/ReaSpeech/Makefile b/reascripts/ReaSpeech/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..2227184bcbdec1d9a9f67a70d239aaebe2844644 --- /dev/null +++ b/reascripts/ReaSpeech/Makefile @@ -0,0 +1,50 @@ +LUA53=lua5.3 +LUAC53=luac5.3 +LUA54=lua5.4 +LUAC54=luac5.4 +LUACHECK=luacheck + +LIBS=ImGuiTheme.lua OptionsConfig.lua Polo.lua ReaIter.lua ReaUtil.lua Tempfile.lua +VENDOR=json.lua url.lua + +source:=$(wildcard source/*.lua source/include/*.lua) +libs:=$(foreach lib, $(LIBS), ../common/libs/$(lib)) +vendor:=$(foreach lib, $(VENDOR), ../common/vendor/$(lib)) +tests:=$(wildcard tests/Test*.lua) +dest=../../app/static/reascripts/ReaSpeech + +all: lint test publish + +lint: $(source) $(libs) + $(LUACHECK) $? + +test: $(source) $(libs) $(tests) + true $(foreach test, $(tests), && $(LUA53) $(test) -v) + true $(foreach test, $(tests), && $(LUA54) $(test) -v) + +.PHONY: build +build: build/ReaSpeech-5.3.luac build/ReaSpeech-5.4.luac + +build/ReaSpeechBundle.lua: $(source) $(libs) $(vendor) version.lua + cat source/include/header.lua source/include/globals.lua $(libs) $(vendor) resources/images/*.lua source/*.lua version.lua source/include/main.lua > build/ReaSpeechBundle.lua + +build/ReaSpeech-5.3.luac: build/ReaSpeechBundle.lua + $(LUAC53) -o build/ReaSpeech-5.3.luac -s build/ReaSpeechBundle.lua + +build/ReaSpeech-5.4.luac: build/ReaSpeechBundle.lua + $(LUAC54) -o build/ReaSpeech-5.4.luac -s build/ReaSpeechBundle.lua + +.PHONY: publish +publish: build + cp build/ReaSpeech-5.3.luac $(dest)/ + cp build/ReaSpeech-5.4.luac $(dest)/ + +.PHONY: tags +tags: + find source/ -type f -iname '*.lua' -print0 | xargs -0 etags + find ../common/libs/ -type f -iname '*.lua' -print0 | xargs -0 etags --append + find ../common/vendor/ -type f -iname '*.lua' -print0 | xargs -0 etags --append + +.PHONY: clean +clean: + rm -f build/ReaSpeech-5.3.luac build/ReaSpeech-5.4.luac build/ReaSpeechBundle.lua TAGS diff --git a/reascripts/ReaSpeech/ReaSpeechDev.lua b/reascripts/ReaSpeech/ReaSpeechDev.lua new file mode 100644 index 0000000000000000000000000000000000000000..cbf5eb71800588238f6be4d8d3db866fcec07208 --- /dev/null +++ b/reascripts/ReaSpeech/ReaSpeechDev.lua @@ -0,0 +1,52 @@ +--[[ + +ReaSpeechDev.lua - ReaSpeech UI development version + +]]-- + +-- Set to true to enable product activation flow, false to disable +local ENABLE_ACTIVATION = true + +local script_path, _ = ({reaper.get_action_context()})[2]:match("(.-)([^/\\]+).lua$") + +dofile(script_path .. 'source/include/globals.lua') +dofile(script_path .. '../common/vendor/json.lua') +dofile(script_path .. '../common/vendor/url.lua') + +-- Considering extracting this into a function, but we're getting into +-- chicken/egg territory with where that would be defined and how it'd be used. +for _, source_dir in pairs({'resources/images', '../common/libs', 'source'}) do + local source_file, source_index = '', 0 + while (source_file ~= nil) do + source_file = reaper.EnumerateFiles(script_path .. source_dir, source_index) + if source_file and source_file:sub(-4) == '.lua' then + reaper.ShowConsoleMsg(script_path .. source_dir .. '/' .. source_file .. '\n') + dofile(script_path .. source_dir .. '/' .. source_file) + end + source_index = source_index + 1 + end +end + +-- Might be time before too long to standardize this interface across apps. +if not ENABLE_ACTIVATION then + function ReaSpeechProductActivation:activation_state_check() end + ReaSpeechProductActivation.state = 'activated' + + local activation = ReaSpeechProductActivation.new() + activation.config:set('eula_signed', true) +end + +-- We're not inside of docker! We're undocked! +Script = { + name = "ReaSpeechDev", + host = "localhost:9000", + lua = _VERSION:match('[%d.]+'), + timeout = 30000, +} + +Fonts.LOCAL_FILE = script_path .. "../../app/static/reascripts/ReaSpeech/icons.ttf" + +dofile(script_path .. 'source/include/main.lua') + +-- Uncomment the following line to load example response data +-- dofile(script_path .. 'examples/response_data_json.lua') diff --git a/reascripts/ReaSpeech/build/.gitignore b/reascripts/ReaSpeech/build/.gitignore new file mode 100644 index 0000000000000000000000000000000000000000..5e7d2734cfc60289debf74293817c0a8f572ff32 --- /dev/null +++ b/reascripts/ReaSpeech/build/.gitignore @@ -0,0 +1,4 @@ +# Ignore everything in this directory +* +# Except this file +!.gitignore diff --git a/reascripts/ReaSpeech/examples/response_data_json.lua b/reascripts/ReaSpeech/examples/response_data_json.lua new file mode 100644 index 0000000000000000000000000000000000000000..210949d81ee36a99c895d096a81f58ccf05341c0 --- /dev/null +++ b/reascripts/ReaSpeech/examples/response_data_json.lua @@ -0,0 +1,11 @@ +response = json.decode([[ +{"language": "en", "segments": [{"id": 1, "seek": 2203, "start": 0.0, "end": 4.68, "text": " Don't worry about it, but maybe I will fill out one of those suggestion cards after all.", "tokens": [50364, 1468, 380, 3292, 466, 309, 11, 457, 1310, 286, 486, 2836, 484, 472, 295, 729, 16541, 5632, 934, 439, 13, 50604], "temperature": 0.0, "avg_logprob": -0.20203689297476968, "compression_ratio": 1.5044247787610618, "no_speech_prob": 0.01001946534961462, "words": [{"start": 0.0, "end": 0.48, "word": " Don't", "probability": 0.9738937318325043}, {"start": 0.48, "end": 0.72, "word": " worry", "probability": 0.9985054731369019}, {"start": 0.72, "end": 0.94, "word": " about", "probability": 0.9956048727035522}, {"start": 0.94, "end": 1.2, "word": " it,", "probability": 0.9958335161209106}, {"start": 1.46, "end": 1.68, "word": " but", "probability": 0.9893428087234497}, {"start": 1.68, "end": 2.06, "word": " maybe", "probability": 0.985736072063446}, {"start": 2.06, "end": 2.2, "word": " I", "probability": 0.9841459393501282}, {"start": 2.2, "end": 2.36, "word": " will", "probability": 0.9575523734092712}, {"start": 2.36, "end": 2.54, "word": " fill", "probability": 0.9963791966438293}, {"start": 2.54, "end": 2.68, "word": " out", "probability": 0.9883477091789246}, {"start": 2.68, "end": 2.84, "word": " one", "probability": 0.9910874962806702}, {"start": 2.84, "end": 2.9, "word": " of", "probability": 0.9967635869979858}, {"start": 2.9, "end": 3.2, "word": " those", "probability": 0.9798358678817749}, {"start": 3.2, "end": 3.64, "word": " suggestion", "probability": 0.9172356724739075}, {"start": 3.64, "end": 4.0, "word": " cards", "probability": 0.9973626732826233}, {"start": 4.0, "end": 4.44, "word": " after", "probability": 0.8614948391914368}, {"start": 4.44, "end": 4.68, "word": " all.", "probability": 0.982586681842804}]}, {"id": 2, "seek": 2203, "start": 5.02, "end": 7.66, "text": " You need to rearrange your store and hire more people.", "tokens": [50604, 509, 643, 281, 39568, 428, 3531, 293, 11158, 544, 561, 13, 50748], "temperature": 0.0, "avg_logprob": -0.20203689297476968, "compression_ratio": 1.5044247787610618, "no_speech_prob": 0.01001946534961462, "words": [{"start": 5.02, "end": 5.2, "word": " You", "probability": 0.9949402809143066}, {"start": 5.2, "end": 5.4, "word": " need", "probability": 0.9888043999671936}, {"start": 5.4, "end": 5.68, "word": " to", "probability": 0.995278000831604}, {"start": 5.68, "end": 6.02, "word": " rearrange", "probability": 0.9680283069610596}, {"start": 6.02, "end": 6.34, "word": " your", "probability": 0.9937660694122314}, {"start": 6.34, "end": 6.48, "word": " store", "probability": 0.9905586242675781}, {"start": 6.48, "end": 6.72, "word": " and", "probability": 0.9858508706092834}, {"start": 6.72, "end": 6.92, "word": " hire", "probability": 0.9985997080802917}, {"start": 6.92, "end": 7.16, "word": " more", "probability": 0.9978284239768982}, {"start": 7.16, "end": 7.66, "word": " people.", "probability": 0.9991713762283325}]}, {"id": 3, "seek": 2203, "start": 8.2, "end": 11.26, "text": " Haha, sounds good to me. Good luck finding that book.", "tokens": [50748, 19131, 11, 3263, 665, 281, 385, 13, 2205, 3668, 5006, 300, 1446, 13, 50940], "temperature": 0.0, "avg_logprob": -0.20203689297476968, "compression_ratio": 1.5044247787610618, "no_speech_prob": 0.01001946534961462, "words": [{"start": 8.2, "end": 8.42, "word": " Haha,", "probability": 0.12257678806781769}, {"start": 8.8, "end": 9.04, "word": " sounds", "probability": 0.9721860885620117}, {"start": 9.04, "end": 9.24, "word": " good", "probability": 0.9988430738449097}, {"start": 9.24, "end": 9.44, "word": " to", "probability": 0.9985108971595764}, {"start": 9.44, "end": 9.66, "word": " me.", "probability": 0.9993407130241394}, {"start": 10.0, "end": 10.18, "word": " Good", "probability": 0.9959205389022827}, {"start": 10.18, "end": 10.38, "word": " luck", "probability": 0.9987674951553345}, {"start": 10.38, "end": 10.68, "word": " finding", "probability": 0.9981703758239746}, {"start": 10.68, "end": 11.02, "word": " that", "probability": 0.9961445331573486}, {"start": 11.02, "end": 11.26, "word": " book.", "probability": 0.9984509944915771}]}, {"id": 4, "seek": 2203, "start": 11.78, "end": 12.24, "text": " Thanks.", "tokens": [50940, 2561, 13, 50992], "temperature": 0.0, "avg_logprob": -0.20203689297476968, "compression_ratio": 1.5044247787610618, "no_speech_prob": 0.01001946534961462, "words": [{"start": 11.78, "end": 12.24, "word": " Thanks.", "probability": 0.9695364832878113}]}, {"id": 5, "seek": 2203, "start": 12.9, "end": 16.16, "text": " You're welcome. Okay, who's the next person in line?", "tokens": [50992, 509, 434, 2928, 13, 1033, 11, 567, 311, 264, 958, 954, 294, 1622, 30, 51164], "temperature": 0.0, "avg_logprob": -0.20203689297476968, "compression_ratio": 1.5044247787610618, "no_speech_prob": 0.01001946534961462, "words": [{"start": 12.9, "end": 13.12, "word": " You're", "probability": 0.9849036931991577}, {"start": 13.12, "end": 13.38, "word": " welcome.", "probability": 0.9983079433441162}, {"start": 14.12, "end": 14.5, "word": " Okay,", "probability": 0.7263695001602173}, {"start": 14.86, "end": 15.2, "word": " who's", "probability": 0.9723688066005707}, {"start": 15.2, "end": 15.3, "word": " the", "probability": 0.9945037364959717}, {"start": 15.3, "end": 15.44, "word": " next", "probability": 0.9980334639549255}, {"start": 15.44, "end": 15.74, "word": " person", "probability": 0.9987584352493286}, {"start": 15.74, "end": 15.98, "word": " in", "probability": 0.9971809387207031}, {"start": 15.98, "end": 16.16, "word": " line?", "probability": 0.9988143444061279}]}, {"id": 6, "seek": 2203, "start": 18.49, "end": 22.87, "text": " Now get ready to answer the questions. You may use your notes to help you answer.", "tokens": [51200, 823, 483, 1919, 281, 1867, 264, 1651, 13, 509, 815, 764, 428, 5570, 281, 854, 291, 1867, 13, 51428], "temperature": 0.0, "avg_logprob": -0.20203689297476968, "compression_ratio": 1.5044247787610618, "no_speech_prob": 0.01001946534961462, "words": [{"start": 18.49, "end": 18.75, "word": " Now", "probability": 0.9738929867744446}, {"start": 18.75, "end": 18.97, "word": " get", "probability": 0.9094610810279846}, {"start": 18.97, "end": 19.21, "word": " ready", "probability": 0.9989386200904846}, {"start": 19.21, "end": 19.51, "word": " to", "probability": 0.9987198114395142}, {"start": 19.51, "end": 19.75, "word": " answer", "probability": 0.9979331493377686}, {"start": 19.75, "end": 19.95, "word": " the", "probability": 0.995278000831604}, {"start": 19.95, "end": 20.33, "word": " questions.", "probability": 0.9946743249893188}, {"start": 20.95, "end": 21.09, "word": " You", "probability": 0.9968999624252319}, {"start": 21.09, "end": 21.27, "word": " may", "probability": 0.9985644221305847}, {"start": 21.27, "end": 21.47, "word": " use", "probability": 0.9987497329711914}, {"start": 21.47, "end": 21.77, "word": " your", "probability": 0.9975970387458801}, {"start": 21.77, "end": 21.99, "word": " notes", "probability": 0.9959502220153809}, {"start": 21.99, "end": 22.21, "word": " to", "probability": 0.9991000890731812}, {"start": 22.21, "end": 22.33, "word": " help", "probability": 0.998537540435791}, {"start": 22.33, "end": 22.55, "word": " you", "probability": 0.9948883652687073}, {"start": 22.55, "end": 22.87, "word": " answer.", "probability": 0.9985874891281128}]}], "text": " Don't worry about it, but maybe I will fill out one of those suggestion cards after all. You need to rearrange your store and hire more people. Haha, sounds good to me. Good luck finding that book. Thanks. You're welcome. Okay, who's the next person in line? Now get ready to answer the questions. You may use your notes to help you answer."} +]]) + +for _, segment in pairs(response.segments) do + for _, s in pairs(TranscriptSegment.from_whisper(segment, {}, {})) do + app.transcript:add_segment(s) + end + end + app.transcript:update() + \ No newline at end of file diff --git a/reascripts/ReaSpeech/resources/icons/.fontello b/reascripts/ReaSpeech/resources/icons/.fontello new file mode 100644 index 0000000000000000000000000000000000000000..790c98369affbc80cb59426aad859e29a3579cfb --- /dev/null +++ b/reascripts/ReaSpeech/resources/icons/.fontello @@ -0,0 +1 @@ +d0dcf108b52a898b9077bdc832d3333b \ No newline at end of file diff --git a/reascripts/ReaSpeech/resources/icons/Makefile b/reascripts/ReaSpeech/resources/icons/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..b2bb7ba8c7e5dbd9a0407e150ea8f02cd18911f7 --- /dev/null +++ b/reascripts/ReaSpeech/resources/icons/Makefile @@ -0,0 +1,39 @@ +# Based on Makefile example from https://gist.github.com/puzrin/5537065 + +FONT_DIR ?= ./fontello +FONTELLO_HOST ?= https://fontello.com + +dest=../../../../app/static/reascripts/ReaSpeech + +all: publish + +.PHONY: publish +publish: + cp fontello/font/fontello.ttf $(dest)/icons.ttf + +fontopen: + @if test ! `which curl` ; then \ + echo 'Install curl first.' >&2 ; \ + exit 128 ; \ + fi + curl --silent --show-error --fail --output .fontello \ + --form "config=@${FONT_DIR}/config.json" \ + ${FONTELLO_HOST} + x-www-browser ${FONTELLO_HOST}/`cat .fontello` + +fontsave: + @if test ! `which unzip` ; then \ + echo 'Install unzip first.' >&2 ; \ + exit 128 ; \ + fi + @if test ! -e .fontello ; then \ + echo 'Run `make fontopen` first.' >&2 ; \ + exit 128 ; \ + fi + rm -rf .fontello.src .fontello.zip + curl --silent --show-error --fail --output .fontello.zip \ + ${FONTELLO_HOST}/`cat .fontello`/get + unzip .fontello.zip -d .fontello.src + rm -rf ${FONT_DIR} + mv `find ./.fontello.src -maxdepth 1 -name 'fontello-*'` ${FONT_DIR} + rm -rf .fontello.src .fontello.zip diff --git a/reascripts/ReaSpeech/resources/icons/fontello/LICENSE.txt b/reascripts/ReaSpeech/resources/icons/fontello/LICENSE.txt new file mode 100644 index 0000000000000000000000000000000000000000..8fa3da363e9fc1a7718943afa7f5ed21448040bd --- /dev/null +++ b/reascripts/ReaSpeech/resources/icons/fontello/LICENSE.txt @@ -0,0 +1,12 @@ +Font license info + + +## Font Awesome + + Copyright (C) 2016 by Dave Gandy + + Author: Dave Gandy + License: SIL () + Homepage: http://fortawesome.github.com/Font-Awesome/ + + diff --git a/reascripts/ReaSpeech/resources/icons/fontello/README.txt b/reascripts/ReaSpeech/resources/icons/fontello/README.txt new file mode 100644 index 0000000000000000000000000000000000000000..d870892e392b2a96b2724b773088c61c10fd2cf0 --- /dev/null +++ b/reascripts/ReaSpeech/resources/icons/fontello/README.txt @@ -0,0 +1,75 @@ +This webfont is generated by https://fontello.com open source project. + + +================================================================================ +Please, note, that you should obey original font licenses, used to make this +webfont pack. Details available in LICENSE.txt file. + +- Usually, it's enough to publish content of LICENSE.txt file somewhere on your + site in "About" section. + +- If your project is open-source, usually, it will be ok to make LICENSE.txt + file publicly available in your repository. + +- Fonts, used in Fontello, don't require a clickable link on your site. + But any kind of additional authors crediting is welcome. +================================================================================ + + +Comments on archive content +--------------------------- + +- /font/* - fonts in different formats + +- /css/* - different kinds of css, for all situations. Should be ok with + twitter bootstrap. Also, you can skip style and assign icon classes + directly to text elements, if you don't mind about IE7. + +- demo.html - demo file, to show your webfont content + +- LICENSE.txt - license info about source fonts, used to build your one. + +- config.json - keeps your settings. You can import it back into fontello + anytime, to continue your work + + +Why so many CSS files ? +----------------------- + +Because we like to fit all your needs :) + +- basic file, .css - is usually enough, it contains @font-face + and character code definitions + +- *-ie7.css - if you need IE7 support, but still don't wish to put char codes + directly into html + +- *-codes.css and *-ie7-codes.css - if you like to use your own @font-face + rules, but still wish to benefit from css generation. That can be very + convenient for automated asset build systems. When you need to update font - + no need to manually edit files, just override old version with archive + content. See fontello source code for examples. + +- *-embedded.css - basic css file, but with embedded WOFF font, to avoid + CORS issues in Firefox and IE9+, when fonts are hosted on the separate domain. + We strongly recommend to resolve this issue by `Access-Control-Allow-Origin` + server headers. But if you ok with dirty hack - this file is for you. Note, + that data url moved to separate @font-face to avoid problems with + + + + + + + + + +
+

fontello font demo

+ +
+
+
+
+ a icon-pencil0x61 +
+
+ b icon-cog0x62 +
+
+ c icon-play-circled0x63 +
+
+ d icon-stop-circle0x64 +
+
+
+ + + diff --git a/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.eot b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.eot new file mode 100644 index 0000000000000000000000000000000000000000..f2a429f0654adc5a168d37c7c9053e3d445c248e Binary files /dev/null and b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.eot differ diff --git a/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.svg b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.svg new file mode 100644 index 0000000000000000000000000000000000000000..87959f62c17fd484b0640aa01abf949929b1f679 --- /dev/null +++ b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.svg @@ -0,0 +1,18 @@ + + + +Copyright (C) 2023 by original authors @ fontello.com + + + + + + + + + + + + + + diff --git a/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.ttf b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.ttf new file mode 100644 index 0000000000000000000000000000000000000000..d918491cccfb0be57947cf82cf92d32dbc6568b2 Binary files /dev/null and b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.ttf differ diff --git a/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.woff b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.woff new file mode 100644 index 0000000000000000000000000000000000000000..f2cae529d2dd12b603a4ebd3bb5ed62a016f4561 Binary files /dev/null and b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.woff differ diff --git a/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.woff2 b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.woff2 new file mode 100644 index 0000000000000000000000000000000000000000..aea3c5597263861d6c0b77bba803156879a3a4bc Binary files /dev/null and b/reascripts/ReaSpeech/resources/icons/fontello/font/fontello.woff2 differ diff --git a/reascripts/ReaSpeech/resources/images/Makefile b/reascripts/ReaSpeech/resources/images/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..69a0c480d47c81acf2df87c9b298419b36acdd28 --- /dev/null +++ b/reascripts/ReaSpeech/resources/images/Makefile @@ -0,0 +1,13 @@ +RUBY=ruby + +input:=$(wildcard *.png) +output:=$(input:.png=.lua) + +all: $(output) + +%.lua: %.png + $(RUBY) ../../../common/scripts/png_to_lua.rb $< $@ + +.PHONY: clean +clean: + rm -f *.lua diff --git a/reascripts/ReaSpeech/resources/images/README.md b/reascripts/ReaSpeech/resources/images/README.md new file mode 100644 index 0000000000000000000000000000000000000000..9acdf0e010e8719a4d601821eba8580bd363ecee --- /dev/null +++ b/reascripts/ReaSpeech/resources/images/README.md @@ -0,0 +1,13 @@ +# Image resources + +## How to build + +Using WSL/Ubuntu on Windows, or Ubuntu Linux: + + sudo apt install ruby ruby-chunky-png + make + +Using RubyGems: + + sudo gem install chunky_png + make \ No newline at end of file diff --git a/reascripts/ReaSpeech/resources/images/heading-logo-tech-audio.lua b/reascripts/ReaSpeech/resources/images/heading-logo-tech-audio.lua new file mode 100644 index 0000000000000000000000000000000000000000..a7545e359d249042dc9819f250e16758a0d468b8 --- /dev/null +++ b/reascripts/ReaSpeech/resources/images/heading-logo-tech-audio.lua @@ -0,0 +1,98 @@ +IMAGES = IMAGES or {} +IMAGES['heading-logo-tech-audio'] = { + width = 49, + height = 41, + pixels = { + {2,63,64,66,255,2,63,65,64,255,1,64,65,67,255,1,64,63,68,255,1,60,64,65,255,1,65,63,64,255,1,66,61,65,255,1,65,64,62,255,1,70,58,68,255,1,71,42,72,255,1,148,120,119,255,1,203,187,164,255,1,174,157,139,255,1,149,136,120,255,1,126,112,101,255,1,105,95,86,255,1,95,86,81,255,1,84,77,71,255,1,77,66,70,255,1,59,57,58,255,1,65,57,68,255,1,64,59,66,255,1,66,61,65,255,1,64,62,65,255,1,66,64,65,255,1,63,63,65,255,1,63,65,62,255,1,63,64,66,255,1,64,64,66,255,1,63,63,63,255,1,64,64,64,255,1,63,65,64,255,1,64,66,65,255,1,62,63,65,255,1,63,63,63,255,1,65,66,61,255,1,63,65,62,255,1,64,64,66,255,1,65,63,66,255,1,66,62,61,255,1,66,66,68,255,1,62,64,61,255,1,66,64,65,255,1,67,63,64,255,3,64,64,64,255,}, + {1,64,64,64,255,1,62,64,63,255,1,64,64,64,255,1,68,62,64,255,1,64,63,61,255,1,64,64,62,255,1,67,62,66,255,1,67,62,68,255,1,77,55,78,255,1,89,37,83,255,1,91,22,85,255,1,70,6,84,255,1,192,156,78,255,1,251,224,73,255,1,244,217,78,255,1,247,218,80,255,1,251,217,83,255,1,247,219,76,255,1,245,216,80,255,1,246,217,79,255,1,227,196,69,255,1,137,110,83,255,1,234,219,160,255,1,216,201,144,255,1,195,182,130,255,1,178,161,118,255,1,158,141,98,255,1,144,129,88,255,1,84,63,62,255,1,60,48,60,255,1,85,74,82,255,1,82,73,78,255,1,71,64,72,255,1,69,62,69,255,1,65,59,59,255,1,64,57,64,255,1,65,60,64,255,1,65,63,64,255,1,65,64,59,255,1,65,63,66,255,2,64,64,64,255,1,63,63,61,255,1,64,64,62,255,1,65,63,68,255,4,64,64,64,255,}, + {1,65,64,62,255,1,61,66,60,255,1,65,63,66,255,1,63,63,63,255,1,61,65,64,255,1,80,57,77,255,1,98,47,90,255,1,103,32,102,255,1,104,23,100,255,1,99,24,93,255,1,86,22,83,255,1,72,5,82,255,1,187,139,37,255,1,244,201,1,255,1,237,195,0,255,1,233,189,2,255,1,228,184,0,255,1,226,186,0,255,1,228,183,2,255,1,235,192,0,255,1,225,183,0,255,1,150,117,40,255,1,236,200,26,255,1,237,200,31,255,1,240,202,31,255,1,244,206,35,255,1,242,205,36,255,1,253,217,33,255,1,147,114,45,255,1,170,141,83,255,1,255,232,128,255,1,249,225,115,255,1,246,224,115,255,1,243,219,109,255,1,184,161,83,255,1,138,116,92,255,1,217,201,152,255,1,148,131,103,255,1,54,35,55,255,1,80,62,62,255,1,131,115,82,255,1,74,62,66,255,1,63,63,61,255,1,65,63,64,255,1,65,64,59,255,1,63,65,64,255,3,64,64,64,255,}, + {1,65,63,64,255,1,64,64,64,255,2,64,64,66,255,1,86,55,87,255,1,123,30,119,255,1,116,28,115,255,1,110,27,105,255,1,101,25,100,255,1,94,24,94,255,1,85,21,82,255,1,75,14,81,255,1,124,75,58,255,1,167,125,39,255,1,168,124,37,255,1,190,151,20,255,1,227,183,0,255,1,224,182,2,255,1,219,177,3,255,1,195,156,17,255,1,184,141,28,255,1,141,100,36,255,1,228,187,1,255,1,226,181,0,255,1,224,182,2,255,1,217,176,0,255,1,222,179,2,255,1,233,184,3,255,1,152,116,32,255,1,213,171,7,255,1,231,191,0,255,1,219,173,0,255,1,192,155,14,255,1,222,180,0,255,1,223,183,0,255,1,183,145,36,255,1,244,209,19,255,1,210,172,35,255,1,38,1,45,255,1,131,97,49,255,1,255,222,32,255,1,139,109,49,255,1,64,64,64,255,1,60,66,62,255,1,63,64,68,255,4,64,64,64,255,}, + {1,63,65,64,255,1,67,63,64,255,1,60,65,61,255,1,65,63,66,255,1,104,48,93,255,1,123,29,117,255,1,115,30,113,255,1,108,26,98,255,1,103,22,98,255,1,94,22,88,255,1,88,22,84,255,1,79,19,71,255,1,66,7,73,255,1,53,1,63,255,1,36,1,59,255,1,95,54,48,255,1,226,180,0,255,1,231,182,1,255,1,171,129,29,255,1,40,5,47,255,1,36,0,50,255,1,119,82,38,255,1,229,185,0,255,1,231,182,1,255,1,141,99,41,255,1,82,28,54,255,1,93,43,54,255,1,89,42,52,255,1,93,57,43,255,1,222,176,2,255,1,233,190,0,255,1,130,90,41,255,1,14,1,47,255,1,144,105,38,255,1,223,178,0,255,1,171,134,28,255,1,233,190,0,255,1,205,160,15,255,1,43,4,48,255,1,115,80,40,255,1,242,196,0,255,1,148,112,34,255,1,63,63,65,255,2,64,64,64,255,1,64,64,66,255,3,64,64,64,255,}, + {1,65,63,64,255,1,67,62,66,255,1,61,66,62,255,1,63,63,65,255,1,96,51,92,255,1,124,30,116,255,1,115,26,110,255,1,107,26,103,255,1,101,24,94,255,1,95,22,93,255,1,85,21,82,255,1,81,17,78,255,1,70,15,72,255,1,63,16,58,255,1,47,3,56,255,1,101,63,44,255,1,222,175,1,255,1,225,178,0,255,1,158,117,29,255,1,38,0,47,255,1,33,0,49,255,1,125,84,38,255,1,226,180,0,255,1,229,178,1,255,1,106,71,43,255,1,27,0,50,255,1,35,2,47,255,1,35,0,49,255,1,79,45,44,255,1,220,174,0,255,1,229,180,0,255,1,115,75,40,255,1,21,1,52,255,1,123,82,38,255,1,212,169,2,255,1,170,128,30,255,1,230,179,2,255,1,202,157,12,255,1,33,0,47,255,1,101,65,43,255,1,238,187,0,255,1,153,118,36,255,1,65,59,63,255,1,63,65,60,255,1,64,64,64,255,1,65,63,64,255,3,64,64,64,255,}, + {1,64,64,64,255,1,65,63,64,255,1,63,65,62,255,1,62,64,63,255,1,77,60,76,255,1,90,50,85,255,1,89,49,84,255,1,84,48,84,255,1,82,47,79,255,1,78,45,76,255,1,82,23,81,255,1,77,16,73,255,1,69,16,70,255,1,64,15,62,255,1,47,3,56,255,1,108,68,43,255,1,219,171,1,255,1,221,174,0,255,1,154,111,33,255,1,37,1,47,255,1,34,0,50,255,1,126,86,37,255,1,224,174,1,255,1,218,171,3,255,1,143,106,28,255,1,79,43,43,255,1,60,27,46,255,1,35,2,45,255,1,86,48,47,255,1,216,164,0,255,1,222,175,0,255,1,116,75,43,255,1,49,0,55,255,1,68,25,52,255,1,67,34,45,255,1,140,101,36,255,1,225,173,1,255,1,209,163,7,255,1,130,90,38,255,1,157,116,28,255,1,227,178,0,255,1,165,119,33,255,1,60,58,63,255,1,64,64,64,255,1,63,64,66,255,1,65,63,64,255,3,64,64,64,255,}, + {1,64,64,64,255,2,64,64,62,255,1,63,63,63,255,2,65,65,65,255,1,64,64,62,255,1,64,64,66,255,1,65,65,67,255,1,64,62,65,255,1,83,24,82,255,1,76,19,70,255,1,69,16,68,255,1,63,14,61,255,1,44,2,52,255,1,114,74,39,255,1,217,165,1,255,1,218,167,0,255,1,147,102,35,255,1,37,1,45,255,1,35,1,51,255,1,127,85,35,255,1,220,166,5,255,1,210,160,1,255,1,217,164,0,255,1,229,171,1,255,1,144,101,33,255,1,28,0,51,255,1,84,44,45,255,1,213,159,1,255,1,218,166,4,255,1,111,67,38,255,1,40,2,49,255,1,41,5,43,255,1,17,0,52,255,1,133,90,37,255,1,221,166,1,255,1,211,158,4,255,1,222,171,0,255,1,221,169,0,255,1,213,164,1,255,1,169,123,29,255,1,62,52,60,255,1,67,63,60,255,1,60,66,64,255,1,65,64,60,255,3,64,64,64,255,}, + {1,64,64,66,255,2,64,64,64,255,1,65,63,64,255,1,64,64,64,255,1,63,65,62,255,1,65,64,62,255,1,63,63,61,255,1,65,63,64,255,1,66,61,65,255,1,84,20,80,255,1,74,20,70,255,1,66,16,67,255,1,60,15,58,255,1,41,1,51,255,1,121,78,36,255,1,214,160,0,255,1,215,161,0,255,1,139,94,35,255,1,36,0,46,255,1,35,2,49,255,1,129,86,35,255,1,214,162,0,255,1,206,155,1,255,1,196,144,9,255,1,181,133,22,255,1,113,73,38,255,1,33,0,45,255,1,84,42,43,255,1,206,152,0,255,1,212,159,1,255,1,109,67,42,255,1,43,0,53,255,1,66,24,48,255,1,69,31,44,255,1,135,90,35,255,1,210,160,0,255,1,205,151,0,255,1,180,130,19,255,1,181,131,16,255,1,213,156,4,255,1,176,124,23,255,1,60,50,61,255,1,63,65,64,255,1,65,64,69,255,1,65,64,62,255,3,64,64,64,255,}, + {2,64,64,66,255,1,64,64,64,255,1,64,66,63,255,2,64,64,64,255,1,65,63,64,255,1,64,64,64,255,1,65,63,66,255,1,69,57,69,255,1,80,19,78,255,1,72,18,70,255,1,67,14,66,255,1,59,14,57,255,1,39,1,50,255,1,126,82,35,255,2,211,155,0,255,1,134,89,34,255,2,36,0,48,255,1,132,86,36,255,1,213,154,2,255,1,211,155,0,255,1,119,76,34,255,1,27,0,55,255,1,39,4,44,255,1,37,0,46,255,1,81,43,42,255,1,201,146,1,255,1,210,150,2,255,1,109,62,42,255,1,40,0,59,255,1,118,73,42,255,1,197,140,1,255,1,153,106,28,255,1,208,148,0,255,1,193,135,10,255,1,56,17,48,255,1,71,30,44,255,1,202,146,1,255,1,178,125,11,255,1,61,51,59,255,1,61,65,66,255,1,64,63,61,255,1,64,64,66,255,3,64,64,64,255,}, + {1,64,64,62,255,3,64,64,64,255,1,63,65,64,255,1,64,64,66,255,1,63,63,61,255,1,63,65,64,255,1,63,65,62,255,1,70,56,71,255,1,79,20,74,255,1,70,17,69,255,1,63,15,63,255,1,56,15,57,255,1,38,2,50,255,1,133,86,32,255,1,209,149,1,255,1,208,148,0,255,1,125,81,34,255,1,35,1,51,255,1,37,1,49,255,1,133,86,34,255,1,210,148,1,255,1,204,147,0,255,1,120,75,36,255,1,45,9,47,255,1,64,25,46,255,1,65,26,44,255,1,84,47,41,255,1,195,139,4,255,1,204,146,0,255,1,134,84,33,255,1,37,5,44,255,1,142,94,30,255,1,196,136,3,255,1,147,98,31,255,1,206,140,0,255,1,184,129,10,255,1,53,19,44,255,1,59,20,47,255,1,192,133,5,255,1,182,127,8,255,1,60,51,56,255,1,63,63,61,255,1,64,62,63,255,1,65,63,64,255,3,64,64,64,255,}, + {2,64,64,62,255,1,65,63,64,255,1,65,63,66,255,1,64,64,66,255,1,64,64,64,255,1,65,63,64,255,1,63,65,62,255,1,64,64,62,255,1,70,53,69,255,1,81,18,73,255,1,70,16,66,255,1,66,14,60,255,1,57,15,55,255,1,40,4,52,255,1,138,90,28,255,2,204,143,1,255,1,118,72,36,255,1,36,3,50,255,1,40,2,49,255,1,135,85,32,255,1,203,142,0,255,1,196,137,1,255,1,190,130,10,255,1,180,123,7,255,1,185,132,4,255,1,191,134,3,255,1,130,80,31,255,1,184,125,7,255,1,193,133,0,255,1,192,134,1,255,1,179,123,10,255,1,195,132,1,255,1,184,125,7,255,1,146,96,27,255,1,200,134,0,255,1,184,125,5,255,1,58,20,45,255,1,52,15,46,255,1,187,121,8,255,1,193,128,0,255,1,64,48,58,255,1,64,64,66,255,1,65,64,62,255,1,61,65,66,255,3,64,64,64,255,}, + {1,65,62,69,255,1,64,64,66,255,2,63,65,62,255,1,64,64,64,255,1,64,64,62,255,1,65,63,68,255,1,62,64,63,255,1,65,64,62,255,1,70,53,69,255,1,80,16,76,255,1,67,17,68,255,1,63,14,61,255,1,52,14,53,255,1,38,5,52,255,1,142,93,27,255,2,200,136,0,255,1,112,66,40,255,1,36,3,50,255,1,40,2,49,255,1,136,85,30,255,1,199,135,1,255,1,194,129,0,255,1,190,132,0,255,1,195,132,3,255,1,194,131,0,255,1,197,135,0,255,1,120,70,35,255,1,174,115,15,255,1,195,133,0,255,1,197,130,0,255,1,199,134,4,255,1,204,136,1,255,1,162,101,18,255,1,126,75,32,255,1,202,136,0,255,1,171,112,8,255,1,52,11,43,255,1,33,1,38,255,1,129,75,13,255,1,130,74,15,255,1,60,44,55,255,1,63,64,66,255,1,65,63,66,255,4,64,64,64,255,}, + {1,64,64,66,255,1,65,63,64,255,1,64,64,62,255,1,61,65,64,255,1,64,64,66,255,1,66,62,63,255,1,65,64,62,255,1,63,64,66,255,1,63,65,62,255,1,73,52,67,255,1,73,19,71,255,1,69,17,66,255,1,59,16,62,255,1,54,12,52,255,1,41,4,48,255,1,146,93,27,255,1,196,134,1,255,1,195,132,1,255,1,106,60,37,255,1,36,4,45,255,1,38,3,45,255,1,139,89,28,255,1,201,138,0,255,1,199,132,0,255,1,199,132,2,255,1,194,131,0,255,1,193,128,0,255,1,192,127,1,255,1,87,47,35,255,1,114,65,24,255,1,162,103,1,255,1,150,94,11,255,1,138,87,24,255,1,120,70,33,255,1,63,20,40,255,1,55,18,52,255,1,84,43,37,255,1,75,37,52,255,1,72,45,64,255,1,99,71,83,255,1,126,104,106,255,1,136,117,123,255,1,61,48,57,255,1,64,63,61,255,1,65,64,62,255,1,65,63,68,255,3,64,64,64,255,}, + {1,64,64,66,255,4,64,64,64,255,1,63,65,64,255,1,64,62,65,255,1,65,64,60,255,1,63,64,68,255,1,71,47,69,255,1,73,0,73,255,1,65,6,60,255,1,60,14,60,255,1,53,13,50,255,1,42,7,49,255,1,157,102,19,255,1,203,137,0,255,1,198,131,0,255,1,99,50,45,255,1,49,0,55,255,1,48,1,55,255,1,108,62,36,255,1,153,96,25,255,1,125,78,24,255,1,109,57,20,255,1,99,50,35,255,1,80,38,40,255,1,74,36,49,255,1,65,32,61,255,1,79,54,84,255,1,113,90,106,255,1,88,65,94,255,1,27,0,43,255,1,71,43,65,255,1,192,172,135,255,1,205,192,147,255,1,107,81,82,255,1,189,156,51,255,1,236,203,72,255,1,247,214,83,255,1,249,222,89,255,1,247,220,89,255,1,82,61,56,255,1,64,64,64,255,1,63,65,62,255,4,64,64,64,255,}, + {2,63,65,64,255,1,64,64,64,255,1,67,63,64,255,1,64,64,62,255,1,61,65,64,255,1,65,63,68,255,1,76,51,73,255,1,76,48,71,255,1,79,44,74,255,1,88,111,69,255,1,97,76,85,255,1,60,1,57,255,1,50,0,49,255,1,43,0,46,255,1,81,33,33,255,1,100,51,34,255,1,81,35,38,255,1,52,7,46,255,1,62,24,61,255,1,82,47,69,255,1,53,17,53,255,1,28,2,37,255,1,89,63,92,255,1,166,152,152,255,1,89,63,92,255,1,143,115,68,255,1,202,180,97,255,1,229,206,113,255,1,247,224,122,255,1,255,233,134,255,1,240,217,124,255,1,73,46,53,255,1,103,71,50,255,1,250,209,39,255,1,247,209,38,255,1,142,107,51,255,1,214,172,2,255,1,190,151,14,255,1,95,60,38,255,1,123,86,34,255,1,233,188,1,255,1,94,69,49,255,1,64,62,65,255,1,62,66,65,255,4,64,64,64,255,}, + {1,64,64,64,255,1,63,65,64,255,1,64,64,64,255,1,67,62,66,255,1,64,64,62,255,1,67,62,69,255,1,91,37,89,255,1,94,23,91,255,1,83,5,80,255,1,74,105,63,255,1,64,170,62,255,1,81,104,86,255,1,80,44,70,255,1,123,97,96,255,1,147,125,114,255,1,165,150,143,255,1,189,176,167,255,1,179,169,157,255,1,78,46,59,255,1,204,176,69,255,1,237,213,87,255,1,120,89,60,255,1,17,0,44,255,1,168,143,79,255,1,253,243,109,255,1,147,116,70,255,1,199,161,24,255,1,248,201,1,255,1,201,163,18,255,1,170,136,29,255,1,200,165,9,255,1,246,203,0,255,1,142,108,37,255,1,90,54,42,255,1,231,187,2,255,1,232,184,0,255,1,139,99,38,255,1,203,162,12,255,1,149,109,37,255,1,15,2,45,255,1,63,24,42,255,1,218,177,1,255,1,108,77,46,255,1,64,64,64,255,1,64,64,62,255,1,64,63,68,255,3,64,64,64,255,}, + {1,64,64,62,255,1,63,65,64,255,1,63,64,66,255,1,65,63,66,255,1,64,64,62,255,1,78,54,76,255,1,98,25,96,255,1,90,21,86,255,1,84,24,78,255,1,79,74,70,255,1,60,0,62,255,1,140,96,59,255,1,235,206,88,255,1,250,220,98,255,1,254,229,100,255,1,254,232,95,255,1,250,226,100,255,1,251,226,100,255,1,149,116,63,255,1,214,174,1,255,1,241,199,0,255,1,124,88,40,255,1,22,0,47,255,1,159,119,31,255,1,240,203,1,255,1,136,98,36,255,1,189,151,18,255,1,235,190,1,255,1,91,54,45,255,1,17,0,49,255,1,74,38,42,255,1,226,181,2,255,1,180,140,29,255,1,95,57,46,255,1,222,180,0,255,1,229,183,0,255,1,139,97,37,255,1,193,149,18,255,1,159,119,32,255,1,33,0,47,255,1,57,23,47,255,1,210,164,8,255,1,121,86,44,255,1,65,63,64,255,1,68,63,67,255,1,64,64,62,255,3,64,64,64,255,}, + {1,65,63,64,255,1,64,64,64,255,1,61,65,66,255,1,64,64,66,255,1,64,64,64,255,1,87,47,82,255,1,98,22,94,255,1,90,21,84,255,1,81,35,81,255,1,77,64,71,255,1,84,34,61,255,1,224,184,9,255,1,235,195,0,255,1,226,186,0,255,1,191,152,13,255,1,169,127,27,255,1,213,171,0,255,1,239,195,2,255,1,159,123,29,255,1,207,163,14,255,1,236,191,2,255,1,121,84,40,255,1,20,0,49,255,1,156,115,35,255,1,241,197,0,255,1,130,95,37,255,1,189,144,27,255,1,226,182,0,255,1,94,57,41,255,1,34,1,44,255,1,46,10,48,255,1,194,152,18,255,1,196,155,15,255,1,104,63,43,255,1,218,170,2,255,1,226,174,0,255,1,139,98,34,255,1,183,138,23,255,1,168,122,26,255,1,34,1,46,255,1,48,10,49,255,1,195,147,11,255,1,138,96,38,255,1,62,60,61,255,1,62,64,61,255,1,65,63,68,255,3,64,64,64,255,}, + {1,64,64,64,255,1,65,63,64,255,1,63,65,64,255,1,65,63,64,255,1,64,64,64,255,1,91,42,89,255,1,95,22,91,255,1,89,16,89,255,1,82,49,76,255,1,69,67,68,255,1,102,51,58,255,1,227,185,2,255,1,232,186,2,255,1,202,160,14,255,1,49,9,44,255,1,19,1,43,255,1,172,133,30,255,1,240,198,0,255,1,155,119,35,255,1,198,155,14,255,1,233,183,2,255,1,121,80,36,255,1,24,0,48,255,1,149,110,31,255,1,236,189,0,255,1,133,93,41,255,1,183,138,23,255,1,225,178,2,255,1,94,55,40,255,1,38,3,45,255,1,41,2,47,255,1,179,136,24,255,1,207,162,7,255,1,106,66,40,255,1,210,162,0,255,1,219,168,1,255,1,137,95,35,255,1,175,126,23,255,1,173,126,22,255,1,39,3,51,255,1,71,30,44,255,1,198,144,10,255,1,149,107,31,255,1,60,55,61,255,1,62,67,63,255,1,65,63,64,255,3,64,64,64,255,}, + {1,64,64,66,255,1,65,63,64,255,1,63,65,62,255,1,65,64,62,255,1,63,65,64,255,1,90,40,89,255,1,95,24,90,255,1,87,10,88,255,1,86,65,70,255,1,66,64,65,255,1,113,57,60,255,1,225,180,0,255,1,229,183,2,255,1,188,149,22,255,1,45,9,47,255,1,37,1,49,255,1,172,130,32,255,1,239,192,1,255,1,154,116,33,255,1,197,150,18,255,1,228,176,2,255,1,114,79,39,255,1,23,0,49,255,1,143,103,34,255,1,233,185,1,255,1,131,90,38,255,1,180,134,22,255,1,221,169,0,255,1,93,55,42,255,1,38,2,50,255,1,39,3,47,255,1,171,122,29,255,1,208,159,5,255,1,108,67,39,255,1,207,152,0,255,1,213,159,0,255,1,140,95,30,255,1,162,115,23,255,1,203,145,1,255,1,171,122,20,255,1,194,140,6,255,1,208,153,0,255,1,158,107,24,255,1,60,52,63,255,1,63,61,64,255,1,63,65,62,255,3,64,64,64,255,}, + {1,64,64,62,255,1,63,63,63,255,1,63,65,64,255,1,65,63,64,255,1,64,64,64,255,1,91,37,89,255,1,90,23,90,255,1,84,5,84,255,1,81,90,69,255,1,64,64,62,255,1,119,65,53,255,1,225,180,1,255,1,227,176,0,255,1,185,143,25,255,1,45,9,47,255,1,37,1,47,255,1,169,127,29,255,1,234,186,2,255,1,156,114,30,255,1,194,147,17,255,1,222,172,0,255,1,118,76,38,255,1,23,0,49,255,1,136,96,34,255,1,230,177,3,255,1,127,86,40,255,1,179,128,19,255,1,215,163,1,255,1,91,55,43,255,1,37,1,45,255,1,38,2,46,255,1,175,128,24,255,1,206,151,7,255,1,104,65,36,255,1,203,145,1,255,1,208,151,0,255,1,140,94,34,255,1,154,102,26,255,1,203,145,1,255,1,202,144,1,255,1,200,142,0,255,1,203,142,0,255,1,167,107,21,255,1,63,71,56,255,1,65,82,63,255,1,65,79,62,255,1,61,66,62,255,1,67,63,62,255,1,64,64,66,255,}, + {1,65,63,64,255,1,65,63,66,255,1,64,64,66,255,1,63,65,62,255,1,65,63,64,255,1,96,30,94,255,1,92,21,91,255,1,82,13,78,255,1,99,143,92,255,1,72,47,79,255,1,130,77,46,255,1,225,173,1,255,1,220,177,0,255,1,179,132,26,255,1,33,0,49,255,1,24,1,47,255,1,165,122,28,255,1,231,182,1,255,1,151,113,32,255,1,190,140,15,255,1,217,168,3,255,1,111,73,36,255,1,24,0,50,255,1,133,94,37,255,1,224,172,0,255,1,128,86,38,255,1,175,121,21,255,1,213,157,2,255,1,83,43,43,255,1,23,1,50,255,1,91,53,42,255,1,206,152,0,255,1,193,138,12,255,1,92,56,40,255,1,194,136,2,255,1,206,145,3,255,1,141,92,25,255,1,148,93,29,255,1,199,137,0,255,1,198,131,0,255,1,200,132,0,255,1,200,131,2,255,1,163,101,16,255,1,46,147,45,255,1,43,172,46,255,1,59,131,55,255,1,61,66,62,255,1,67,63,62,255,1,63,64,66,255,}, + {1,63,63,65,255,1,64,64,64,255,1,63,63,63,255,1,63,65,64,255,1,69,62,70,255,1,97,24,93,255,1,92,21,87,255,1,82,21,80,255,1,78,32,79,255,1,55,3,65,255,1,139,90,47,255,1,221,170,0,255,1,214,167,0,255,1,185,140,21,255,1,98,56,40,255,1,122,86,38,255,1,198,151,9,255,1,223,171,0,255,1,152,111,32,255,1,187,132,16,255,1,217,159,0,255,1,109,68,40,255,1,9,1,50,255,1,125,83,33,255,1,219,166,2,255,1,123,81,31,255,1,172,115,25,255,1,204,150,0,255,1,131,83,34,255,1,142,92,31,255,1,197,141,2,255,1,211,152,0,255,1,162,115,25,255,1,77,41,43,255,1,191,131,0,255,1,200,136,2,255,1,146,92,28,255,1,141,82,26,255,1,197,128,1,255,1,183,115,4,255,1,151,113,14,255,1,116,128,18,255,1,74,146,36,255,1,24,180,44,255,1,55,149,52,255,1,63,114,57,255,1,63,95,58,255,1,62,68,68,255,1,64,66,65,255,}, + {1,63,65,64,255,2,64,64,64,255,1,62,64,63,255,1,71,57,72,255,1,97,24,93,255,1,91,23,84,255,1,83,19,80,255,1,78,19,75,255,1,57,3,65,255,1,150,99,42,255,1,218,165,1,255,1,212,159,5,255,1,207,159,0,255,1,216,167,2,255,1,225,171,0,255,1,215,161,0,255,1,218,162,3,255,1,152,106,29,255,1,178,129,11,255,1,212,152,2,255,1,134,87,33,255,1,89,46,40,255,1,174,125,22,255,1,211,155,0,255,1,123,78,39,255,1,166,113,21,255,1,197,142,0,255,1,200,143,2,255,1,202,144,1,255,1,201,139,6,255,1,205,145,0,255,1,121,68,36,255,1,68,23,44,255,1,196,125,0,255,1,194,124,2,255,1,131,94,24,255,1,94,88,38,255,1,115,134,27,255,1,63,147,33,255,1,99,187,103,255,1,103,200,107,255,1,32,177,46,255,1,50,164,51,255,1,48,163,44,255,1,46,170,46,255,1,39,174,46,255,1,56,140,54,255,1,65,63,64,255,}, + {1,63,65,64,255,1,64,62,63,255,1,63,65,64,255,1,66,66,66,255,1,76,55,72,255,1,95,22,95,255,1,87,10,88,255,1,83,66,76,255,1,82,107,68,255,1,55,0,67,255,1,162,113,34,255,1,211,159,0,255,1,207,154,0,255,1,210,153,1,255,1,189,139,16,255,1,163,118,24,255,1,199,146,6,255,1,214,158,0,255,1,151,104,32,255,1,170,121,18,255,1,201,143,0,255,1,196,144,0,255,1,202,144,0,255,1,203,140,0,255,1,204,146,2,255,1,119,74,33,255,1,165,107,23,255,1,194,134,1,255,1,194,131,0,255,1,195,133,0,255,1,196,134,0,255,1,189,123,3,255,1,66,66,40,255,1,63,45,45,255,1,123,111,25,255,1,90,137,25,255,1,48,171,46,255,1,0,186,40,255,1,0,182,37,255,1,0,188,41,255,1,48,197,54,255,1,58,176,54,255,1,65,99,64,255,1,64,64,62,255,1,63,64,66,255,1,60,66,66,255,1,64,94,60,255,1,67,117,56,255,1,64,98,61,255,}, + {1,65,65,67,255,1,65,64,62,255,1,63,65,62,255,1,66,64,69,255,1,81,51,77,255,1,95,24,92,255,1,87,4,82,255,1,90,92,79,255,1,110,125,102,255,1,55,0,60,255,1,170,120,23,255,1,211,152,0,255,1,210,153,4,255,1,150,103,31,255,1,52,12,46,255,1,38,2,50,255,1,174,121,15,255,1,216,157,1,255,1,150,104,29,255,1,171,115,22,255,1,197,135,0,255,1,197,133,1,255,1,192,135,2,255,1,197,135,0,255,1,198,140,4,255,1,116,74,36,255,1,163,106,19,255,1,200,131,1,255,1,196,122,0,255,1,185,117,10,255,1,158,112,16,255,1,95,124,32,255,1,0,182,38,255,1,1,173,37,255,1,50,185,65,255,1,98,199,97,255,1,55,139,51,255,1,45,166,50,255,1,27,177,41,255,1,23,166,38,255,1,43,145,45,255,1,46,152,44,255,1,45,145,46,255,1,58,125,56,255,1,61,122,52,255,1,58,135,53,255,1,49,164,43,255,1,46,168,47,255,1,60,133,54,255,}, + {1,64,64,66,255,1,65,64,62,255,1,63,65,62,255,1,64,62,63,255,1,81,48,75,255,1,94,21,90,255,1,84,20,81,255,1,79,26,80,255,1,72,20,69,255,1,73,17,62,255,1,180,126,18,255,1,202,147,2,255,1,209,149,0,255,1,114,71,37,255,1,32,0,49,255,1,45,11,48,255,1,174,119,18,255,1,211,151,1,255,1,149,101,27,255,1,166,107,17,255,1,195,132,1,255,1,189,129,0,255,1,193,128,0,255,1,197,127,3,255,1,200,130,0,255,1,111,56,36,255,1,146,92,22,255,1,156,119,15,255,1,114,123,30,255,1,66,133,36,255,1,6,155,39,255,1,1,184,44,255,1,28,191,40,255,1,24,193,40,255,1,58,178,55,255,1,70,109,64,255,1,63,63,63,255,1,66,66,66,255,1,63,114,55,255,1,46,165,47,255,1,45,168,44,255,1,42,161,43,255,1,89,160,82,255,1,66,85,63,255,1,64,76,66,255,1,64,78,61,255,1,67,72,66,255,1,65,63,66,255,1,63,65,62,255,}, + {1,63,65,62,255,1,67,62,66,255,1,63,64,66,255,1,67,78,64,255,1,88,64,78,255,1,92,11,90,255,1,86,20,82,255,1,75,17,75,255,1,67,13,65,255,1,73,28,59,255,1,188,129,13,255,1,202,141,0,255,1,201,145,0,255,1,109,61,39,255,1,38,0,52,255,1,47,11,47,255,1,169,115,17,255,1,205,149,4,255,1,146,96,27,255,1,169,102,21,255,1,197,128,0,255,1,196,125,1,255,1,188,122,0,255,1,168,115,13,255,1,142,123,28,255,1,64,134,36,255,1,48,159,41,255,1,25,169,45,255,1,2,168,44,255,1,3,180,38,255,1,25,188,37,255,1,27,191,43,255,1,31,191,39,255,1,31,189,40,255,1,55,134,51,255,1,63,63,63,255,1,65,63,64,255,1,63,63,61,255,1,64,64,66,255,1,63,65,64,255,1,62,111,56,255,1,47,161,48,255,1,66,161,59,255,1,61,77,64,255,1,65,63,64,255,1,63,65,64,255,1,64,64,64,255,1,66,64,65,255,1,64,64,62,255,}, + {1,65,63,66,255,1,67,69,66,255,1,54,145,49,255,1,6,194,37,255,1,92,82,80,255,1,90,2,89,255,1,83,17,81,255,1,78,19,75,255,1,65,10,68,255,1,83,36,52,255,1,189,131,6,255,1,194,134,2,255,1,199,137,0,255,1,97,48,41,255,1,35,1,52,255,1,50,5,44,255,1,168,114,18,255,1,215,142,3,255,1,143,87,28,255,1,104,82,33,255,1,156,117,16,255,1,122,117,23,255,1,88,130,32,255,1,39,162,37,255,1,0,189,45,255,1,8,195,44,255,1,16,196,39,255,1,53,143,55,255,1,65,67,66,255,1,62,87,57,255,1,54,119,53,255,1,38,147,38,255,1,26,177,38,255,1,31,191,41,255,1,84,198,77,255,1,60,154,58,255,1,60,114,54,255,1,61,119,58,255,1,46,153,49,255,1,83,168,77,255,1,79,116,73,255,1,63,65,64,255,1,64,64,64,255,1,62,64,63,255,1,63,63,65,255,1,65,66,61,255,3,64,64,64,255,}, + {1,64,62,63,255,1,67,68,63,255,1,62,121,57,255,1,47,167,44,255,1,95,62,79,255,1,92,27,91,255,1,83,23,77,255,1,73,13,73,255,1,59,10,65,255,1,88,43,50,255,1,195,127,2,255,1,192,129,0,255,1,194,129,1,255,1,91,68,37,255,1,42,61,42,255,1,46,28,42,255,1,143,94,27,255,1,157,120,14,255,1,89,114,30,255,1,1,142,37,255,1,0,152,38,255,1,1,171,38,255,1,0,187,40,255,1,12,188,41,255,1,29,176,34,255,1,30,188,42,255,1,28,191,40,255,1,46,172,47,255,1,63,82,62,255,1,63,65,62,255,1,65,63,66,255,1,65,64,62,255,1,60,94,57,255,1,56,157,55,255,1,96,158,93,255,1,66,108,58,255,1,60,114,56,255,1,59,138,55,255,1,57,146,52,255,1,64,155,59,255,1,90,153,83,255,1,65,88,62,255,1,63,64,59,255,1,64,64,66,255,1,64,63,68,255,1,64,64,62,255,3,64,64,64,255,}, + {1,63,65,64,255,1,62,64,61,255,1,63,63,65,255,1,64,86,65,255,1,52,157,54,255,1,46,161,40,255,1,62,152,54,255,1,76,24,72,255,1,59,0,69,255,1,100,50,53,255,1,199,132,1,255,1,195,125,1,255,1,189,110,5,255,1,78,110,35,255,1,1,180,36,255,1,35,159,39,255,1,44,148,35,255,1,2,169,40,255,1,0,183,37,255,1,24,188,40,255,1,30,193,40,255,1,41,171,47,255,1,54,124,52,255,1,51,124,45,255,1,43,136,45,255,1,30,153,37,255,1,32,174,40,255,1,27,189,41,255,1,30,186,40,255,1,50,173,49,255,1,38,169,41,255,1,99,162,91,255,1,73,91,69,255,1,68,64,65,255,1,68,62,64,255,1,62,64,63,255,1,62,66,65,255,1,64,62,63,255,1,64,64,66,255,1,63,63,63,255,1,65,63,64,255,1,64,62,65,255,1,63,65,64,255,1,64,64,62,255,1,64,64,64,255,1,63,65,64,255,3,64,64,64,255,}, + {1,64,64,64,255,1,63,68,61,255,1,69,94,62,255,1,64,127,56,255,1,18,188,38,255,1,16,190,41,255,1,8,192,34,255,1,61,171,59,255,1,66,79,62,255,1,101,50,46,255,1,164,115,12,255,1,130,119,27,255,1,83,130,34,255,1,26,162,38,255,1,30,192,44,255,1,30,195,43,255,1,17,196,43,255,1,28,193,41,255,1,38,180,42,255,1,43,176,48,255,1,30,188,41,255,1,55,137,52,255,1,64,64,62,255,1,64,64,64,255,1,65,75,64,255,1,64,89,60,255,1,58,92,57,255,1,59,114,56,255,1,53,139,48,255,1,39,160,44,255,1,28,169,39,255,1,115,190,105,255,1,90,150,88,255,1,63,76,59,255,1,50,146,46,255,1,99,172,91,255,1,72,90,68,255,1,65,64,62,255,1,63,65,64,255,1,63,65,62,255,1,63,65,64,255,1,64,65,67,255,1,64,64,66,255,1,66,65,63,255,1,63,63,61,255,1,65,65,65,255,3,64,64,64,255,}, + {1,61,114,62,255,1,43,169,46,255,1,30,190,40,255,1,28,188,40,255,1,24,191,42,255,1,32,188,43,255,1,21,191,41,255,1,2,192,32,255,1,24,186,38,255,1,45,156,36,255,1,36,160,38,255,1,2,180,40,255,1,0,188,42,255,1,28,191,40,255,1,30,183,41,255,1,30,181,40,255,1,30,187,46,255,1,50,163,49,255,1,67,72,66,255,1,66,70,69,255,1,63,89,60,255,1,55,150,50,255,1,46,169,45,255,1,40,158,44,255,1,139,206,127,255,1,128,205,127,255,1,68,88,61,255,1,65,63,64,255,1,63,69,67,255,1,63,65,64,255,1,64,66,65,255,1,63,61,66,255,1,64,64,64,255,1,67,67,65,255,1,63,85,64,255,1,67,96,68,255,1,64,76,62,255,1,64,64,62,255,1,67,63,64,255,1,65,64,62,255,1,63,63,61,255,1,64,64,64,255,1,65,63,64,255,1,64,65,67,255,1,61,65,66,255,1,64,62,63,255,3,64,64,64,255,}, + {1,36,180,48,255,1,29,189,39,255,1,29,189,41,255,1,26,191,39,255,1,16,189,37,255,1,1,187,26,255,1,45,191,48,255,1,70,196,70,255,1,25,190,38,255,1,21,195,46,255,1,20,192,44,255,1,25,192,42,255,1,27,186,40,255,1,35,161,36,255,1,34,153,37,255,1,28,156,37,255,1,27,163,37,255,1,35,177,43,255,1,64,88,62,255,1,63,64,66,255,1,65,66,60,255,1,63,126,56,255,1,47,169,50,255,1,30,186,41,255,1,33,186,46,255,1,47,173,48,255,1,63,110,58,255,1,50,155,52,255,1,93,192,84,255,1,69,104,62,255,1,64,64,62,255,1,65,63,66,255,1,67,63,64,255,1,64,64,66,255,1,63,64,59,255,1,64,64,62,255,2,64,65,67,255,1,64,64,66,255,1,65,63,66,255,1,64,63,68,255,1,59,63,64,255,1,66,65,63,255,1,66,62,61,255,1,63,63,63,255,1,64,66,65,255,3,64,64,64,255,}, + {1,57,146,52,255,1,30,188,41,255,1,27,189,42,255,1,34,189,51,255,1,64,195,63,255,1,106,205,99,255,1,190,232,186,255,1,126,199,118,255,1,57,115,56,255,1,64,99,57,255,1,54,152,49,255,1,33,191,42,255,1,49,157,46,255,1,61,78,60,255,1,63,70,63,255,1,64,78,61,255,1,62,88,61,255,1,55,121,49,255,1,50,154,43,255,1,65,107,59,255,1,62,66,65,255,1,62,63,65,255,1,61,65,68,255,1,62,74,62,255,1,65,70,64,255,1,62,66,65,255,1,66,64,67,255,1,64,91,60,255,1,65,89,63,255,1,66,73,65,255,1,63,64,66,255,2,63,65,60,255,1,60,66,66,255,1,68,64,61,255,1,63,65,64,255,1,64,62,63,255,1,64,64,62,255,1,60,64,65,255,1,64,63,61,255,1,65,64,60,255,2,64,64,64,255,1,65,67,64,255,1,64,66,63,255,4,64,64,64,255,}, + {1,64,78,63,255,1,53,155,46,255,1,32,180,42,255,1,60,180,57,255,1,118,193,110,255,1,110,190,103,255,1,73,156,66,255,1,63,90,57,255,1,64,64,64,255,1,63,63,65,255,1,65,64,62,255,1,48,162,48,255,1,60,140,55,255,2,64,64,64,255,1,68,64,65,255,1,64,64,64,255,1,64,64,66,255,1,63,75,61,255,1,57,125,52,255,1,55,150,50,255,1,57,136,53,255,1,58,139,47,255,1,102,168,97,255,1,83,113,77,255,2,65,63,64,255,1,63,64,66,255,1,63,65,62,255,1,64,64,64,255,1,65,63,64,255,1,64,64,66,255,1,64,64,64,255,1,64,64,66,255,1,64,63,68,255,1,63,64,68,255,1,65,64,62,255,1,64,64,64,255,1,63,65,64,255,1,65,63,64,255,1,64,64,62,255,1,64,64,64,255,1,64,64,66,255,1,64,64,64,255,1,63,63,63,255,1,65,63,64,255,3,64,64,64,255,}, + {1,63,63,63,255,1,63,65,62,255,1,68,70,65,255,1,66,71,64,255,1,65,72,64,255,1,64,71,63,255,1,66,64,65,255,1,60,64,65,255,1,64,64,66,255,1,65,63,66,255,1,65,64,62,255,1,55,137,53,255,1,47,155,46,255,1,65,63,64,255,1,63,65,64,255,1,65,63,68,255,1,65,63,64,255,1,63,65,60,255,1,65,65,63,255,1,64,62,63,255,1,62,81,59,255,1,63,116,60,255,1,47,155,46,255,1,86,183,80,255,1,103,188,97,255,1,63,88,58,255,1,64,64,66,255,1,65,64,62,255,1,66,64,69,255,1,63,64,66,255,19,64,64,64,255,}, + {1,64,64,66,255,1,63,63,63,255,1,65,65,63,255,1,64,64,62,255,1,63,65,62,255,1,63,65,64,255,1,64,65,60,255,1,67,62,66,255,1,61,66,62,255,1,65,65,65,255,1,64,112,60,255,1,30,186,40,255,1,68,177,70,255,1,64,66,63,255,1,64,65,60,255,1,62,64,63,255,1,63,64,66,255,1,65,63,64,255,1,65,64,62,255,1,65,65,65,255,2,64,64,64,255,1,65,64,62,255,1,64,64,64,255,1,62,72,61,255,1,65,65,63,255,1,64,64,64,255,1,65,64,62,255,1,66,64,67,255,1,65,64,60,255,19,64,64,64,255,}, + {1,64,64,66,255,2,64,64,64,255,1,65,63,64,255,2,65,63,66,255,1,67,62,68,255,2,65,63,66,255,1,65,73,60,255,1,19,181,34,255,1,99,201,92,255,1,161,201,151,255,1,64,66,63,255,1,66,64,65,255,1,63,65,62,255,1,61,66,62,255,1,64,64,66,255,1,63,62,67,255,1,64,64,64,255,1,64,64,62,255,1,64,64,64,255,1,63,64,66,255,1,63,64,68,255,1,64,63,61,255,1,66,65,63,255,1,64,66,65,255,1,64,64,62,255,1,60,64,65,255,1,63,65,64,255,19,64,64,64,255,}, + {5,64,64,64,255,1,63,63,63,255,1,65,66,61,255,1,62,63,65,255,1,63,65,62,255,1,65,72,65,255,1,49,151,42,255,1,78,185,71,255,1,74,117,71,255,1,63,65,62,255,1,65,63,64,255,1,65,64,62,255,1,64,64,62,255,1,63,64,66,255,1,64,65,67,255,2,64,64,62,255,2,63,65,64,255,1,63,65,62,255,1,67,66,62,255,1,63,63,65,255,1,64,63,68,255,1,64,66,65,255,1,63,63,65,255,1,68,63,67,255,19,64,64,64,255,}, + }, + bytes = table.concat({ +string.char(137,80,78,71,13,10,26,10,0,0,0,13,73,72,68,82,0,0,0,49,0,0,0,41,8,2,0,0,0,19,123,86,189,0,0,0,117,116,69,88,116,82,97,119,32,112,114,111,102,105,108,101,32,116,121,112,101,32,65,80,80,49,0,10,103,101,110,101,114,105,99,32,112,114,111,102,105,108,101,10,32,32,32,32,32,32,51,52,10,52,57,52,57,50,97,48,48,48,56,48), +string.char(48,48,48,48,48,48,49,48,48,51,49,48,49,48,50,48,48,48,55,48,48,48,48,48,48,49,97,48,48,48,48,48,48,48,48,48,48,48,48,48,48,53,48,54,57,54,51,54,49,55,51,54,49,48,48,48,48,10,230,102,122,210,0,0,17,182,73,68,65,84,120,156,157,88,105,120,148,85,150,190,203,183,85,85,170,82,169,44,149,132,4,194,14,34,139,2,14,123,22,22), +string.char(145,25,181,237,166,29,109,181,157,118,198,5,165,181,155,113,68,7,109,113,233,118,199,5,27,241,81,108,68,68,116,4,55,180,167,7,198,22,81,20,69,5,4,130,52,134,45,33,59,149,90,191,237,46,125,110,37,35,225,143,63,166,158,122,178,124,219,61,223,123,222,243,158,247,92,82,91,215,160,190,245,117,240,173,171,159,93,87,59,103,70,93,125,61,28,154,89,95,95,55), +string.char(107,222,180,57,23,158,51,255,217,59,151,126,185,117,253,230,53,143,253,241,225,59,127,247,155,235,23,93,123,249,181,151,95,182,240,226,11,47,110,152,55,125,234,180,250,169,115,234,166,55,192,245,117,179,234,27,234,234,107,107,235,107,235,103,193,3,234,224,177,181,181,117,117,249,39,55,192,33,56,85,91,223,48,19,206,170,255,107,27,26,102,205,108,104,152,51,171,110,38,220), +string.char(53,91,93,216,247,33,24,9,137,56,150,72,231,8,9,15,97,207,20,152,72,45,228,133,203,91,74,75,143,20,201,35,5,231,14,107,156,48,252,213,121,51,118,92,247,211,67,247,44,58,189,226,238,220,170,251,229,186,199,45,248,121,201,228,138,215,215,93,189,122,197,37,247,221,85,191,228,95,39,95,125,241,224,203,26,226,165,2,241,30,175,52,24,45,14,22,154,200,50,184), +string.char(142,93,143,120,146,112,140,56,37,82,45,39,177,16,132,114,238,18,132,165,164,112,230,135,79,62,38,184,86,18,31,75,78,104,212,45,28,148,44,31,118,98,200,128,99,101,193,116,208,71,130,72,225,39,26,221,246,255,118,59,222,77,157,124,183,253,232,166,206,163,91,18,71,223,205,182,190,35,210,91,195,230,1,210,253,159,165,242,201,177,37,107,26,198,110,90,88,247,241,175), +string.char(175,216,85,59,49,113,235,53,241,55,94,26,247,218,234,225,111,173,29,179,121,205,168,165,55,148,47,152,94,181,250,137,43,126,54,103,100,246,36,43,148,108,246,196,88,80,16,136,0,150,70,216,150,253,99,66,146,80,161,193,33,140,105,69,46,62,166,229,156,88,119,133,233,83,78,144,175,222,64,210,128,220,254,81,233,123,91,70,108,253,203,240,3,135,99,82,34,193,136,237), +string.char(225,67,71,138,26,15,86,217,9,230,164,91,179,221,199,123,186,78,182,156,248,190,235,84,83,215,209,19,58,73,143,25,184,165,173,113,249,137,221,79,124,177,245,133,206,239,86,142,44,219,52,182,250,243,1,214,178,89,163,183,245,52,229,38,14,77,223,125,203,215,245,227,19,176,168,79,133,64,1,64,238,76,76,16,43,149,136,97,128,9,197,92,3,73,128,77,8,233,18,46), +string.char(32,149,20,19,77,207,237,220,86,184,226,193,242,187,127,95,246,242,159,170,36,71,38,146,123,246,84,94,191,104,224,226,127,47,235,56,77,176,144,58,198,155,54,141,188,107,241,5,247,222,57,233,55,75,206,251,244,211,80,64,247,137,68,175,109,26,125,201,13,131,223,125,127,4,188,95,38,235,100,187,187,253,76,151,78,105,79,194,78,159,58,22,160,73,41,177,38,32,87,76), +string.char(161,243,67,76,68,32,128,10,99,44,144,116,168,196,16,26,227,192,46,65,24,6,142,73,65,132,169,89,50,88,232,135,8,55,2,140,249,216,99,50,157,65,194,71,28,73,131,32,14,9,102,232,251,102,178,171,153,237,108,116,191,56,200,83,25,32,139,96,57,148,76,74,19,233,157,237,40,107,19,233,17,158,37,156,243,28,147,169,36,101,54,210,13,66,152,144,152,229,51,40), +string.char(206,196,4,89,19,24,72,163,72,215,29,232,105,42,57,126,42,126,170,189,172,187,59,158,76,196,19,156,72,38,125,56,133,165,6,87,155,186,228,12,49,248,122,249,116,67,194,129,110,30,206,186,100,248,208,228,164,49,110,69,133,68,132,153,148,66,64,46,228,30,209,42,73,119,125,22,185,237,174,49,39,90,12,223,147,166,225,255,234,23,246,148,137,201,172,141,10,48,247,37), +string.char(68,163,201,126,32,161,94,196,242,75,194,57,148,209,179,173,133,93,173,133,29,93,209,211,137,88,38,85,148,5,252,160,38,160,32,129,88,92,50,195,96,80,154,220,69,174,13,55,10,172,78,11,230,104,78,86,44,152,214,188,106,249,193,133,11,210,174,48,4,19,220,199,204,214,124,199,183,137,215,210,101,127,221,100,186,89,204,92,68,124,126,245,101,135,235,38,55,167,82,88), +string.char(211,24,241,56,149,140,67,178,242,104,253,16,83,223,55,95,147,128,160,170,74,41,33,31,128,17,92,41,4,86,57,226,152,115,41,77,141,195,171,113,15,57,14,92,3,48,41,46,120,204,103,62,234,236,10,236,57,92,122,188,89,215,145,135,132,112,93,233,185,64,73,13,10,168,162,88,159,52,200,33,72,100,61,56,200,14,54,22,31,58,28,69,14,53,177,7,79,103,192,88), +string.char(161,159,197,167,51,104,81,217,87,153,138,94,20,14,82,166,49,9,196,128,165,101,62,92,169,107,18,216,195,60,228,3,232,72,177,155,194,29,14,209,145,124,111,91,197,13,203,6,127,176,213,208,16,6,105,241,61,44,108,140,20,231,140,73,231,121,203,127,125,32,22,117,69,6,101,122,130,15,62,89,253,204,11,85,217,28,39,56,3,52,22,192,31,233,72,164,245,143,9,64), +string.char(203,243,11,18,136,243,149,136,184,34,25,112,92,2,219,128,132,146,1,72,24,82,140,0,39,96,15,228,206,6,156,48,85,185,3,60,61,233,58,234,101,98,1,167,162,148,84,85,10,170,1,30,212,99,162,162,204,230,122,178,188,52,35,33,75,14,21,14,178,33,245,132,101,114,52,151,214,52,2,197,4,121,243,21,10,253,57,142,250,194,81,15,85,32,33,37,14,189,71,16), +string.char(135,162,3,242,67,202,185,10,146,16,93,99,190,135,160,244,128,79,192,110,138,225,29,133,235,73,39,135,231,253,195,169,135,150,28,186,235,150,131,191,91,220,56,110,112,46,155,162,217,52,154,118,254,177,21,119,52,77,30,113,50,147,213,128,120,158,7,8,248,32,28,174,47,253,12,13,97,71,3,176,176,14,143,98,164,95,238,176,48,17,2,141,0,165,239,5,12,190,26,112), +string.char(165,87,87,165,0,213,39,28,97,95,42,98,105,64,72,87,125,61,175,15,106,136,149,185,88,0,246,36,87,24,96,1,168,3,204,115,57,45,151,179,96,17,157,201,146,144,23,208,80,71,91,168,171,83,135,117,211,105,234,217,162,59,229,236,220,167,111,255,162,144,41,221,129,101,224,193,253,181,64,53,22,133,138,232,101,58,172,140,61,168,97,248,13,215,185,212,39,74,183,49), +string.char(0,6,241,64,32,76,96,159,97,207,83,113,2,65,41,188,14,149,58,69,1,69,11,14,138,234,59,248,104,59,250,228,235,192,209,19,120,239,145,208,142,189,37,111,126,88,179,230,173,178,191,124,86,242,232,43,195,215,190,127,110,201,144,210,134,121,99,121,245,53,198,152,197,88,131,228,128,102,176,94,173,233,175,5,234,143,60,197,5,167,16,33,37,80,59,210,16,16,79,55), +string.char(97,89,160,171,80,122,65,113,40,224,152,20,130,147,110,206,10,179,96,80,4,143,183,88,123,14,86,124,240,69,249,134,247,99,47,191,53,244,201,13,163,30,125,117,66,71,178,242,251,236,140,111,210,139,62,62,249,179,167,95,47,254,166,165,230,190,213,215,174,218,124,109,124,196,216,37,75,229,250,151,172,103,159,68,203,238,72,220,118,99,83,105,24,72,163,222,95,246,215,2), +string.char(88,172,183,255,245,198,4,8,66,209,133,210,225,112,15,181,184,94,120,27,171,153,87,129,210,156,167,37,118,249,150,173,145,231,223,24,242,216,218,17,123,15,224,112,185,49,100,92,100,215,183,69,15,172,43,56,146,156,51,245,218,7,115,225,169,181,23,87,45,127,106,194,250,205,147,54,190,49,98,197,147,85,171,87,15,28,55,46,88,83,149,156,54,124,237,57,69,43,135,86), +string.char(30,175,136,126,179,111,247,206,155,110,248,246,254,59,119,165,142,109,44,176,220,124,109,209,179,52,243,76,243,83,4,71,224,43,42,79,149,198,154,99,230,129,128,63,51,219,54,221,237,169,194,37,99,194,23,253,106,252,77,247,204,89,248,31,183,94,121,207,239,175,251,195,210,202,145,225,21,171,173,23,54,196,102,207,55,179,194,188,244,39,233,127,190,234,192,180,201,29,83,198), +string.char(237,31,87,186,182,249,192,203,239,109,124,101,215,182,7,140,212,131,115,103,245,164,19,78,119,199,223,146,173,135,185,147,206,230,130,141,7,227,111,239,52,63,249,216,74,102,81,196,226,138,62,2,20,240,172,220,41,114,1,69,69,30,173,144,99,154,25,147,9,31,23,11,253,184,201,46,71,211,157,150,23,214,197,238,93,70,110,185,41,117,229,79,247,204,60,255,127,231,79,253), +string.char(236,194,121,90,80,108,211,82,175,107,178,75,71,212,110,219,147,62,244,108,246,116,83,46,231,165,179,232,153,149,3,110,188,179,228,254,251,6,119,117,144,17,53,142,78,53,219,21,138,204,66,230,210,70,48,232,199,4,184,41,221,73,155,225,128,162,178,38,117,32,113,127,156,20,84,249,140,170,127,3,96,13,184,170,62,30,162,34,135,203,135,200,133,87,124,141,178,27,219,154), +string.char(94,218,254,222,235,255,245,220,182,191,237,122,49,115,252,165,160,217,228,102,195,190,67,160,253,65,113,112,166,97,208,104,80,12,55,96,59,18,218,93,17,246,155,155,141,99,39,163,209,194,84,192,224,44,71,184,13,180,224,185,172,30,160,30,180,36,59,77,146,9,90,20,244,160,43,123,80,85,210,232,207,39,213,233,136,202,170,2,76,243,77,14,154,9,34,18,241,29,202,99), +string.char(69,64,122,151,219,164,237,84,100,201,111,7,221,124,127,180,241,240,64,32,37,99,6,200,38,119,160,14,161,24,165,11,221,197,129,90,213,60,240,12,12,30,38,161,158,124,71,38,123,140,144,233,128,157,116,28,144,49,44,56,118,50,8,68,78,39,122,188,4,60,154,12,133,92,144,76,68,123,165,251,172,126,167,216,141,242,100,87,129,169,86,47,181,168,122,131,144,1,50,43), +string.char(56,71,169,172,197,16,141,33,64,17,92,7,88,82,33,125,205,135,102,172,90,18,6,213,230,12,150,228,28,66,178,133,122,71,169,3,246,78,74,199,32,188,112,121,14,121,54,52,34,31,50,59,176,184,243,193,165,199,151,221,122,56,24,200,5,144,11,15,4,152,4,57,75,199,25,52,178,94,195,169,212,28,4,6,4,31,222,187,66,245,99,3,234,2,130,5,54,120,189,165), +string.char(161,214,133,94,70,212,111,9,221,67,203,107,158,11,79,118,8,99,16,46,247,96,1,213,206,125,104,145,154,225,11,56,46,56,232,42,0,9,68,118,210,234,57,131,202,186,13,226,249,89,80,53,78,139,93,82,42,73,68,117,153,188,114,18,197,39,104,38,240,19,34,85,147,130,171,195,111,97,121,177,201,113,43,81,20,42,0,40,144,11,205,63,139,51,54,73,171,222,0,22), +string.char(10,22,36,12,80,103,20,122,63,36,159,11,234,249,60,175,233,4,26,34,36,40,133,244,201,19,217,160,242,182,116,90,207,216,154,109,99,101,90,133,237,217,148,57,146,165,37,202,33,176,120,204,243,19,255,148,237,248,69,194,155,228,246,85,127,190,227,98,37,81,32,192,136,104,210,0,170,170,212,90,164,243,233,116,98,127,42,16,198,160,254,94,14,25,150,95,63,197,190,238), +string.char(231,246,224,234,110,39,11,119,0,48,24,217,92,248,90,18,114,148,133,62,72,124,223,229,158,42,177,9,163,236,165,87,117,223,124,197,33,176,32,199,219,195,201,36,229,190,128,220,65,133,186,89,14,126,33,217,163,125,211,29,121,147,71,95,27,91,210,49,88,22,180,27,90,130,144,106,232,133,6,244,62,2,236,81,109,11,43,243,68,252,188,172,67,183,1,8,211,192,85,20), +string.char(10,248,224,206,132,163,151,21,119,222,113,219,225,27,127,126,184,40,104,251,54,16,131,219,28,98,32,21,197,238,79,198,230,206,59,7,204,54,132,105,121,128,148,131,230,206,56,54,99,210,9,13,40,239,162,198,189,33,251,52,214,61,228,101,80,146,176,189,70,193,102,163,242,225,154,234,221,11,174,76,207,191,242,224,160,96,101,143,233,70,156,196,148,108,219,165,25,111,126,10), +string.char(243,188,78,42,23,9,253,149,80,29,198,2,104,197,105,13,141,103,218,21,64,34,82,16,113,160,237,251,190,15,245,219,210,28,249,248,203,242,158,164,161,26,167,160,142,99,102,109,49,107,234,209,223,94,119,168,186,176,3,156,2,99,158,50,221,14,246,50,194,205,194,93,94,79,7,221,37,52,191,12,255,57,85,177,58,28,223,116,113,217,186,235,200,19,19,130,219,71,91,95), +string.char(5,191,249,182,237,203,193,239,198,34,47,134,10,215,148,69,86,151,84,188,29,9,181,19,57,148,231,201,164,160,130,112,168,206,116,208,26,228,121,102,37,173,138,13,1,238,90,102,22,248,68,165,246,201,215,3,151,222,55,240,129,199,43,91,91,3,160,31,106,76,245,52,6,182,41,139,115,96,111,125,184,9,238,164,2,124,1,40,179,141,114,61,193,195,29,209,117,184,250,195), +string.char(219,197,199,203,157,63,47,24,101,205,154,111,30,211,199,118,5,7,248,90,188,131,251,187,237,76,91,174,107,220,105,111,166,39,166,165,196,220,116,114,94,162,115,162,205,134,186,208,204,251,154,139,80,78,91,64,145,243,168,204,126,136,15,63,119,52,86,68,130,134,111,195,80,196,216,145,99,70,87,134,154,1,21,58,88,40,184,12,188,182,210,27,6,125,80,128,145,37,57,230), +string.char(159,70,137,20,221,159,137,108,198,3,31,137,86,222,52,38,190,110,70,65,85,128,196,75,168,229,230,154,55,28,181,182,20,148,190,24,29,182,61,18,151,112,84,130,179,57,183,112,152,117,200,164,251,34,198,103,86,108,125,188,248,185,98,186,205,82,78,146,8,77,57,152,40,54,163,160,221,160,10,96,239,129,43,12,133,184,69,1,1,225,48,228,231,136,161,164,77,250,76,122), +string.char(96,170,124,68,125,141,48,105,194,100,232,161,147,153,224,78,183,106,239,37,232,165,217,197,247,156,63,240,169,177,145,237,227,73,241,96,81,105,130,38,193,53,136,239,103,238,62,53,240,130,146,224,175,67,69,47,22,21,125,86,160,35,189,29,183,131,183,193,105,33,211,32,172,48,116,168,20,16,21,144,137,236,203,18,59,110,111,209,98,68,216,18,132,7,22,7,215,18,180,92), +string.char(76,92,7,202,219,65,153,180,26,246,169,7,131,2,14,186,146,219,248,175,93,5,141,217,232,7,78,217,139,86,205,189,227,203,151,52,20,125,62,211,58,54,92,84,157,10,214,190,80,54,253,149,226,234,173,209,208,126,223,79,193,34,184,186,174,146,95,229,251,19,115,98,10,63,125,101,162,228,198,112,145,94,20,93,27,21,127,210,113,26,68,9,19,16,70,69,32,37,5,202), +string.char(136,65,38,172,54,107,116,139,73,187,36,169,32,200,128,194,86,137,212,77,1,124,6,29,135,87,241,74,152,57,35,199,226,162,181,43,176,179,35,244,197,0,242,252,12,237,223,134,150,61,116,65,225,198,169,86,75,165,86,157,206,150,185,162,0,108,113,181,35,10,25,238,208,67,159,135,7,108,42,31,178,169,184,162,41,82,112,40,106,236,83,18,45,138,121,200,165,221,233,211), +string.char(161,225,65,18,7,197,247,21,161,149,125,60,179,97,160,76,181,15,222,113,167,85,244,86,233,129,107,91,247,253,225,171,206,251,218,105,8,52,77,43,48,169,166,75,10,217,233,214,210,231,136,198,165,94,246,1,190,230,2,235,250,243,42,95,155,107,198,199,209,129,163,201,40,20,184,96,123,193,200,143,180,200,161,16,250,222,215,18,168,187,200,59,122,83,71,122,113,186,229,230), +string.char(214,33,215,85,135,121,65,193,43,193,230,181,39,217,126,157,236,9,232,91,44,115,115,68,174,215,90,55,116,203,83,48,245,232,74,14,49,176,178,191,31,135,126,162,33,99,16,71,163,237,166,65,61,39,170,241,23,163,186,146,43,211,230,165,160,222,238,247,137,208,95,189,202,85,37,213,187,39,209,234,2,203,31,166,181,76,176,42,203,241,8,174,23,103,44,228,227,72,147,30), +string.char(252,36,24,254,164,184,108,71,241,144,143,226,149,155,139,106,222,47,140,239,181,178,105,63,135,68,180,42,82,52,189,216,47,240,78,47,60,61,124,113,85,176,92,213,184,18,92,104,169,68,181,94,213,69,148,52,114,160,218,153,152,248,249,162,235,250,228,119,255,210,113,100,102,143,65,181,168,32,97,105,126,90,218,114,240,150,182,207,111,206,45,31,91,185,124,102,120,211,52,139), +string.char(84,19,80,109,148,226,97,155,150,157,176,170,159,45,143,175,42,29,188,166,24,133,197,233,69,137,182,95,118,45,120,160,182,238,246,217,65,166,25,95,89,193,255,41,46,127,185,100,216,243,149,187,239,253,238,212,27,109,154,109,132,119,196,142,109,105,119,82,170,75,104,34,175,211,34,63,201,194,232,161,38,20,152,249,251,245,224,214,134,84,170,196,85,193,83,233,8,255,184,76), +string.char(148,35,115,142,54,2,192,107,27,25,236,169,49,226,5,100,128,16,230,9,166,157,96,162,149,243,54,59,25,241,120,148,113,143,147,118,100,181,25,24,233,185,42,190,170,235,173,183,142,189,227,151,241,228,213,153,129,203,138,140,145,66,77,210,249,77,27,53,28,118,250,252,152,135,178,224,221,116,24,132,160,204,97,88,84,155,5,106,70,241,148,207,212,250,229,174,180,73,83,13), +string.char(24,97,75,20,76,48,42,23,197,46,122,122,244,163,83,232,132,154,247,99,99,214,134,134,28,210,45,137,202,78,26,149,155,202,171,55,85,14,123,183,124,232,251,229,37,159,27,169,134,110,50,198,229,72,151,219,3,145,213,177,154,167,74,107,30,143,21,254,177,136,238,11,68,62,178,154,159,239,228,199,33,24,0,192,99,202,46,32,93,25,53,130,192,1,130,152,192,88,15,89), +string.char(199,142,174,246,227,48,12,61,202,142,139,51,27,80,196,220,22,138,31,213,117,27,85,218,198,5,98,188,150,212,30,219,242,240,246,149,59,204,175,44,121,196,10,109,12,23,158,52,66,137,0,77,129,147,196,90,74,35,109,52,240,89,81,248,157,152,19,115,221,133,73,255,162,12,95,208,57,234,151,67,99,99,194,224,209,212,56,213,98,136,54,9,154,66,212,136,102,192,144,4), +string.char(201,2,33,83,216,168,156,17,170,230,109,250,127,123,41,36,95,247,232,172,61,49,153,68,193,13,209,97,207,196,141,149,230,151,15,237,219,247,200,33,119,189,228,167,172,222,93,13,184,194,108,213,211,133,118,126,7,18,12,40,205,255,225,145,28,49,183,135,204,173,1,163,203,204,70,101,79,97,162,228,31,75,68,133,7,96,16,196,168,52,101,222,193,246,77,253,202,48,50,12), +string.char(129,161,222,205,194,62,159,221,119,170,215,1,156,181,159,9,192,9,12,94,2,140,27,124,193,67,194,80,105,194,17,24,139,161,70,38,136,214,243,210,165,101,229,96,39,225,161,92,205,153,121,153,5,2,82,66,18,58,222,173,197,94,141,118,60,210,211,248,248,17,210,174,255,16,132,154,16,255,191,31,210,23,166,218,160,83,139,17,181,25,107,169,49,156,107,200,242,191,157,123), +string.char(106,110,225,164,107,202,47,203,67,170,171,237,38,162,180,68,8,61,223,40,69,254,77,53,160,14,244,93,220,167,45,249,205,30,172,253,248,194,63,30,83,95,17,42,51,174,118,94,184,208,128,188,106,109,240,212,3,79,88,157,61,157,175,191,249,182,204,127,128,15,52,175,255,249,33,81,244,222,133,148,215,245,40,85,206,27,172,44,16,134,170,189,71,250,99,203,254,232,231,239), +string.char(205,144,88,185,221,172,58,132,0,0,0,0,73,69,78,68,174,66,96,130), + }) +} diff --git a/reascripts/ReaSpeech/resources/images/heading-logo-tech-audio.png b/reascripts/ReaSpeech/resources/images/heading-logo-tech-audio.png new file mode 100644 index 0000000000000000000000000000000000000000..efc6779f7025ea6cccfd036c300cdc854ceba931 Binary files /dev/null and b/reascripts/ReaSpeech/resources/images/heading-logo-tech-audio.png differ diff --git a/reascripts/ReaSpeech/resources/images/reaspeech-logo-small.lua b/reascripts/ReaSpeech/resources/images/reaspeech-logo-small.lua new file mode 100644 index 0000000000000000000000000000000000000000..22ea734074b2fc7223291d477f1a315524d17430 --- /dev/null +++ b/reascripts/ReaSpeech/resources/images/reaspeech-logo-small.lua @@ -0,0 +1,261 @@ +IMAGES = IMAGES or {} +IMAGES['reaspeech-logo-small'] = { + width = 142, + height = 100, + pixels = { + {142,0,0,0,0,}, + {39,0,0,0,0,1,0,0,0,38,1,0,0,0,81,1,0,0,0,118,1,0,0,0,153,1,0,0,0,186,1,0,0,0,203,2,0,0,0,204,1,0,0,0,173,1,0,0,0,120,1,0,0,0,71,1,0,0,0,10,91,0,0,0,0,}, + {36,0,0,0,0,1,0,0,0,63,1,0,0,0,157,1,0,0,0,231,1,0,0,0,255,1,23,63,42,255,1,52,123,86,255,1,70,159,112,255,1,78,176,124,255,1,86,193,137,255,1,91,203,144,255,1,89,200,142,255,1,76,171,121,255,1,56,130,91,255,1,6,25,15,255,1,0,0,0,255,1,0,0,0,208,1,0,0,0,142,1,0,0,0,80,1,0,0,0,25,11,0,0,0,0,1,0,0,0,3,1,0,0,0,84,1,0,0,0,155,1,0,0,0,214,15,0,0,0,255,1,0,0,0,232,1,0,0,0,201,1,0,0,0,151,1,0,0,0,89,1,0,0,0,9,7,0,0,0,0,1,0,0,0,10,1,0,0,0,51,1,0,0,0,98,1,0,0,0,115,1,0,0,0,153,1,0,0,0,175,1,0,0,0,204,1,0,0,0,219,2,0,0,0,255,1,0,0,0,206,1,0,0,0,81,33,0,0,0,0,}, + {33,0,0,0,0,1,0,0,0,51,1,0,0,0,144,1,0,0,0,232,1,0,0,0,255,1,63,145,102,255,1,94,209,148,255,1,109,241,172,255,6,116,255,182,255,2,109,241,172,255,1,116,255,182,255,1,114,251,179,255,1,102,225,160,255,1,86,193,137,255,1,62,144,101,255,1,27,71,48,255,1,0,0,0,255,1,0,0,0,178,1,0,0,0,59,7,0,0,0,0,1,0,0,0,10,1,0,0,0,112,1,0,0,0,227,1,27,72,48,255,1,67,153,108,255,1,89,198,141,255,1,102,226,161,255,5,104,231,165,255,3,100,221,157,255,1,103,227,162,255,4,104,231,165,255,1,104,230,164,255,1,94,210,149,255,1,83,187,132,255,1,66,152,107,255,1,31,78,53,255,1,0,0,0,224,1,0,0,0,14,2,0,0,0,0,1,0,0,0,72,1,0,0,0,103,1,0,0,0,143,1,0,0,0,180,1,0,0,0,250,1,0,0,0,255,1,42,102,70,255,1,53,124,86,255,1,72,164,116,255,1,76,171,121,255,1,88,197,140,255,1,91,203,144,255,2,100,221,157,255,1,78,177,125,255,1,0,2,1,255,1,0,0,0,113,32,0,0,0,0,}, + {31,0,0,0,0,1,0,0,0,61,1,0,0,0,192,1,0,0,0,255,1,61,141,99,255,1,91,204,145,255,1,112,246,175,255,6,116,255,182,255,1,112,246,175,255,1,78,176,125,255,1,31,79,54,255,2,0,0,0,255,1,55,129,90,255,1,106,234,167,255,4,116,255,182,255,1,102,227,161,255,1,71,161,114,255,1,0,0,0,255,1,0,0,0,169,1,0,0,0,33,3,0,0,0,0,1,0,0,0,10,1,0,0,0,126,1,0,0,0,235,1,32,81,55,255,1,88,197,139,255,1,113,248,177,255,20,116,255,182,255,1,114,250,178,255,1,51,120,84,255,1,0,0,0,206,1,0,0,0,156,1,0,0,0,231,1,8,32,19,255,1,52,123,86,255,1,64,146,103,255,1,79,179,127,255,1,99,220,157,255,1,114,252,180,255,9,116,255,182,255,1,97,216,154,255,1,0,0,0,255,1,0,0,0,52,31,0,0,0,0,}, + {30,0,0,0,0,1,0,0,0,136,1,0,0,0,255,1,71,163,115,255,1,107,236,168,255,8,116,255,182,255,1,112,246,175,255,1,39,96,66,255,1,113,89,17,255,1,206,165,39,255,1,190,152,36,255,1,198,158,38,255,1,154,123,27,255,1,20,56,37,255,1,96,213,152,255,5,116,255,182,255,1,106,233,166,255,1,58,135,94,255,1,0,0,0,247,1,0,0,0,145,1,0,0,0,28,1,0,0,0,84,1,0,0,0,217,1,36,89,61,255,1,91,203,144,255,1,114,251,179,255,23,116,255,182,255,1,110,243,173,255,1,78,177,125,255,1,75,170,120,255,1,87,196,139,255,1,111,246,175,255,15,116,255,182,255,1,71,161,114,255,1,0,0,0,198,31,0,0,0,0,}, + {29,0,0,0,0,1,0,0,0,142,1,22,60,40,255,1,100,222,158,255,10,116,255,182,255,1,80,180,128,255,1,107,84,15,255,1,250,201,50,255,1,255,205,51,255,1,202,162,39,255,1,222,178,43,255,1,255,205,51,255,1,190,152,36,255,1,20,56,37,255,1,106,235,167,255,6,116,255,182,255,1,96,213,151,255,1,39,96,66,255,1,0,0,0,255,1,19,55,37,255,1,78,176,124,255,1,114,250,179,255,45,116,255,182,255,1,108,239,171,255,1,0,0,0,255,1,0,0,0,61,30,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,84,1,14,43,28,255,1,102,226,161,255,7,116,255,182,255,1,114,250,178,255,1,95,211,150,255,1,71,161,114,255,1,53,124,86,255,1,5,25,14,255,1,202,162,38,255,1,255,205,51,255,1,250,201,50,255,1,56,43,4,255,1,196,157,37,255,1,113,89,17,255,2,0,0,0,255,1,55,129,90,255,1,108,239,170,255,6,116,255,182,255,1,114,252,180,255,1,104,231,165,255,1,111,245,175,255,7,116,255,182,255,1,109,241,172,255,1,91,203,144,255,1,87,195,138,255,1,85,190,135,255,1,76,171,121,255,1,75,170,120,255,1,74,169,119,255,3,75,170,120,255,1,76,171,121,255,1,88,196,139,255,1,91,203,144,255,1,93,206,146,255,1,103,228,162,255,1,112,247,176,255,25,116,255,182,255,1,69,157,110,255,1,0,0,0,171,30,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,213,1,79,178,126,255,5,116,255,182,255,1,111,245,175,255,1,85,191,135,255,1,51,121,84,255,1,0,0,0,255,1,15,5,18,255,1,107,63,119,255,1,133,79,146,255,1,0,0,0,255,1,237,190,47,255,1,255,205,51,255,1,207,166,39,255,1,54,41,4,255,1,158,126,28,255,1,73,41,81,255,1,63,35,71,255,3,0,0,0,255,1,62,143,100,255,1,97,215,153,255,1,115,253,181,255,10,116,255,182,255,1,108,237,169,255,1,78,177,125,255,1,50,119,83,255,1,0,0,0,255,1,51,28,57,255,2,112,66,124,255,1,101,59,112,255,1,73,42,81,255,1,86,50,96,255,1,68,38,75,255,1,23,10,26,255,1,99,58,110,255,1,23,10,26,255,6,0,0,0,255,1,42,101,70,255,1,86,192,136,255,1,116,254,181,255,2,116,255,182,255,1,116,254,181,255,1,105,233,166,255,1,104,231,165,255,1,103,228,162,255,1,92,205,146,255,1,91,203,144,255,1,89,198,141,255,1,75,170,120,255,1,73,167,118,255,1,56,131,92,255,1,49,115,80,255,3,0,0,0,255,1,58,134,94,255,1,110,242,172,255,3,116,255,182,255,1,103,228,162,255,1,0,0,0,255,1,0,0,0,37,29,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,55,1,0,0,0,255,1,111,245,175,255,4,116,255,182,255,1,101,225,160,255,1,20,56,37,255,1,84,48,93,255,1,170,102,187,255,1,201,122,221,255,1,224,137,247,255,1,232,142,255,255,1,199,121,219,255,1,9,6,1,255,2,255,205,51,255,1,161,129,28,255,1,153,122,27,255,1,114,91,17,255,1,181,110,199,255,1,232,142,255,255,1,209,127,230,255,1,149,89,164,255,1,65,36,72,255,2,0,0,0,255,1,41,99,68,255,1,95,212,151,255,8,116,255,182,255,1,108,237,169,255,1,32,81,56,255,1,48,25,54,255,1,136,81,150,255,1,198,121,218,255,1,230,141,253,255,2,232,142,255,255,1,234,149,254,255,1,235,153,253,255,5,232,142,255,255,1,220,135,242,255,2,210,128,231,255,1,194,118,213,255,1,185,112,203,255,1,173,105,191,255,2,0,0,0,255,1,100,221,157,255,1,116,255,182,255,1,114,250,179,255,1,40,97,67,255,8,0,0,0,255,1,92,53,102,255,1,126,75,139,255,2,154,93,170,255,1,146,88,161,255,1,3,1,3,255,1,51,120,83,255,1,115,252,180,255,3,116,255,182,255,1,58,134,94,255,1,0,0,0,146,29,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,137,1,58,136,95,255,4,116,255,182,255,1,113,248,177,255,1,27,71,48,255,1,152,92,168,255,1,231,142,254,255,4,232,142,255,255,1,186,113,205,255,1,101,80,14,255,2,255,205,51,255,1,123,98,19,255,1,194,155,36,255,1,0,0,0,255,1,206,125,227,255,3,232,142,255,255,1,228,139,251,255,1,186,113,205,255,1,85,49,94,255,2,0,0,0,255,1,67,153,108,255,1,110,242,172,255,6,116,255,182,255,1,52,122,86,255,1,111,65,123,255,1,223,137,245,255,5,232,142,255,255,1,244,181,246,255,1,255,213,237,255,1,254,211,238,255,1,251,201,241,255,1,250,200,241,255,1,247,192,243,255,1,246,189,244,255,1,244,182,246,255,1,242,175,248,255,1,240,169,249,255,1,237,160,252,255,1,238,162,251,255,1,233,145,254,255,1,70,40,78,255,1,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,92,206,146,255,1,0,0,0,255,1,167,101,184,255,2,185,112,203,255,1,185,112,204,255,1,186,113,204,255,1,202,123,222,255,1,210,128,231,255,1,227,139,249,255,5,232,142,255,255,1,145,87,160,255,1,0,0,0,255,1,83,186,131,255,3,116,255,182,255,1,95,212,151,255,1,0,0,0,244,1,0,0,0,12,28,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,185,1,77,174,123,255,4,116,255,182,255,1,104,231,164,255,1,0,0,0,255,1,224,137,246,255,5,232,142,255,255,1,177,107,195,255,1,124,98,19,255,2,255,205,51,255,1,68,53,7,255,1,204,164,39,255,1,0,0,0,255,1,211,128,232,255,1,232,142,255,255,1,234,148,254,255,1,237,160,252,255,1,235,153,253,255,1,232,142,255,255,1,224,137,246,255,1,143,85,157,255,2,0,0,0,255,1,40,97,67,255,1,109,241,172,255,4,116,255,182,255,1,101,223,159,255,1,0,0,0,255,1,170,103,187,255,6,232,142,255,255,1,234,148,254,255,1,242,176,248,255,1,251,201,241,255,3,255,213,237,255,1,255,212,237,255,5,255,213,237,255,1,238,164,251,255,1,32,16,36,255,1,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,70,161,113,255,1,85,49,94,255,13,232,142,255,255,1,205,125,225,255,1,0,0,0,255,1,28,72,49,255,4,116,255,182,255,1,31,80,54,255,1,0,0,0,97,28,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,201,1,79,178,126,255,4,116,255,182,255,1,91,204,145,255,1,6,3,7,255,6,232,142,255,255,1,171,103,189,255,1,124,98,19,255,2,255,205,51,255,1,52,40,4,255,1,205,164,39,255,1,0,0,0,255,1,204,124,224,255,1,232,142,255,255,1,240,171,249,255,2,255,213,237,255,1,251,201,241,255,1,241,172,249,255,1,232,142,255,255,1,169,102,186,255,2,0,0,0,255,1,40,97,67,255,1,114,250,179,255,3,116,255,182,255,1,84,189,134,255,1,0,0,0,255,1,185,112,203,255,9,232,142,255,255,1,235,153,253,255,1,242,177,248,255,1,249,196,242,255,6,255,213,237,255,1,235,152,253,255,2,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,38,93,64,255,1,140,84,154,255,14,232,142,255,255,1,85,49,94,255,1,0,0,0,255,1,104,231,164,255,3,116,255,182,255,1,75,169,120,255,1,0,0,0,179,28,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,201,1,79,178,126,255,4,116,255,182,255,1,89,198,141,255,1,53,29,59,255,6,232,142,255,255,1,172,104,189,255,1,124,98,19,255,2,255,205,51,255,1,58,45,5,255,1,204,164,39,255,1,116,92,18,255,1,185,112,203,255,2,232,142,255,255,1,242,176,248,255,1,254,211,238,255,2,255,213,237,255,1,249,197,242,255,1,233,146,254,255,1,161,97,177,255,2,0,0,0,255,1,81,182,129,255,3,116,255,182,255,1,73,166,117,255,1,0,0,0,255,1,198,120,217,255,12,232,142,255,255,1,237,161,251,255,1,250,198,242,255,3,255,213,237,255,1,251,204,240,255,1,213,130,234,255,2,0,0,0,255,1,100,223,158,255,1,116,255,182,255,1,0,0,0,255,1,173,105,190,255,14,232,142,255,255,1,146,88,161,255,1,0,0,0,255,1,90,201,142,255,3,116,255,182,255,1,92,205,146,255,1,0,0,0,219,28,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,201,1,79,178,126,255,4,116,255,182,255,1,76,172,121,255,1,2,1,2,255,6,232,142,255,255,1,171,103,189,255,1,124,98,19,255,2,255,205,51,255,1,138,109,23,255,1,173,138,31,255,1,143,114,24,255,1,159,95,175,255,3,232,142,255,255,1,236,158,252,255,1,252,206,239,255,2,255,213,237,255,1,250,199,241,255,1,233,146,254,255,1,142,85,157,255,1,0,0,0,255,1,23,61,41,255,1,110,243,173,255,2,116,255,182,255,1,56,131,92,255,1,0,0,0,255,1,210,128,231,255,14,232,142,255,255,1,240,169,249,255,1,254,211,238,255,1,255,213,237,255,1,249,195,242,255,1,205,125,226,255,2,0,0,0,255,1,112,246,175,255,1,108,237,169,255,1,0,0,0,255,1,201,123,221,255,5,232,142,255,255,1,231,142,255,255,8,232,142,255,255,1,162,98,178,255,1,0,0,0,255,1,75,170,120,255,3,116,255,182,255,1,104,231,165,255,1,0,0,0,255,1,0,0,0,1,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,174,1,76,171,121,255,4,116,255,182,255,1,75,170,120,255,1,82,47,91,255,6,232,142,255,255,1,176,106,194,255,1,111,88,16,255,1,233,187,46,255,1,196,157,37,255,1,143,113,24,255,1,143,114,24,255,1,104,82,15,255,1,154,92,170,255,4,232,142,255,255,1,235,153,253,255,1,252,206,239,255,2,255,213,237,255,1,247,192,243,255,1,227,139,250,255,1,74,42,83,255,1,0,0,0,255,1,76,173,122,255,2,116,255,182,255,1,52,124,86,255,1,0,0,0,255,1,210,128,231,255,15,232,142,255,255,1,235,155,253,255,1,249,198,242,255,1,239,167,250,255,1,180,109,198,255,1,0,0,0,255,1,46,110,76,255,1,116,255,182,255,1,104,231,165,255,1,0,0,0,255,1,221,135,243,255,11,232,142,255,255,1,229,140,251,255,2,232,142,255,255,1,188,114,207,255,1,0,0,0,255,1,58,135,95,255,3,116,255,182,255,1,112,246,176,255,1,0,0,0,255,1,0,0,0,48,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,153,1,75,170,120,255,4,116,255,182,255,1,70,160,113,255,1,112,66,124,255,6,232,142,255,255,1,189,115,208,255,2,0,0,0,255,1,112,89,17,255,1,165,132,29,255,1,169,135,30,255,1,81,63,9,255,1,83,47,92,255,1,207,126,228,255,4,232,142,255,255,1,237,159,252,255,1,254,211,238,255,2,255,213,237,255,1,238,165,250,255,1,186,113,204,255,1,0,0,0,255,1,21,58,39,255,1,113,249,178,255,1,116,255,182,255,1,51,121,84,255,1,19,8,22,255,1,229,140,252,255,18,232,142,255,255,1,152,91,167,255,1,0,0,0,255,1,52,123,86,255,1,116,255,182,255,1,98,217,155,255,1,45,24,50,255,5,232,142,255,255,1,216,132,238,255,1,206,126,227,255,1,209,127,230,255,1,229,140,251,255,6,232,142,255,255,1,210,128,231,255,1,0,0,0,255,1,1,6,3,255,4,116,255,182,255,1,0,0,0,255,1,0,0,0,55,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,153,1,70,159,112,255,4,116,255,182,255,1,53,124,87,255,1,112,66,124,255,5,232,142,255,255,1,206,125,227,255,1,50,26,56,255,1,0,0,0,255,1,81,63,9,255,1,99,78,13,255,1,0,0,0,255,1,112,89,17,255,1,186,149,35,255,1,162,129,28,255,1,57,31,64,255,1,218,133,239,255,4,232,142,255,255,1,244,183,246,255,2,255,213,237,255,1,248,193,243,255,1,228,139,250,255,1,54,29,60,255,1,0,0,0,255,1,95,211,150,255,1,116,255,182,255,1,15,44,28,255,1,94,54,104,255,19,232,142,255,255,1,112,66,124,255,1,0,0,0,255,1,58,135,95,255,1,116,255,182,255,1,89,199,141,255,1,112,66,123,255,3,232,142,255,255,1,206,125,226,255,1,73,41,81,255,1,58,58,58,255,1,43,43,43,255,1,71,71,71,255,1,71,57,75,255,1,165,100,182,255,1,228,139,251,255,4,232,142,255,255,1,225,138,248,255,2,0,0,0,255,1,109,240,171,255,3,116,255,182,255,1,41,99,68,255,1,0,0,0,100,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,131,1,52,123,86,255,4,116,255,182,255,1,52,123,86,255,1,141,84,155,255,4,232,142,255,255,1,228,139,251,255,1,100,58,110,255,1,0,0,0,255,1,135,107,22,255,1,201,161,38,255,1,204,163,39,255,1,215,173,42,255,1,182,146,34,255,1,48,36,3,255,1,201,161,38,255,1,153,122,26,255,1,108,63,119,255,4,232,142,255,255,1,233,145,254,255,1,252,206,239,255,1,255,213,237,255,1,251,201,241,255,1,232,142,255,255,1,127,75,140,255,1,0,0,0,255,1,89,199,142,255,1,116,255,182,255,1,0,0,0,255,1,112,66,123,255,19,232,142,255,255,1,71,40,79,255,1,0,0,0,255,1,75,169,120,255,1,116,255,182,255,1,75,169,120,255,1,120,71,133,255,2,232,142,255,255,1,223,137,246,255,1,68,42,74,255,1,46,46,46,255,1,76,76,76,255,1,56,56,56,255,1,82,82,82,255,1,51,51,51,255,1,57,57,57,255,1,144,86,159,255,5,232,142,255,255,1,30,15,34,255,1,0,0,0,255,1,104,229,163,255,3,116,255,182,255,1,52,124,86,255,1,0,0,0,102,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,115,1,52,123,86,255,4,116,255,182,255,1,52,124,86,255,1,154,92,170,255,4,232,142,255,255,1,181,110,199,255,1,0,0,0,255,1,204,163,39,255,5,255,205,51,255,1,203,163,39,255,1,0,0,0,255,1,226,181,44,255,1,58,45,5,255,1,195,119,215,255,4,232,142,255,255,1,242,177,247,255,1,255,213,237,255,1,255,212,237,255,1,233,146,254,255,1,154,93,170,255,1,0,0,0,255,1,75,170,120,255,1,115,253,180,255,1,0,0,0,255,1,116,69,128,255,18,232,142,255,255,1,224,137,246,255,2,0,0,0,255,1,89,199,141,255,1,116,255,182,255,1,54,127,89,255,1,159,96,175,255,2,232,142,255,255,1,164,99,180,255,1,41,41,41,255,1,78,78,78,255,1,92,92,92,255,1,97,97,97,255,1,112,112,112,255,1,69,69,69,255,1,82,82,82,255,1,39,39,39,255,1,213,130,234,255,4,232,142,255,255,1,113,67,125,255,1,0,0,0,255,1,87,195,138,255,3,116,255,182,255,1,52,124,86,255,1,0,0,0,113,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,102,1,52,124,86,255,4,116,255,182,255,1,52,124,86,255,1,154,92,169,255,4,232,142,255,255,1,136,81,150,255,1,127,101,21,255,7,255,205,51,255,1,126,100,20,255,1,180,144,33,255,1,160,128,28,255,1,139,83,154,255,4,232,142,255,255,1,235,153,253,255,2,255,213,237,255,1,237,160,252,255,1,161,98,178,255,1,0,0,0,255,1,75,170,120,255,1,105,233,166,255,1,0,0,0,255,1,146,88,161,255,17,232,142,255,255,1,228,140,251,255,1,143,86,157,255,2,0,0,0,255,1,108,238,169,255,1,116,255,182,255,1,32,81,55,255,1,183,111,201,255,2,232,142,255,255,1,75,42,83,255,1,67,67,67,255,1,77,77,77,255,1,59,59,59,255,1,71,71,71,255,1,78,78,78,255,1,59,59,59,255,1,108,108,108,255,1,88,88,88,255,1,171,103,188,255,4,232,142,255,255,1,151,91,167,255,1,0,0,0,255,1,75,170,120,255,3,116,255,182,255,1,55,129,91,255,1,0,0,0,142,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,102,1,48,114,79,255,4,116,255,182,255,1,53,125,87,255,1,154,92,170,255,4,232,142,255,255,1,112,66,124,255,1,130,103,21,255,7,255,205,51,255,1,178,142,32,255,1,136,108,22,255,1,203,163,38,255,1,112,66,124,255,5,232,142,255,255,1,251,203,240,255,1,255,213,237,255,1,237,160,252,255,1,164,99,180,255,1,0,0,0,255,1,75,170,120,255,1,100,223,158,255,1,0,0,0,255,1,154,92,170,255,8,232,142,255,255,1,218,133,240,255,3,185,112,203,255,1,183,111,201,255,1,181,109,199,255,1,169,102,186,255,1,153,92,169,255,1,120,71,132,255,1,10,4,12,255,2,0,0,0,255,1,48,115,80,255,1,116,255,182,255,1,112,248,177,255,1,0,0,0,255,1,193,117,212,255,2,232,142,255,255,1,19,7,22,255,1,72,72,72,255,1,101,101,101,255,1,89,89,89,255,1,101,101,101,255,1,112,112,112,255,1,79,79,79,255,1,102,102,102,255,1,54,54,54,255,1,157,94,173,255,4,232,142,255,255,1,164,99,181,255,1,0,0,0,255,1,67,154,108,255,3,116,255,182,255,1,53,125,87,255,1,0,0,0,123,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,60,1,0,0,0,255,4,116,255,182,255,1,55,128,90,255,1,152,91,168,255,4,232,142,255,255,1,112,66,124,255,1,123,98,19,255,7,255,205,51,255,1,190,152,36,255,1,123,97,19,255,1,203,163,39,255,1,112,66,124,255,5,232,142,255,255,1,246,189,244,255,1,255,213,237,255,1,233,145,254,255,1,177,107,194,255,1,0,0,0,255,1,75,169,120,255,1,91,203,144,255,1,0,0,0,255,1,182,110,200,255,8,232,142,255,255,1,131,78,145,255,10,0,0,0,255,1,17,49,32,255,1,97,215,153,255,1,116,255,182,255,1,102,227,161,255,1,0,0,0,255,1,210,128,231,255,2,232,142,255,255,1,120,71,132,255,1,50,50,50,255,1,61,61,61,255,1,68,68,68,255,1,77,77,77,255,1,78,78,78,255,1,46,46,46,255,1,91,91,91,255,1,57,57,57,255,1,174,105,192,255,4,232,142,255,255,1,184,112,203,255,1,0,0,0,255,1,52,124,86,255,3,116,255,182,255,1,52,124,86,255,1,0,0,0,102,27,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,44,1,0,0,0,255,1,110,243,174,255,3,116,255,182,255,1,54,126,88,255,1,154,92,169,255,4,232,142,255,255,1,123,73,136,255,1,0,0,0,255,1,249,200,50,255,6,255,205,51,255,1,136,108,22,255,1,170,136,31,255,1,191,153,35,255,1,115,68,127,255,5,232,142,255,255,1,246,188,245,255,1,250,199,241,255,1,232,142,255,255,1,156,94,172,255,1,0,0,0,255,1,89,200,142,255,1,88,196,139,255,1,0,0,0,255,1,188,114,207,255,8,232,142,255,255,1,112,66,124,255,1,0,0,0,255,1,37,92,63,255,2,50,117,82,255,1,48,114,79,255,1,52,122,85,255,1,52,123,86,255,1,69,157,110,255,1,86,192,136,255,1,102,225,160,255,1,114,251,179,255,2,116,255,182,255,1,89,200,142,255,1,0,0,0,255,1,225,138,247,255,2,232,142,255,255,1,189,115,208,255,1,32,32,32,255,1,68,68,68,255,1,105,105,105,255,1,83,83,83,255,1,112,112,112,255,1,75,75,75,255,1,72,72,72,255,1,80,67,44,255,1,192,116,211,255,4,232,142,255,255,1,207,126,227,255,1,0,0,0,255,1,16,47,30,255,3,116,255,182,255,1,52,124,86,255,1,0,0,0,102,27,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,104,231,165,255,3,116,255,182,255,1,55,128,90,255,1,154,92,170,255,4,232,142,255,255,1,170,103,187,255,1,0,0,0,255,1,226,181,44,255,5,255,205,51,255,1,211,169,41,255,1,0,0,0,255,1,218,175,42,255,1,77,60,8,255,1,179,109,197,255,5,232,142,255,255,1,243,179,247,255,1,242,177,247,255,1,232,142,255,255,1,139,83,153,255,1,0,0,0,255,1,99,220,156,255,1,75,170,120,255,1,0,0,0,255,1,211,129,232,255,8,232,142,255,255,1,104,61,115,255,1,0,0,0,255,1,102,225,160,255,11,116,255,182,255,1,78,175,124,255,1,68,38,75,255,3,232,142,255,255,1,166,100,183,255,1,129,108,74,255,1,62,51,33,255,1,78,78,78,255,1,52,52,52,255,1,81,81,81,255,1,49,40,25,255,1,154,129,89,255,1,111,89,77,255,1,210,128,231,255,4,232,142,255,255,1,231,141,254,255,2,0,0,0,255,1,111,244,174,255,2,116,255,182,255,1,52,124,86,255,1,0,0,0,102,27,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,100,221,157,255,3,116,255,182,255,1,72,164,116,255,1,154,92,170,255,4,232,142,255,255,1,226,138,248,255,1,62,34,69,255,1,155,123,27,255,5,255,205,51,255,1,137,109,23,255,1,114,91,17,255,1,94,74,12,255,1,122,72,134,255,1,230,140,252,255,7,232,142,255,255,1,222,136,244,255,1,23,10,26,255,1,16,47,30,255,1,114,251,179,255,1,73,167,118,255,1,0,0,0,255,9,232,142,255,255,1,28,13,32,255,1,0,0,0,255,1,104,231,165,255,11,116,255,182,255,1,61,142,99,255,1,111,65,122,255,3,232,142,255,255,1,218,133,240,255,1,130,96,113,255,1,154,129,89,255,1,142,119,82,255,1,151,127,87,255,1,149,124,86,255,1,152,127,87,255,1,92,76,51,255,1,199,121,219,255,6,232,142,255,255,1,72,41,80,255,1,0,0,0,255,1,98,217,154,255,2,116,255,182,255,1,52,124,86,255,1,0,0,0,102,27,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,214,1,91,203,144,255,3,116,255,182,255,1,69,157,111,255,1,154,92,170,255,5,232,142,255,255,1,206,126,227,255,1,50,26,56,255,1,123,98,19,255,1,187,150,35,255,1,215,173,41,255,1,231,186,45,255,1,220,176,43,255,1,146,116,25,255,1,0,0,0,255,1,165,100,182,255,1,229,140,252,255,8,232,142,255,255,1,183,111,201,255,1,0,0,0,255,1,64,146,103,255,1,116,255,182,255,1,55,128,89,255,1,0,0,0,255,8,232,142,255,255,1,216,132,237,255,3,0,0,0,255,1,40,98,67,255,1,52,124,86,255,1,53,125,87,255,1,61,141,99,255,1,75,170,120,255,1,78,177,125,255,1,90,201,143,255,1,97,215,153,255,1,113,248,177,255,2,116,255,182,255,1,52,123,86,255,1,112,66,124,255,4,232,142,255,255,1,224,137,246,255,1,38,19,42,255,1,130,130,130,255,1,131,131,131,255,1,146,146,146,255,1,168,168,168,255,1,50,26,56,255,4,232,142,255,255,1,236,156,252,255,1,251,201,241,255,1,233,144,255,255,1,112,66,124,255,1,0,0,0,255,1,82,185,131,255,2,116,255,182,255,1,52,124,86,255,1,0,0,0,102,27,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,202,1,85,190,135,255,3,116,255,182,255,1,67,154,108,255,1,154,92,170,255,6,232,142,255,255,1,219,134,240,255,1,167,101,184,255,1,116,68,128,255,1,50,26,56,255,1,0,0,0,255,1,25,11,28,255,1,131,78,145,255,1,209,128,230,255,9,232,142,255,255,1,228,139,250,255,1,101,59,112,255,1,0,0,0,255,1,101,224,160,255,1,116,255,182,255,1,52,124,86,255,1,68,39,75,255,8,232,142,255,255,1,225,137,247,255,1,155,93,171,255,1,137,82,151,255,1,112,66,123,255,1,72,41,80,255,1,104,61,115,255,1,36,18,40,255,5,0,0,0,255,1,25,66,44,255,2,116,255,182,255,1,32,81,55,255,1,139,83,153,255,5,232,142,255,255,1,135,80,148,255,1,165,165,165,255,2,198,198,198,255,1,181,181,181,255,1,106,62,118,255,4,232,142,255,255,1,235,153,253,255,1,255,213,237,255,1,238,163,251,255,1,139,83,154,255,1,0,0,0,255,1,58,135,95,255,2,116,255,182,255,1,66,152,107,255,1,0,0,0,153,27,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,189,1,76,172,122,255,3,116,255,182,255,1,57,133,93,255,1,155,93,171,255,10,232,142,255,255,1,230,141,253,255,11,232,142,255,255,1,228,139,250,255,1,122,72,135,255,1,0,0,0,255,1,67,153,108,255,2,116,255,182,255,1,47,113,79,255,1,111,66,123,255,16,232,142,255,255,1,229,140,251,255,1,211,129,232,255,1,189,115,208,255,2,0,0,0,255,1,116,255,182,255,1,113,249,178,255,1,0,0,0,255,1,171,103,188,255,5,232,142,255,255,1,180,109,199,255,1,116,116,116,255,2,198,198,198,255,1,162,162,162,255,1,122,72,134,255,5,232,142,255,255,1,255,213,237,255,1,241,173,248,255,1,159,96,175,255,1,0,0,0,255,1,37,90,62,255,2,116,255,182,255,1,88,196,139,255,1,0,0,0,214,27,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,190,1,76,172,122,255,3,116,255,182,255,1,52,124,86,255,1,176,107,194,255,21,232,142,255,255,1,209,128,230,255,1,110,65,122,255,1,0,0,0,255,1,40,97,67,255,1,112,247,176,255,2,116,255,182,255,1,5,20,11,255,1,112,66,124,255,18,232,142,255,255,1,192,117,211,255,1,0,0,0,255,1,37,91,62,255,1,116,255,182,255,1,102,226,161,255,1,0,0,0,255,1,185,112,203,255,5,232,142,255,255,1,219,134,241,255,1,36,36,36,255,2,198,198,198,255,1,152,152,152,255,1,154,92,170,255,1,208,127,229,255,1,214,131,236,255,3,232,142,255,255,1,240,171,249,255,1,233,146,254,255,1,186,113,205,255,2,0,0,0,255,1,115,253,181,255,1,116,255,182,255,1,104,229,163,255,1,0,0,0,255,1,0,0,0,22,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,190,1,76,172,122,255,3,116,255,182,255,1,52,124,86,255,1,170,103,187,255,18,232,142,255,255,1,223,136,245,255,1,191,116,210,255,1,120,71,132,255,2,0,0,0,255,1,29,76,51,255,1,109,240,171,255,3,116,255,182,255,1,0,0,0,255,1,134,80,148,255,18,232,142,255,255,1,168,101,185,255,1,0,0,0,255,1,60,139,97,255,1,116,255,182,255,1,89,200,142,255,1,0,0,0,255,1,185,113,204,255,6,232,142,255,255,1,40,20,45,255,1,175,175,175,255,1,198,198,198,255,1,131,131,131,255,1,107,63,119,255,1,97,57,108,255,1,216,132,238,255,5,232,142,255,255,1,212,129,233,255,2,0,0,0,255,1,106,234,167,255,2,116,255,182,255,1,15,46,30,255,1,0,0,0,69,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,190,1,76,172,122,255,3,116,255,182,255,1,52,124,86,255,1,171,104,188,255,14,232,142,255,255,1,218,133,240,255,1,125,74,138,255,1,112,66,124,255,1,98,57,108,255,4,0,0,0,255,1,62,143,100,255,1,109,240,171,255,3,116,255,182,255,1,113,248,177,255,1,0,0,0,255,1,154,92,169,255,18,232,142,255,255,1,128,76,141,255,1,0,0,0,255,1,75,170,120,255,1,116,255,182,255,1,75,170,120,255,1,0,0,0,255,1,204,125,225,255,5,232,142,255,255,1,216,131,237,255,1,2,1,3,255,1,78,78,78,255,1,97,97,97,255,1,57,57,57,255,1,61,33,68,255,1,219,134,241,255,4,232,142,255,255,1,241,172,249,255,1,243,180,247,255,1,231,142,254,255,2,0,0,0,255,1,99,220,157,255,2,116,255,182,255,1,53,124,86,255,1,0,0,0,102,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,192,1,76,173,122,255,3,116,255,182,255,1,52,124,86,255,1,184,111,202,255,14,232,142,255,255,1,219,134,241,255,1,88,50,97,255,2,0,0,0,255,1,29,74,50,255,1,54,127,89,255,1,77,175,124,255,1,99,220,157,255,5,116,255,182,255,1,104,230,164,255,1,0,0,0,255,1,154,93,170,255,18,232,142,255,255,1,81,46,90,255,1,0,0,0,255,1,89,199,141,255,1,116,255,182,255,1,62,143,101,255,1,0,0,0,255,1,218,133,240,255,4,232,142,255,255,1,145,87,159,255,2,0,0,0,255,1,1,0,2,255,1,2,1,3,255,1,15,5,18,255,1,95,55,105,255,1,216,132,238,255,4,232,142,255,255,1,243,178,247,255,1,251,201,241,255,1,232,142,255,255,1,98,57,109,255,1,0,0,0,255,1,91,203,144,255,2,116,255,182,255,1,52,124,86,255,1,0,0,0,102,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,204,1,88,197,140,255,3,116,255,182,255,1,52,124,86,255,1,179,108,197,255,15,232,142,255,255,1,219,134,241,255,1,105,62,116,255,1,0,0,0,255,1,41,99,68,255,1,110,241,172,255,7,116,255,182,255,1,96,213,151,255,1,0,0,0,255,1,173,105,191,255,17,232,142,255,255,1,225,137,247,255,2,0,0,0,255,1,103,228,162,255,1,116,255,182,255,1,53,124,86,255,1,0,0,0,255,5,232,142,255,255,1,214,130,235,255,1,210,128,231,255,1,211,129,232,255,2,232,142,255,255,1,227,139,250,255,1,210,128,231,255,1,159,95,175,255,1,174,105,191,255,3,232,142,255,255,1,242,175,248,255,1,251,201,241,255,1,232,142,255,255,1,116,68,128,255,1,0,0,0,255,1,77,175,124,255,2,116,255,182,255,1,53,125,87,255,1,0,0,0,121,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,204,1,91,203,144,255,3,116,255,182,255,1,52,124,86,255,1,171,103,188,255,7,232,142,255,255,1,224,137,247,255,1,222,136,245,255,7,232,142,255,255,1,225,137,247,255,1,88,51,98,255,1,0,0,0,255,1,42,101,70,255,1,110,243,173,255,6,116,255,182,255,1,91,203,144,255,1,0,0,0,255,1,185,112,203,255,7,232,142,255,255,1,223,137,246,255,1,220,134,241,255,8,232,142,255,255,1,200,122,220,255,1,0,0,0,255,1,6,24,14,255,1,116,254,181,255,1,116,255,182,255,1,52,123,86,255,1,100,58,111,255,13,232,142,255,255,1,212,129,233,255,3,232,142,255,255,1,242,175,248,255,1,253,207,239,255,1,232,142,255,255,1,154,92,169,255,1,0,0,0,255,1,62,143,101,255,2,116,255,182,255,1,57,133,93,255,1,0,0,0,151,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,204,1,91,203,144,255,3,116,255,182,255,1,64,147,103,255,1,154,93,170,255,7,232,142,255,255,1,185,112,203,255,1,102,60,113,255,8,232,142,255,255,1,219,134,241,255,1,87,50,96,255,1,0,0,0,255,1,56,130,91,255,1,114,250,178,255,5,116,255,182,255,1,91,204,144,255,1,0,0,0,255,1,186,113,204,255,7,232,142,255,255,1,130,77,144,255,1,6,2,7,255,1,132,79,146,255,1,183,111,201,255,1,220,134,242,255,5,232,142,255,255,1,178,108,196,255,1,0,0,0,255,1,60,138,97,255,2,116,255,182,255,1,6,26,15,255,1,149,90,165,255,17,232,142,255,255,1,242,175,248,255,1,255,213,237,255,1,232,142,255,255,1,154,92,170,255,1,0,0,0,255,1,50,117,82,255,2,116,255,182,255,1,47,113,78,255,1,0,0,0,150,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,230,1,93,208,147,255,3,116,255,182,255,1,75,170,120,255,1,153,92,169,255,7,232,142,255,255,1,185,112,203,255,1,2,1,3,255,1,224,137,247,255,8,232,142,255,255,1,211,129,232,255,1,51,27,57,255,1,0,0,0,255,1,57,134,93,255,1,114,250,179,255,4,116,255,182,255,1,91,203,144,255,1,0,0,0,255,1,203,124,224,255,7,232,142,255,255,1,50,27,56,255,4,0,0,0,255,1,92,53,102,255,1,146,87,160,255,1,155,93,170,255,1,161,97,177,255,1,154,93,170,255,1,91,53,101,255,1,0,0,0,255,1,86,193,137,255,1,116,255,182,255,1,107,237,169,255,1,0,0,0,255,1,174,105,192,255,17,232,142,255,255,1,242,175,248,255,1,255,213,237,255,1,234,148,254,255,1,164,99,181,255,1,0,0,0,255,1,11,36,23,255,2,116,255,182,255,1,44,106,73,255,1,0,0,0,103,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,100,222,158,255,3,116,255,182,255,1,74,169,119,255,1,154,92,170,255,7,232,142,255,255,1,207,126,227,255,1,0,0,0,255,1,182,110,200,255,9,232,142,255,255,1,206,125,226,255,1,26,12,30,255,1,0,0,0,255,1,63,146,102,255,4,116,255,182,255,1,83,187,132,255,1,0,0,0,255,1,210,128,231,255,7,232,142,255,255,2,0,0,0,255,1,84,190,134,255,1,75,170,120,255,1,59,136,95,255,1,11,36,23,255,6,0,0,0,255,1,103,227,162,255,1,116,255,182,255,1,95,211,150,255,1,0,0,0,255,1,195,119,215,255,17,232,142,255,255,1,242,175,248,255,1,255,213,237,255,1,237,160,251,255,1,189,115,208,255,2,0,0,0,255,1,115,254,181,255,1,116,255,182,255,1,47,111,77,255,1,0,0,0,102,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,100,222,158,255,3,116,255,182,255,1,61,141,99,255,1,157,95,173,255,7,232,142,255,255,1,227,139,249,255,1,0,0,0,255,1,73,41,81,255,1,227,139,249,255,9,232,142,255,255,1,190,115,209,255,2,0,0,0,255,1,83,186,132,255,3,116,255,182,255,1,75,170,120,255,1,0,0,0,255,1,211,129,232,255,6,232,142,255,255,1,230,140,252,255,3,0,0,0,255,1,38,93,64,255,1,61,140,98,255,1,67,154,108,255,1,19,54,36,255,1,6,22,13,255,1,11,36,23,255,1,0,1,1,255,1,0,0,0,255,1,56,132,92,255,2,116,255,182,255,1,81,182,129,255,1,0,0,0,255,1,214,130,235,255,17,232,142,255,255,1,246,188,245,255,1,255,213,237,255,1,241,174,248,255,1,219,134,241,255,2,0,0,0,255,1,104,231,165,255,1,116,255,182,255,1,53,124,87,255,1,0,0,0,102,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,100,222,158,255,3,116,255,182,255,1,52,124,86,255,1,176,106,193,255,8,232,142,255,255,1,82,47,91,255,1,0,0,0,255,1,167,101,184,255,10,232,142,255,255,1,165,100,182,255,1,0,0,0,255,1,11,37,23,255,1,101,224,159,255,2,116,255,182,255,1,71,162,114,255,1,0,0,0,255,1,228,140,251,255,6,232,142,255,255,1,231,141,254,255,1,172,104,189,255,1,154,92,170,255,1,130,78,144,255,1,112,66,123,255,1,105,61,116,255,1,112,66,123,255,1,86,50,96,255,3,0,0,0,255,1,42,101,70,255,1,93,206,147,255,2,116,255,182,255,1,68,156,110,255,1,46,24,52,255,18,232,142,255,255,1,246,189,244,255,1,255,213,237,255,1,243,181,247,255,1,232,142,255,255,1,82,47,91,255,1,0,0,0,255,1,96,213,151,255,1,116,255,182,255,1,52,123,86,255,1,0,0,0,102,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,100,222,158,255,3,116,255,182,255,1,52,123,86,255,1,163,99,180,255,8,232,142,255,255,1,111,65,122,255,1,0,0,0,255,1,18,7,21,255,1,215,132,237,255,9,232,142,255,255,1,231,141,254,255,1,124,74,137,255,1,0,0,0,255,1,22,59,40,255,1,105,232,166,255,1,116,255,182,255,1,52,124,86,255,1,83,48,92,255,15,232,142,255,255,1,219,134,241,255,1,207,126,228,255,1,181,110,200,255,1,72,40,80,255,1,0,0,0,255,1,107,236,168,255,1,116,255,182,255,1,50,119,83,255,1,124,74,137,255,10,232,142,255,255,1,223,136,245,255,1,213,130,235,255,6,232,142,255,255,1,246,189,244,255,1,255,213,237,255,1,246,189,244,255,1,232,142,255,255,1,113,67,125,255,1,0,0,0,255,1,84,189,134,255,1,116,255,182,255,1,52,124,86,255,1,0,0,0,102,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,101,224,160,255,3,116,255,182,255,1,56,131,91,255,1,156,94,171,255,8,232,142,255,255,1,119,70,131,255,2,0,0,0,255,1,115,68,127,255,10,232,142,255,255,1,228,139,250,255,1,109,64,121,255,1,0,0,0,255,1,49,115,80,255,1,116,255,182,255,1,49,117,81,255,1,112,66,124,255,18,232,142,255,255,1,122,72,134,255,1,0,0,0,255,1,104,231,165,255,1,116,254,181,255,1,1,5,3,255,1,165,100,182,255,8,232,142,255,255,1,197,120,216,255,1,1,0,1,255,1,0,0,0,255,1,46,24,51,255,1,231,141,254,255,5,232,142,255,255,1,246,189,244,255,1,255,213,237,255,1,248,195,243,255,1,232,142,255,255,1,137,82,151,255,1,0,0,0,255,1,72,164,116,255,1,116,255,182,255,1,52,124,86,255,1,0,0,0,121,26,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,255,1,104,231,165,255,3,116,255,182,255,1,75,170,120,255,1,139,83,154,255,8,232,142,255,255,1,153,92,169,255,3,0,0,0,255,1,178,108,196,255,10,232,142,255,255,1,218,133,239,255,1,9,3,11,255,1,0,0,0,255,1,116,255,182,255,1,5,22,13,255,1,125,74,138,255,18,232,142,255,255,1,112,66,124,255,1,0,0,0,255,1,110,243,173,255,1,108,238,169,255,1,0,0,0,255,1,196,119,216,255,8,232,142,255,255,1,167,101,184,255,3,0,0,0,255,1,211,129,232,255,5,232,142,255,255,1,247,191,244,255,1,255,213,237,255,1,253,209,238,255,1,232,142,255,255,1,154,92,170,255,1,0,0,0,255,1,52,123,86,255,1,116,255,182,255,1,65,150,106,255,1,0,0,0,156,26,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,37,1,0,0,0,255,1,109,241,172,255,3,116,255,182,255,1,77,174,123,255,1,115,68,127,255,8,232,142,255,255,1,168,102,185,255,1,0,0,0,255,1,26,68,46,255,1,48,114,79,255,1,37,19,42,255,1,209,127,230,255,10,232,142,255,255,1,83,48,92,255,1,1,3,2,255,1,116,255,182,255,1,0,0,0,255,1,153,92,168,255,18,232,142,255,255,1,85,49,94,255,1,0,0,0,255,1,116,255,182,255,1,102,226,161,255,1,0,0,0,255,1,214,131,236,255,8,232,142,255,255,1,184,111,202,255,3,0,0,0,255,1,189,115,208,255,5,232,142,255,255,1,251,201,241,255,2,255,213,237,255,1,235,152,253,255,1,163,99,180,255,1,0,0,0,255,1,24,63,43,255,1,116,255,182,255,1,74,169,119,255,1,0,0,0,218,26,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,51,1,0,0,0,255,1,116,254,181,255,3,116,255,182,255,1,77,175,124,255,1,114,67,126,255,8,232,142,255,255,1,184,112,202,255,2,0,0,0,255,1,109,241,172,255,1,22,59,39,255,1,77,44,85,255,1,224,137,246,255,8,232,142,255,255,1,214,131,235,255,1,4,2,5,255,1,69,158,111,255,1,114,250,178,255,1,0,0,0,255,1,154,93,170,255,17,232,142,255,255,1,219,134,241,255,1,0,0,0,255,1,4,19,10,255,1,116,255,182,255,1,91,204,145,255,1,5,2,6,255,9,232,142,255,255,1,185,112,203,255,1,0,0,0,255,1,24,64,43,255,1,0,0,0,255,1,163,98,180,255,5,232,142,255,255,1,253,208,239,255,2,255,213,237,255,1,237,161,251,255,1,185,112,203,255,2,0,0,0,255,1,110,243,174,255,1,85,191,136,255,1,0,0,0,255,1,0,0,0,32,25,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,67,1,6,26,15,255,4,116,255,182,255,1,82,183,130,255,1,112,66,124,255,8,232,142,255,255,1,199,121,218,255,2,0,0,0,255,1,116,255,182,255,1,101,223,159,255,1,17,51,33,255,1,121,72,134,255,1,228,139,251,255,6,232,142,255,255,1,217,133,239,255,1,88,51,97,255,1,19,54,36,255,1,109,239,171,255,1,105,231,165,255,1,0,0,0,255,1,167,101,184,255,17,232,142,255,255,1,192,117,211,255,1,0,0,0,255,1,55,129,90,255,1,116,255,182,255,1,81,183,130,255,1,96,56,107,255,9,232,142,255,255,1,184,112,203,255,1,0,0,0,255,1,70,159,112,255,1,35,86,59,255,1,107,63,119,255,5,232,142,255,255,1,247,191,244,255,2,246,189,244,255,1,235,152,253,255,1,205,125,225,255,2,0,0,0,255,1,95,212,151,255,1,103,228,162,255,1,0,0,0,255,1,0,0,0,171,1,0,0,0,101,1,0,0,0,12,23,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,102,1,51,120,84,255,4,116,255,182,255,1,75,171,121,255,1,112,66,124,255,8,232,142,255,255,1,210,128,230,255,2,0,0,0,255,2,116,255,182,255,1,97,216,154,255,1,20,56,37,255,1,130,78,144,255,1,228,140,251,255,4,232,142,255,255,1,191,116,210,255,1,59,32,66,255,1,20,56,37,255,1,98,218,155,255,1,116,255,182,255,1,104,231,165,255,1,0,0,0,255,1,129,77,142,255,1,221,135,243,255,16,232,142,255,255,1,139,83,153,255,1,0,0,0,255,1,75,169,120,255,1,116,255,182,255,1,61,141,99,255,1,140,84,154,255,9,232,142,255,255,1,162,98,179,255,1,0,0,0,255,1,75,170,120,255,1,66,150,106,255,1,0,0,0,255,1,228,139,251,255,8,232,142,255,255,1,222,136,244,255,2,0,0,0,255,1,79,178,126,255,1,116,255,182,255,1,92,206,146,255,1,33,83,57,255,1,0,0,0,255,1,0,0,0,224,1,0,0,0,26,22,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,119,1,52,123,86,255,4,116,255,182,255,1,91,203,144,255,1,100,59,111,255,8,232,142,255,255,1,210,128,231,255,2,0,0,0,255,3,116,255,182,255,1,102,226,161,255,1,30,77,52,255,1,113,67,125,255,1,191,116,210,255,1,209,127,230,255,1,185,112,204,255,1,131,78,144,255,1,0,0,0,255,1,57,133,93,255,1,109,241,171,255,2,116,255,182,255,1,112,246,175,255,1,40,97,67,255,1,0,0,0,255,1,50,27,56,255,1,164,99,180,255,1,210,128,231,255,13,232,142,255,255,1,219,134,241,255,1,45,24,51,255,1,0,0,0,255,1,89,199,142,255,1,116,255,182,255,1,34,84,58,255,1,158,95,174,255,9,232,142,255,255,1,146,87,161,255,1,0,0,0,255,1,86,193,137,255,1,87,194,138,255,1,0,0,0,255,1,167,101,184,255,1,172,104,189,255,1,144,87,159,255,1,113,67,125,255,1,143,86,158,255,1,162,98,178,255,1,179,109,197,255,1,143,86,158,255,1,146,88,161,255,1,153,92,168,255,2,0,0,0,255,1,61,142,99,255,2,116,255,182,255,1,114,250,179,255,1,102,226,161,255,1,57,133,93,255,1,0,0,0,183,22,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,153,1,70,160,113,255,4,116,255,182,255,1,98,218,155,255,1,1,0,1,255,1,173,104,190,255,1,215,131,237,255,1,231,141,254,255,4,232,142,255,255,1,231,142,254,255,1,180,109,198,255,1,0,0,0,255,1,33,84,57,255,1,116,255,182,255,1,112,246,176,255,2,116,255,182,255,1,107,237,169,255,1,69,157,111,255,1,2,10,5,255,2,0,0,0,255,1,33,84,57,255,1,90,201,142,255,5,116,255,182,255,1,103,229,163,255,1,3,13,7,255,3,0,0,0,255,1,88,50,97,255,1,112,66,124,255,1,123,73,136,255,1,153,92,169,255,1,154,92,170,255,1,166,100,183,255,1,163,99,180,255,1,138,83,153,255,1,153,92,169,255,1,154,92,170,255,1,164,99,181,255,1,155,93,171,255,1,154,92,170,255,1,97,57,108,255,2,0,0,0,255,1,107,236,168,255,1,115,253,181,255,1,0,0,0,255,1,137,82,151,255,1,184,112,203,255,1,173,105,191,255,1,185,112,203,255,1,180,109,198,255,1,169,102,186,255,1,170,103,187,255,1,185,112,203,255,1,176,106,193,255,1,153,92,168,255,1,70,40,78,255,1,0,0,0,255,1,91,203,144,255,1,105,233,166,255,13,0,0,0,255,1,53,125,87,255,4,116,255,182,255,1,103,229,163,255,1,0,0,0,255,1,0,0,0,22,21,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,183,1,79,178,126,255,5,116,255,182,255,1,71,162,114,255,1,19,54,35,255,1,0,0,0,255,1,15,6,18,255,1,70,39,78,255,1,71,40,79,255,1,29,14,33,255,3,0,0,0,255,1,36,89,61,255,1,102,226,161,255,1,110,242,172,255,1,77,173,122,255,1,95,211,150,255,3,116,255,182,255,1,115,253,180,255,1,107,237,169,255,1,107,236,168,255,6,116,255,182,255,1,111,246,175,255,1,116,255,182,255,1,98,217,154,255,1,67,153,108,255,1,23,62,42,255,16,0,0,0,255,1,65,149,105,255,1,116,255,182,255,1,114,251,179,255,1,62,144,101,255,12,0,0,0,255,1,95,211,150,255,1,116,255,182,255,1,60,139,98,255,12,0,0,0,255,1,67,153,107,255,5,116,255,182,255,1,0,0,0,255,1,0,0,0,51,21,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,231,1,94,210,150,255,6,116,255,182,255,1,112,247,176,255,1,105,231,165,255,1,93,208,147,255,1,85,190,135,255,1,72,165,116,255,1,66,151,106,255,1,91,204,145,255,1,93,208,148,255,1,108,239,170,255,2,116,255,182,255,1,83,186,132,255,1,82,185,131,255,1,108,238,169,255,11,116,255,182,255,1,102,226,161,255,1,85,190,135,255,1,91,203,144,255,3,116,255,182,255,1,107,236,168,255,1,104,231,164,255,1,105,231,165,255,1,95,212,151,255,2,91,203,144,255,1,86,192,137,255,3,83,186,132,255,1,89,200,142,255,1,91,203,144,255,1,101,224,160,255,2,104,231,165,255,1,111,244,174,255,3,116,255,182,255,1,115,253,180,255,1,84,188,133,255,2,75,170,120,255,1,68,156,110,255,1,53,124,86,255,1,52,124,86,255,1,42,101,70,255,2,38,94,65,255,1,40,98,68,255,1,52,124,86,255,1,63,145,102,255,1,111,246,175,255,5,116,255,182,255,1,116,254,181,255,1,112,247,176,255,6,104,231,165,255,1,109,240,171,255,2,116,255,182,255,1,112,247,176,255,3,116,255,182,255,1,38,93,64,255,1,0,0,0,92,21,0,0,0,0,}, + {26,0,0,0,0,1,0,0,0,92,1,21,57,38,255,1,112,247,176,255,6,116,255,182,255,1,112,246,175,255,1,89,199,141,255,1,41,98,68,255,4,0,0,0,255,1,19,54,36,255,1,69,158,111,255,1,107,237,169,255,1,116,255,182,255,1,113,248,177,255,1,82,184,130,255,1,114,250,179,255,1,83,186,132,255,1,74,169,119,255,1,55,128,89,255,1,44,105,73,255,1,52,122,85,255,1,52,124,86,255,1,55,129,90,255,1,52,124,86,255,1,53,124,86,255,1,80,181,128,255,1,110,243,173,255,1,109,241,172,255,1,72,164,116,255,1,83,187,133,255,1,105,232,165,255,32,116,255,182,255,1,110,243,173,255,1,83,186,132,255,1,53,124,87,255,1,69,156,110,255,1,113,248,177,255,3,116,255,182,255,2,105,232,165,255,1,109,241,172,255,1,116,254,182,255,3,116,255,182,255,1,112,246,176,255,1,91,204,145,255,1,77,174,123,255,1,43,103,71,255,1,20,56,37,255,1,89,200,142,255,2,116,255,182,255,1,53,124,86,255,1,0,0,0,110,21,0,0,0,0,}, + {25,0,0,0,0,1,0,0,0,48,1,0,0,0,244,1,84,188,133,255,6,116,255,182,255,1,82,185,131,255,1,19,55,36,255,1,66,45,82,255,1,150,107,183,255,1,184,132,223,255,3,191,137,231,255,1,173,124,209,255,1,110,77,134,255,1,5,23,14,255,1,75,169,120,255,1,113,249,178,255,1,110,242,172,255,1,73,166,117,255,1,49,33,62,255,1,101,71,124,255,1,127,90,154,255,1,141,100,171,255,1,162,116,197,255,1,141,101,172,255,1,140,99,170,255,1,134,95,162,255,1,102,72,125,255,1,69,47,85,255,1,21,57,38,255,1,101,223,159,255,1,107,236,168,255,2,89,199,141,255,2,116,255,182,255,1,107,236,168,255,1,104,231,164,255,1,103,228,162,255,1,97,215,153,255,1,104,230,163,255,1,99,219,156,255,1,91,203,144,255,1,100,222,158,255,2,104,231,165,255,1,115,254,181,255,2,116,255,182,255,1,111,245,175,255,1,104,231,164,255,1,93,208,147,255,1,84,188,133,255,1,74,167,118,255,1,73,166,117,255,1,63,145,102,255,1,44,106,73,255,1,38,92,63,255,1,33,81,56,255,1,79,177,126,255,1,114,250,179,255,4,116,255,182,255,1,98,217,155,255,1,28,73,50,255,1,70,48,86,255,1,145,103,176,255,1,64,44,80,255,1,51,120,83,255,1,101,225,160,255,1,62,143,100,255,1,19,53,35,255,3,0,0,0,255,1,37,91,63,255,1,106,234,167,255,1,116,255,182,255,1,85,190,135,255,1,20,56,37,255,1,73,50,90,255,1,112,79,136,255,1,91,63,111,255,1,0,0,0,255,1,53,124,86,255,2,116,255,182,255,1,57,133,93,255,1,0,0,0,137,21,0,0,0,0,}, + {24,0,0,0,0,1,0,0,0,31,1,0,0,0,237,1,74,168,119,255,5,116,255,182,255,1,112,247,176,255,1,60,138,97,255,1,45,30,56,255,1,172,123,208,255,1,211,152,255,255,1,224,168,252,255,1,236,183,248,255,1,242,189,246,255,1,245,193,245,255,1,239,186,247,255,1,230,176,250,255,1,218,161,253,255,1,174,125,211,255,1,77,53,95,255,1,58,134,94,255,1,109,241,172,255,1,17,50,33,255,1,140,99,170,255,8,211,152,255,255,1,207,149,250,255,1,132,94,161,255,1,20,56,37,255,1,107,236,168,255,1,92,205,145,255,1,116,255,182,255,1,113,249,178,255,1,40,97,67,255,10,0,0,0,255,1,57,133,93,255,1,111,244,174,255,1,61,140,98,255,2,0,0,0,255,1,58,39,72,255,1,104,73,127,255,1,139,99,169,255,1,140,99,170,255,1,143,102,173,255,1,168,120,203,255,1,152,108,184,255,2,0,0,0,255,1,92,205,146,255,3,116,255,182,255,1,98,217,154,255,1,6,26,16,255,1,146,104,177,255,1,207,149,251,255,1,211,152,255,255,1,139,99,168,255,1,0,0,0,255,1,37,91,62,255,1,93,65,113,255,1,168,120,203,255,1,180,129,218,255,1,191,137,231,255,1,173,124,209,255,1,0,0,0,255,1,63,146,102,255,1,116,255,182,255,1,0,0,0,255,1,160,114,193,255,1,226,171,251,255,1,230,175,250,255,1,147,105,178,255,1,0,0,0,255,1,56,131,91,255,2,116,255,182,255,1,75,170,120,255,1,0,0,0,153,21,0,0,0,0,}, + {24,0,0,0,0,1,0,0,0,185,1,59,137,96,255,5,116,255,182,255,1,115,253,180,255,1,54,126,88,255,1,80,55,98,255,1,198,143,240,255,1,222,166,252,255,1,249,198,244,255,7,255,205,242,255,1,216,159,254,255,1,190,137,230,255,1,0,0,0,255,1,66,151,106,255,1,0,0,0,255,1,144,103,175,255,3,211,152,255,255,1,210,151,253,255,1,184,132,223,255,1,195,147,214,255,1,250,200,239,255,1,247,196,245,255,1,224,169,251,255,1,209,150,252,255,1,93,65,113,255,1,56,130,91,255,2,116,255,182,255,1,91,203,144,255,1,0,0,0,255,1,198,142,239,255,1,218,161,253,255,1,218,160,253,255,1,227,172,251,255,1,225,169,251,255,1,221,164,252,255,1,223,167,252,255,1,224,172,243,255,1,212,156,247,255,1,162,116,197,255,1,0,0,0,255,1,60,138,97,255,1,0,0,0,255,1,187,135,227,255,1,206,149,250,255,1,211,152,255,255,1,217,160,253,255,1,221,165,252,255,1,229,174,250,255,1,234,181,249,255,1,223,167,252,255,1,191,137,231,255,2,0,0,0,255,1,91,203,144,255,2,116,255,182,255,1,109,240,171,255,1,21,57,38,255,1,146,104,178,255,3,211,152,255,255,1,168,120,203,255,2,0,0,0,255,1,139,99,169,255,3,211,152,255,255,1,205,148,248,255,2,0,0,0,255,1,116,254,182,255,1,0,0,0,255,1,200,150,225,255,1,255,205,242,255,1,247,196,245,255,1,168,120,203,255,1,0,0,0,255,1,56,130,91,255,2,116,255,182,255,1,78,177,125,255,1,0,0,0,188,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,58,1,0,1,0,255,1,107,236,169,255,5,116,255,182,255,1,78,175,124,255,1,61,41,75,255,1,199,143,241,255,1,230,176,250,255,1,254,204,242,255,1,255,205,242,255,1,249,198,244,255,1,235,182,248,255,1,230,176,250,255,1,231,176,250,255,1,242,190,246,255,1,255,205,242,255,1,245,194,245,255,1,211,152,255,255,1,206,149,249,255,1,0,0,0,255,1,1,2,1,255,1,0,0,0,255,1,147,105,178,255,2,211,152,255,255,1,178,128,215,255,1,89,163,160,255,1,69,180,162,255,1,73,190,171,255,1,93,155,149,255,1,226,181,214,255,1,255,205,242,255,1,224,168,252,255,1,187,135,227,255,1,0,0,0,255,1,97,216,154,255,1,116,255,182,255,1,75,171,121,255,1,72,49,88,255,1,211,152,255,255,1,234,181,249,255,1,234,180,249,255,1,248,197,244,255,4,255,205,242,255,1,235,182,248,255,1,151,108,183,255,1,0,0,0,255,1,68,156,110,255,1,0,0,0,255,1,211,152,255,255,1,236,183,248,255,1,217,159,254,255,1,248,197,244,255,3,255,205,242,255,1,242,190,246,255,1,191,137,231,255,2,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,114,250,179,255,1,44,105,73,255,1,111,78,135,255,4,211,152,255,255,1,190,137,230,255,2,0,0,0,255,1,114,81,139,255,4,211,152,255,255,1,20,11,26,255,1,0,0,0,255,1,104,231,164,255,1,0,0,0,255,1,202,151,228,255,1,255,205,242,255,1,247,196,245,255,1,168,120,203,255,1,0,0,0,255,1,54,127,89,255,2,116,255,182,255,1,87,195,138,255,1,0,0,0,204,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,144,1,66,152,107,255,5,116,255,182,255,1,99,219,156,255,1,0,0,0,255,1,177,127,214,255,1,224,169,251,255,1,255,205,242,255,1,253,202,243,255,1,230,176,250,255,5,211,152,255,255,1,234,180,249,255,1,221,165,252,255,1,211,152,255,255,1,169,121,205,255,3,0,0,0,255,1,149,106,180,255,1,211,152,255,255,1,168,130,207,255,1,61,161,145,255,1,151,167,161,255,1,179,183,179,255,1,178,182,178,255,1,99,162,149,255,1,80,166,154,255,1,234,188,222,255,1,244,192,246,255,1,211,152,255,255,1,80,55,98,255,1,63,145,102,255,1,116,255,182,255,1,68,155,109,255,1,126,89,153,255,4,211,152,255,255,1,211,153,255,255,1,222,166,252,255,1,230,176,250,255,1,245,193,245,255,1,222,166,252,255,1,140,99,170,255,1,0,0,0,255,1,66,152,107,255,1,19,11,25,255,1,211,152,255,255,1,235,181,249,255,1,213,155,254,255,1,218,161,253,255,1,227,173,251,255,1,235,182,248,255,1,242,190,246,255,1,228,173,250,255,1,204,147,247,255,2,0,0,0,255,1,90,201,143,255,1,116,255,182,255,1,77,174,123,255,1,55,37,69,255,1,201,145,243,255,4,211,152,255,255,1,178,128,216,255,2,0,0,0,255,1,98,69,119,255,4,211,152,255,255,1,82,57,101,255,1,0,0,0,255,1,96,214,152,255,1,0,0,0,255,1,192,138,230,255,1,255,205,242,255,1,247,196,245,255,1,168,120,203,255,1,0,0,0,255,1,52,124,86,255,2,116,255,182,255,1,91,203,144,255,1,0,0,0,204,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,166,1,75,170,120,255,5,116,255,182,255,1,51,119,83,255,1,99,70,121,255,1,211,152,255,255,2,249,198,244,255,1,217,160,253,255,9,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,51,121,84,255,1,0,0,0,255,1,145,103,176,255,1,196,141,237,255,1,61,156,141,255,1,173,180,175,255,2,224,229,224,255,1,157,161,157,255,1,181,186,181,255,1,76,198,178,255,1,162,148,166,255,1,247,196,245,255,1,211,152,255,255,1,118,83,143,255,1,20,56,37,255,1,116,255,182,255,1,53,124,86,255,1,146,104,177,255,9,211,152,255,255,1,113,80,138,255,1,0,0,0,255,1,72,164,116,255,1,96,67,117,255,9,211,152,255,255,2,0,0,0,255,1,87,196,139,255,1,105,233,166,255,1,0,0,0,255,1,172,124,209,255,5,211,152,255,255,1,179,128,216,255,2,0,0,0,255,1,83,58,102,255,4,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,91,203,144,255,1,3,12,6,255,1,173,124,210,255,1,250,199,244,255,1,247,196,245,255,1,168,120,203,255,1,0,0,0,255,1,52,124,86,255,2,116,255,182,255,1,90,200,142,255,1,0,0,0,204,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,164,1,75,170,120,255,4,116,255,182,255,1,102,227,162,255,1,0,0,0,255,1,168,120,204,255,1,211,152,255,255,1,233,179,249,255,1,218,161,253,255,9,211,152,255,255,1,173,124,210,255,2,0,0,0,255,1,94,209,149,255,1,0,0,0,255,1,142,101,172,255,1,162,116,197,255,1,86,168,153,255,1,222,227,222,255,1,175,179,175,255,1,112,114,112,255,1,194,199,194,255,1,192,197,192,255,1,102,169,156,255,1,128,157,162,255,1,244,192,246,255,1,211,152,255,255,1,139,99,168,255,1,0,0,0,255,1,111,244,174,255,1,34,86,59,255,1,169,121,205,255,9,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,72,163,115,255,1,101,71,124,255,9,211,152,255,255,1,47,31,59,255,1,0,0,0,255,1,88,197,140,255,1,65,149,105,255,1,110,77,134,255,6,211,152,255,255,1,186,134,225,255,2,0,0,0,255,1,79,55,97,255,4,211,152,255,255,1,101,71,123,255,1,0,0,0,255,1,91,203,144,255,1,34,84,57,255,1,168,120,203,255,1,244,192,246,255,1,247,196,245,255,1,168,120,204,255,1,0,0,0,255,1,52,124,86,255,2,116,255,182,255,1,76,172,122,255,1,0,0,0,171,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,164,1,75,170,120,255,4,116,255,182,255,1,87,194,138,255,1,0,0,0,255,1,200,144,242,255,1,244,192,246,255,1,228,173,250,255,9,211,152,255,255,1,207,149,250,255,1,67,46,83,255,1,0,0,0,255,1,43,104,72,255,1,116,255,182,255,1,14,43,28,255,1,140,99,170,255,1,143,102,173,255,1,118,187,173,255,1,224,229,224,255,1,146,149,146,255,1,189,192,189,255,1,176,180,176,255,1,196,201,196,255,1,116,157,148,255,1,120,152,164,255,1,223,167,252,255,1,211,152,255,255,1,118,83,143,255,1,0,0,0,255,1,106,234,166,255,1,0,0,0,255,1,193,139,233,255,9,211,152,255,255,1,93,65,114,255,1,0,0,0,255,1,75,169,120,255,1,133,94,161,255,9,211,152,255,255,1,92,64,112,255,1,0,0,0,255,1,78,175,124,255,1,0,0,0,255,1,182,131,221,255,6,211,152,255,255,1,172,123,208,255,1,0,0,0,255,1,11,36,23,255,1,19,11,24,255,4,211,152,255,255,1,118,83,143,255,1,0,0,0,255,1,91,203,144,255,1,52,123,86,255,1,168,120,203,255,1,239,186,247,255,1,252,202,243,255,1,168,120,203,255,1,0,0,0,255,1,52,122,85,255,2,116,255,182,255,1,63,146,103,255,1,0,0,0,145,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,161,1,75,170,120,255,4,116,255,182,255,1,75,170,120,255,1,89,62,109,255,1,211,152,255,255,1,247,195,245,255,1,225,169,251,255,4,211,152,255,255,1,176,126,213,255,2,140,99,170,255,1,164,117,199,255,1,203,146,245,255,1,145,103,176,255,2,0,0,0,255,1,91,204,145,255,1,116,255,182,255,1,11,36,23,255,1,147,105,179,255,1,167,120,203,255,1,65,170,153,255,1,216,221,216,255,1,211,216,211,255,1,155,158,155,255,1,213,218,213,255,1,216,220,216,255,1,79,170,154,255,1,136,135,166,255,1,211,152,255,255,1,204,147,246,255,1,46,31,58,255,1,14,44,29,255,1,102,225,160,255,1,0,0,0,255,1,209,150,252,255,8,211,152,255,255,1,201,145,243,255,1,0,0,0,255,1,44,106,73,255,1,74,169,119,255,1,140,99,170,255,7,211,152,255,255,1,200,144,242,255,1,138,98,167,255,1,43,28,54,255,1,0,0,0,255,1,37,91,62,255,1,76,53,94,255,7,211,152,255,255,1,167,120,202,255,1,0,0,0,255,1,46,110,76,255,1,0,0,0,255,4,211,152,255,255,1,139,99,169,255,1,0,0,0,255,1,76,173,122,255,1,51,121,84,255,1,168,120,203,255,1,236,183,248,255,1,254,204,242,255,1,168,120,203,255,1,0,0,0,255,1,53,124,86,255,2,116,255,182,255,1,53,124,87,255,1,0,0,0,127,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,145,1,67,154,109,255,4,116,255,182,255,1,78,176,125,255,1,105,74,129,255,6,211,152,255,255,1,173,124,210,255,8,0,0,0,255,1,114,251,179,255,1,116,255,182,255,1,0,1,1,255,1,167,120,203,255,1,197,142,238,255,1,68,176,159,255,1,122,148,141,255,1,214,219,214,255,1,224,229,224,255,1,208,212,208,255,1,124,167,157,255,1,71,185,166,255,1,184,134,216,255,1,211,152,255,255,1,137,97,166,255,1,0,0,0,255,1,58,136,95,255,1,86,192,136,255,1,88,61,108,255,4,211,152,255,255,1,148,106,180,255,1,152,109,185,255,3,168,120,203,255,1,126,89,153,255,1,0,0,0,255,1,65,149,105,255,1,60,140,98,255,1,144,103,175,255,4,211,152,255,255,1,205,148,248,255,1,173,124,210,255,1,126,90,154,255,1,25,15,33,255,4,0,0,0,255,1,152,109,185,255,7,211,152,255,255,1,142,101,173,255,1,0,0,0,255,1,53,124,86,255,1,0,0,0,255,1,203,146,245,255,3,211,152,255,255,1,140,99,170,255,1,0,0,0,255,1,76,171,121,255,1,52,123,86,255,1,168,120,203,255,1,229,174,250,255,1,255,205,242,255,1,168,120,203,255,1,0,0,0,255,1,51,121,84,255,2,116,255,182,255,1,53,124,87,255,1,0,0,0,126,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,93,1,38,94,64,255,4,116,255,182,255,1,75,170,120,255,1,116,82,141,255,5,211,152,255,255,1,206,148,249,255,1,46,31,58,255,1,0,0,0,255,1,17,50,32,255,1,49,117,82,255,4,0,0,0,255,1,71,161,114,255,2,116,255,182,255,1,2,11,5,255,1,167,119,202,255,1,211,152,255,255,1,166,129,205,255,1,67,175,157,255,1,66,175,157,255,1,101,178,163,255,1,68,178,160,255,1,67,173,156,255,1,166,121,202,255,1,211,152,255,255,1,156,111,189,255,2,0,0,0,255,1,92,206,146,255,1,62,143,101,255,1,143,102,174,255,3,211,152,255,255,1,196,141,237,255,7,0,0,0,255,1,88,196,139,255,1,52,124,86,255,1,166,119,202,255,2,211,152,255,255,1,203,146,246,255,1,99,69,121,255,4,0,0,0,255,1,18,52,34,255,1,68,155,109,255,1,84,188,133,255,1,0,0,0,255,1,198,143,240,255,6,211,152,255,255,1,206,148,249,255,1,85,59,104,255,1,0,0,0,255,1,69,156,110,255,1,0,0,0,255,1,191,137,231,255,3,211,152,255,255,1,140,99,170,255,1,0,0,0,255,1,76,171,121,255,1,52,124,86,255,1,168,120,203,255,1,220,163,253,255,1,254,204,242,255,1,169,121,203,255,1,0,0,0,255,1,3,16,8,255,2,116,255,182,255,1,53,124,87,255,1,0,0,0,126,21,0,0,0,0,}, + {23,0,0,0,0,1,0,0,0,31,1,0,0,0,255,1,108,238,170,255,3,116,255,182,255,1,87,195,138,255,1,78,54,97,255,5,211,152,255,255,1,185,133,224,255,1,0,0,0,255,1,2,9,4,255,1,111,244,174,255,1,116,255,182,255,1,112,246,176,255,1,92,204,145,255,1,55,127,89,255,1,51,119,83,255,1,108,238,170,255,2,116,255,182,255,1,14,42,27,255,1,167,120,203,255,2,211,152,255,255,1,196,141,237,255,1,157,112,190,255,1,143,139,188,255,1,152,109,184,255,1,198,142,239,255,1,204,147,246,255,1,130,93,158,255,1,0,0,0,255,1,1,2,1,255,1,46,111,77,255,1,115,252,180,255,1,15,46,30,255,1,179,129,217,255,3,211,152,255,255,1,190,136,229,255,1,0,0,0,255,1,45,107,74,255,1,52,122,85,255,1,53,124,87,255,1,51,121,84,255,1,53,124,86,255,1,70,160,113,255,1,113,249,178,255,1,52,122,85,255,1,168,120,203,255,2,211,152,255,255,1,192,138,232,255,2,0,0,0,255,1,42,101,70,255,1,81,182,129,255,1,99,219,156,255,1,114,251,179,255,1,116,255,182,255,1,94,209,149,255,1,48,32,60,255,6,211,152,255,255,1,189,136,229,255,1,79,55,97,255,2,0,0,0,255,1,90,201,142,255,1,25,65,44,255,1,176,126,214,255,3,211,152,255,255,1,150,107,182,255,1,0,0,0,255,1,76,171,121,255,1,54,126,88,255,1,156,111,189,255,1,211,152,255,255,1,219,162,253,255,1,168,120,203,255,2,0,0,0,255,1,116,254,182,255,1,116,255,182,255,1,53,125,87,255,1,0,0,0,131,21,0,0,0,0,}, + {24,0,0,0,0,1,0,0,0,214,1,79,179,127,255,3,116,255,182,255,1,107,236,168,255,1,0,0,0,255,1,194,139,234,255,4,211,152,255,255,1,184,132,223,255,1,45,29,56,255,1,0,0,0,255,1,88,197,140,255,1,112,247,176,255,7,116,255,182,255,1,0,2,1,255,1,168,120,203,255,5,211,152,255,255,1,200,144,242,255,1,149,106,180,255,1,56,38,70,255,2,0,0,0,255,1,36,90,62,255,1,107,237,169,255,1,106,235,167,255,1,0,0,0,255,1,199,143,241,255,3,211,152,255,255,1,168,120,203,255,1,0,0,0,255,1,67,153,108,255,1,104,231,165,255,1,106,234,167,255,1,105,231,165,255,1,114,251,179,255,2,116,255,182,255,1,50,119,83,255,1,168,120,203,255,3,211,152,255,255,2,0,0,0,255,1,75,170,120,255,1,85,190,135,255,1,71,162,115,255,1,53,124,87,255,1,52,124,86,255,1,44,106,74,255,1,102,72,125,255,5,211,152,255,255,1,191,137,231,255,1,45,29,56,255,3,0,0,0,255,1,100,222,158,255,1,54,126,88,255,1,154,110,187,255,3,211,152,255,255,1,167,120,203,255,1,0,0,0,255,1,76,171,121,255,1,72,164,116,255,1,140,99,170,255,1,212,154,255,255,1,245,193,245,255,1,174,127,204,255,2,0,0,0,255,1,113,250,178,255,1,116,255,182,255,1,45,109,76,255,1,0,0,0,109,21,0,0,0,0,}, + {24,0,0,0,0,1,0,0,0,98,1,32,81,55,255,1,112,247,176,255,3,116,255,182,255,1,54,126,88,255,1,110,78,135,255,1,207,149,250,255,4,211,152,255,255,1,204,147,247,255,1,129,92,157,255,1,16,9,22,255,1,5,20,11,255,1,63,146,102,255,1,106,235,167,255,5,116,255,182,255,1,4,17,9,255,1,172,124,209,255,3,211,152,255,255,1,203,146,246,255,1,110,77,134,255,4,0,0,0,255,1,58,134,94,255,1,110,242,172,255,1,116,255,182,255,1,100,223,159,255,1,3,1,4,255,4,211,152,255,255,1,161,115,195,255,5,0,0,0,255,1,38,93,64,255,1,108,238,170,255,1,116,255,182,255,1,49,116,81,255,1,168,120,203,255,3,211,152,255,255,1,65,44,80,255,2,0,0,0,255,1,72,50,89,255,1,125,88,152,255,1,96,67,118,255,2,0,0,0,255,1,130,93,159,255,4,211,152,255,255,1,210,151,254,255,1,84,58,103,255,2,0,0,0,255,1,20,56,37,255,1,84,190,134,255,1,116,255,182,255,1,72,165,116,255,1,138,98,167,255,3,211,152,255,255,1,155,111,189,255,1,0,0,0,255,1,76,171,121,255,1,75,170,120,255,1,140,99,170,255,1,211,152,255,255,1,229,174,250,255,1,184,133,220,255,2,0,0,0,255,1,104,231,165,255,1,116,254,181,255,1,0,0,0,255,1,0,0,0,57,21,0,0,0,0,}, + {24,0,0,0,0,1,0,0,0,10,1,0,0,0,209,1,78,176,125,255,3,116,255,182,255,1,104,230,164,255,1,7,28,17,255,1,107,75,130,255,1,195,140,236,255,5,211,152,255,255,1,207,149,251,255,1,179,129,217,255,1,109,77,134,255,1,0,0,0,255,1,72,163,115,255,1,114,251,179,255,3,116,255,182,255,1,0,0,0,255,1,191,137,231,255,3,211,152,255,255,1,196,141,237,255,3,0,0,0,255,1,58,134,94,255,1,95,211,150,255,3,116,255,182,255,1,91,203,144,255,1,100,70,122,255,5,211,152,255,255,1,204,147,246,255,1,200,144,241,255,1,208,149,251,255,1,193,139,234,255,1,127,90,154,255,1,0,0,0,255,1,58,134,94,255,1,116,255,182,255,1,47,112,78,255,1,168,120,203,255,3,211,152,255,255,1,185,133,224,255,1,168,120,203,255,1,199,143,241,255,2,211,152,255,255,1,158,113,192,255,2,0,0,0,255,1,140,99,170,255,4,211,152,255,255,1,178,128,216,255,2,0,0,0,255,1,68,155,109,255,1,111,244,174,255,2,116,255,182,255,1,83,186,132,255,1,103,72,126,255,3,211,152,255,255,1,140,99,170,255,1,0,0,0,255,1,80,181,128,255,1,77,173,123,255,1,119,85,145,255,2,211,152,255,255,1,184,132,223,255,2,0,0,0,255,1,104,231,165,255,1,116,255,182,255,1,0,0,0,255,1,0,0,0,51,21,0,0,0,0,}, + {25,0,0,0,0,1,0,0,0,74,1,0,0,0,254,1,88,197,140,255,1,115,253,180,255,2,116,255,182,255,1,102,226,161,255,1,47,112,78,255,1,0,0,0,255,1,121,85,147,255,1,188,135,227,255,6,211,152,255,255,1,185,133,224,255,1,70,48,87,255,1,58,135,95,255,1,114,250,179,255,1,116,255,182,255,1,115,254,181,255,1,0,0,0,255,1,191,137,231,255,4,211,152,255,255,1,1,0,1,255,1,0,0,0,255,1,76,171,121,255,5,116,255,182,255,1,77,174,123,255,1,122,86,148,255,9,211,152,255,255,1,110,78,134,255,1,0,0,0,255,1,63,146,103,255,1,116,255,182,255,1,6,24,14,255,1,177,127,214,255,8,211,152,255,255,1,165,118,200,255,2,0,0,0,255,1,140,100,170,255,4,211,152,255,255,1,149,106,181,255,1,0,0,0,255,1,49,116,80,255,4,116,255,182,255,1,91,203,144,255,1,94,66,115,255,3,211,152,255,255,1,140,99,170,255,1,0,0,0,255,1,53,125,87,255,1,38,92,63,255,1,101,71,124,255,2,211,152,255,255,1,190,136,229,255,2,0,0,0,255,1,104,231,164,255,1,116,255,182,255,1,0,0,0,255,1,0,0,0,51,21,0,0,0,0,}, + {26,0,0,0,0,1,0,0,0,94,1,0,0,0,245,1,31,80,55,255,1,97,215,153,255,2,116,255,182,255,1,116,254,182,255,1,95,210,150,255,1,55,130,91,255,1,0,0,0,255,1,129,92,157,255,1,195,140,236,255,5,211,152,255,255,1,193,139,234,255,1,26,16,34,255,1,70,160,113,255,1,116,255,182,255,1,106,235,167,255,1,0,0,0,255,1,197,142,238,255,4,211,152,255,255,1,82,57,100,255,1,0,0,0,255,1,75,171,121,255,5,116,255,182,255,1,75,169,120,255,1,140,99,170,255,9,211,152,255,255,1,69,47,85,255,1,0,0,0,255,1,83,186,131,255,1,116,255,182,255,1,0,0,0,255,1,191,137,231,255,8,211,152,255,255,1,166,119,202,255,2,0,0,0,255,1,141,100,171,255,4,211,152,255,255,1,139,99,169,255,1,0,0,0,255,1,87,196,139,255,4,116,255,182,255,1,92,205,145,255,1,27,17,35,255,3,211,152,255,255,1,158,113,191,255,2,0,0,0,255,1,111,78,136,255,1,186,133,225,255,2,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,104,229,163,255,1,116,255,182,255,1,0,0,0,255,1,0,0,0,51,21,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,12,1,0,0,0,129,1,1,2,1,255,1,102,226,161,255,4,116,255,182,255,1,97,216,154,255,1,55,129,90,255,1,53,35,66,255,1,199,143,240,255,5,211,152,255,255,1,135,96,164,255,1,0,0,0,255,1,98,218,155,255,1,104,231,165,255,1,0,0,0,255,5,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,75,170,120,255,5,116,255,182,255,1,64,147,104,255,1,141,100,171,255,8,211,152,255,255,1,201,144,242,255,2,0,0,0,255,1,91,202,144,255,1,116,255,182,255,1,0,0,0,255,1,191,137,231,255,6,211,152,255,255,1,210,152,254,255,1,194,140,235,255,1,152,108,184,255,2,0,0,0,255,1,140,99,170,255,4,211,152,255,255,1,151,108,183,255,1,0,0,0,255,1,81,182,129,255,4,116,255,182,255,1,101,224,160,255,1,0,0,0,255,4,211,152,255,255,1,191,137,231,255,1,203,146,246,255,4,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,92,205,145,255,1,116,255,182,255,1,9,34,21,255,1,0,0,0,72,21,0,0,0,0,}, + {29,0,0,0,0,1,0,0,0,152,1,45,109,76,255,5,116,255,182,255,1,114,251,179,255,1,49,115,80,255,1,146,104,177,255,5,211,152,255,255,1,174,125,210,255,1,0,0,0,255,1,63,145,102,255,1,103,228,162,255,1,0,0,0,255,5,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,75,170,120,255,5,116,255,182,255,1,37,91,62,255,1,170,122,206,255,4,211,152,255,255,3,191,137,231,255,1,192,138,232,255,1,179,129,217,255,2,0,0,0,255,1,98,218,155,255,1,113,248,177,255,1,0,0,0,255,1,191,137,231,255,3,211,152,255,255,1,152,108,184,255,1,87,60,106,255,1,86,60,106,255,5,0,0,0,255,1,131,93,160,255,1,225,170,251,255,1,247,196,245,255,2,211,152,255,255,1,199,143,240,255,1,45,29,56,255,1,33,83,57,255,1,91,203,144,255,1,65,149,105,255,1,103,229,163,255,1,116,255,182,255,1,104,231,165,255,1,0,0,0,255,1,204,147,246,255,9,211,152,255,255,1,205,148,248,255,2,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,59,136,96,255,1,0,0,0,130,21,0,0,0,0,}, + {29,0,0,0,0,1,0,0,0,51,1,0,0,0,255,6,116,255,182,255,1,64,147,103,255,1,141,100,171,255,5,211,152,255,255,1,190,137,230,255,1,0,0,0,255,1,7,28,17,255,1,92,206,146,255,1,66,45,82,255,5,211,152,255,255,1,102,71,124,255,1,0,0,0,255,1,75,170,120,255,4,116,255,182,255,1,112,247,176,255,1,0,0,0,255,1,191,137,231,255,3,211,152,255,255,1,162,116,196,255,7,0,0,0,255,1,108,238,169,255,1,106,234,166,255,1,0,0,0,255,1,193,138,233,255,3,211,152,255,255,1,22,13,29,255,3,0,0,0,255,1,5,20,11,255,1,40,96,67,255,1,57,133,93,255,1,66,151,106,255,1,82,57,100,255,1,217,160,253,255,1,255,205,242,255,1,221,165,252,255,2,211,152,255,255,1,185,133,224,255,1,116,82,141,255,1,71,49,87,255,1,64,44,80,255,1,44,105,73,255,1,113,249,178,255,1,104,231,165,255,1,0,0,0,255,1,196,141,237,255,2,211,152,255,255,1,204,147,246,255,1,188,135,227,255,6,211,152,255,255,1,52,35,64,255,1,0,0,0,255,1,90,201,143,255,1,116,255,182,255,1,77,173,123,255,1,0,0,0,178,21,0,0,0,0,}, + {29,0,0,0,0,1,0,0,0,163,1,46,110,77,255,5,116,255,182,255,1,105,232,165,255,1,19,55,36,255,1,171,123,207,255,5,211,152,255,255,1,184,132,223,255,2,0,0,0,255,1,91,203,144,255,1,101,71,124,255,5,211,152,255,255,1,103,73,126,255,1,0,0,0,255,1,75,170,120,255,4,116,255,182,255,1,100,222,158,255,1,2,1,3,255,1,209,151,253,255,3,211,152,255,255,1,124,88,151,255,1,0,0,0,255,1,67,153,108,255,1,60,139,98,255,3,52,124,86,255,1,66,150,106,255,1,116,255,182,255,1,105,232,165,255,1,0,0,0,255,1,203,146,245,255,2,211,152,255,255,1,203,146,245,255,2,0,0,0,255,1,105,232,165,255,1,115,252,180,255,1,116,255,182,255,1,111,246,175,255,1,105,232,165,255,1,106,234,166,255,1,0,0,0,255,1,199,143,241,255,1,251,200,243,255,1,246,194,245,255,5,211,152,255,255,1,161,115,195,255,1,0,0,0,255,1,72,164,116,255,1,105,232,165,255,1,0,0,0,255,1,191,138,231,255,2,211,152,255,255,1,140,99,170,255,2,0,0,0,255,1,68,46,84,255,1,151,108,183,255,3,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,77,175,124,255,1,116,255,182,255,1,90,200,142,255,1,0,0,0,205,21,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,8,1,0,0,0,180,1,28,73,50,255,1,104,230,164,255,2,116,255,182,255,1,114,251,179,255,1,101,223,159,255,1,80,181,128,255,1,28,72,49,255,1,115,81,140,255,6,211,152,255,255,1,158,113,192,255,1,0,0,0,255,1,36,89,61,255,1,91,203,144,255,1,101,71,124,255,5,211,152,255,255,1,102,71,124,255,1,0,0,0,255,1,75,170,120,255,4,116,255,182,255,1,87,195,138,255,1,97,68,119,255,3,211,152,255,255,1,210,151,254,255,1,44,29,55,255,1,0,0,0,255,1,47,113,78,255,1,61,141,99,255,1,68,156,110,255,1,80,180,128,255,1,104,230,164,255,2,116,255,182,255,1,104,231,165,255,1,0,0,0,255,3,211,152,255,255,1,180,129,218,255,2,0,0,0,255,1,73,165,117,255,1,53,125,87,255,1,25,67,45,255,2,0,0,0,255,1,62,142,100,255,1,32,82,56,255,1,163,117,198,255,1,236,182,248,255,1,255,205,242,255,1,232,178,249,255,4,211,152,255,255,1,180,129,217,255,1,0,0,0,255,1,20,56,37,255,1,114,250,178,255,1,0,0,0,255,1,190,137,230,255,2,211,152,255,255,1,164,117,199,255,2,0,0,0,255,1,53,125,87,255,1,102,71,124,255,3,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,68,155,109,255,1,116,255,182,255,1,100,222,158,255,1,0,0,0,254,21,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,136,1,36,90,62,255,1,108,238,170,255,1,116,255,182,255,1,114,250,179,255,1,78,177,125,255,1,30,77,52,255,1,0,0,0,255,1,79,55,97,255,1,156,111,189,255,1,208,150,251,255,5,211,152,255,255,1,207,149,250,255,1,79,55,97,255,1,0,0,0,255,1,60,140,98,255,1,90,202,144,255,1,101,71,124,255,5,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,80,180,127,255,4,116,255,182,255,1,75,170,120,255,1,135,96,165,255,4,211,152,255,255,1,186,133,225,255,1,167,119,202,255,1,146,104,177,255,1,133,95,162,255,1,104,73,127,255,1,77,53,95,255,1,20,56,37,255,1,103,229,163,255,1,116,255,182,255,1,104,230,164,255,1,0,0,0,255,3,211,152,255,255,1,195,141,236,255,1,123,88,150,255,1,104,73,127,255,1,137,98,167,255,1,157,112,191,255,1,176,126,213,255,1,173,124,209,255,2,0,0,0,255,1,80,180,127,255,1,83,58,102,255,1,212,155,251,255,1,253,203,243,255,1,255,205,242,255,1,240,188,247,255,1,225,169,251,255,1,213,155,254,255,1,216,158,254,255,1,191,137,231,255,2,0,0,0,255,1,115,252,180,255,1,19,53,35,255,1,170,121,205,255,2,211,152,255,255,1,186,134,226,255,2,0,0,0,255,1,75,170,120,255,1,98,69,120,255,3,211,152,255,255,1,106,75,129,255,1,1,2,1,255,1,52,123,86,255,1,116,255,182,255,1,104,231,165,255,1,0,0,0,255,1,0,0,0,1,20,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,245,1,95,212,151,255,2,116,255,182,255,1,78,177,125,255,1,68,46,83,255,1,163,117,198,255,1,202,145,244,255,8,211,152,255,255,1,157,112,190,255,2,0,0,0,255,1,89,199,141,255,1,90,202,143,255,1,101,71,124,255,5,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,91,203,144,255,4,116,255,182,255,1,54,126,88,255,1,158,113,192,255,9,211,152,255,255,1,168,120,203,255,1,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,104,230,164,255,1,0,0,0,255,9,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,112,246,176,255,1,27,71,48,255,1,158,113,191,255,1,227,173,251,255,4,255,205,242,255,1,248,197,244,255,1,191,137,231,255,2,0,0,0,255,1,116,254,181,255,1,36,88,61,255,1,168,120,204,255,2,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,73,165,117,255,1,95,67,117,255,3,211,152,255,255,1,121,86,147,255,1,0,0,0,255,1,52,123,86,255,1,116,255,182,255,1,107,237,169,255,1,0,0,0,255,1,0,0,0,27,20,0,0,0,0,}, + {26,0,0,0,0,1,0,0,0,2,1,0,0,0,255,1,104,231,165,255,2,116,255,182,255,1,75,170,120,255,1,127,90,155,255,9,211,152,255,255,1,196,141,237,255,1,45,29,56,255,1,0,0,0,255,1,30,77,52,255,1,114,250,179,255,1,78,176,124,255,1,107,75,131,255,5,211,152,255,255,1,101,71,124,255,1,0,0,0,255,1,91,204,145,255,4,116,255,182,255,1,52,123,86,255,1,168,120,203,255,9,211,152,255,255,1,151,108,183,255,1,0,0,0,255,1,91,203,144,255,1,116,255,182,255,1,104,231,164,255,1,0,0,0,255,9,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,111,245,175,255,1,88,198,140,255,1,14,7,19,255,1,187,134,226,255,1,233,179,249,255,3,255,205,242,255,1,247,196,245,255,1,189,136,229,255,2,0,0,0,255,1,116,255,182,255,1,0,1,0,255,1,183,132,222,255,2,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,56,131,92,255,1,58,39,72,255,3,211,152,255,255,1,140,99,170,255,1,0,0,0,255,1,54,127,89,255,2,116,255,182,255,1,0,0,0,255,1,0,0,0,51,20,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,255,1,104,231,165,255,2,116,255,182,255,1,78,176,125,255,1,101,71,124,255,8,211,152,255,255,1,198,143,240,255,1,77,53,95,255,2,0,0,0,255,1,93,207,147,255,1,116,255,182,255,1,81,182,129,255,1,91,63,111,255,4,211,152,255,255,1,209,150,252,255,1,65,45,81,255,1,0,0,0,255,1,92,204,145,255,3,116,255,182,255,1,112,246,175,255,1,3,14,7,255,1,159,114,193,255,1,208,150,252,255,8,211,152,255,255,1,140,99,170,255,1,0,0,0,255,1,92,204,145,255,1,116,255,182,255,1,104,230,164,255,1,0,0,0,255,9,211,152,255,255,1,191,137,231,255,2,0,0,0,255,1,106,234,167,255,1,116,255,182,255,1,69,157,111,255,1,50,33,62,255,1,169,121,205,255,1,226,171,251,255,1,251,201,243,255,1,255,205,242,255,1,243,192,246,255,1,167,119,202,255,1,0,0,0,255,1,22,59,40,255,1,116,255,182,255,1,0,0,0,255,1,191,137,231,255,2,211,152,255,255,1,192,138,232,255,2,0,0,0,255,1,64,146,103,255,1,0,0,0,255,1,211,152,255,255,1,207,149,250,255,1,176,126,213,255,1,90,63,110,255,1,0,0,0,255,1,65,149,105,255,2,116,255,182,255,1,2,12,6,255,1,0,0,0,66,20,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,227,1,93,207,147,255,2,116,255,182,255,1,92,204,145,255,1,49,33,61,255,7,211,152,255,255,1,188,135,227,255,1,73,51,90,255,2,0,0,0,255,1,75,169,120,255,2,116,255,182,255,1,90,202,143,255,1,0,0,0,255,1,138,98,167,255,1,152,109,185,255,1,141,100,171,255,1,144,102,174,255,1,90,63,110,255,2,0,0,0,255,1,98,217,154,255,3,116,255,182,255,1,109,240,171,255,1,1,3,1,255,1,0,0,0,255,1,64,44,79,255,1,133,95,162,255,2,140,99,170,255,1,145,103,176,255,1,168,120,203,255,1,186,134,226,255,1,192,138,232,255,1,208,150,251,255,1,135,96,164,255,1,0,0,0,255,1,103,229,163,255,1,116,255,182,255,1,98,218,155,255,1,0,0,0,255,1,202,146,245,255,4,211,152,255,255,1,210,151,253,255,1,202,145,244,255,1,197,142,238,255,1,183,131,221,255,1,136,97,165,255,2,0,0,0,255,1,106,234,167,255,1,116,255,182,255,1,114,251,179,255,1,59,136,95,255,1,0,0,0,255,1,130,93,158,255,1,197,142,236,255,1,215,157,254,255,1,208,152,245,255,1,105,74,129,255,1,0,0,0,255,1,52,123,86,255,1,116,255,182,255,1,0,0,0,255,1,163,117,198,255,1,195,140,235,255,1,191,137,231,255,1,174,124,210,255,2,0,0,0,255,1,74,167,118,255,1,0,0,0,255,1,90,63,111,255,1,34,22,43,255,3,0,0,0,255,1,68,156,110,255,2,116,255,182,255,1,51,120,84,255,1,0,0,0,112,20,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,182,1,78,176,125,255,2,116,255,182,255,1,103,229,163,255,1,0,0,0,255,1,202,146,245,255,2,211,152,255,255,1,210,151,254,255,1,191,137,231,255,1,168,121,204,255,1,131,93,160,255,2,0,0,0,255,1,4,16,8,255,1,67,153,108,255,3,116,255,182,255,1,106,235,168,255,1,20,56,37,255,6,0,0,0,255,1,20,56,37,255,1,111,244,174,255,4,116,255,182,255,1,82,184,130,255,1,9,33,21,255,11,0,0,0,255,1,110,242,172,255,1,116,255,182,255,1,110,243,173,255,1,0,1,1,255,1,65,44,80,255,1,126,89,153,255,1,146,104,177,255,1,139,99,169,255,1,104,73,127,255,1,31,19,39,255,6,0,0,0,255,1,115,254,181,255,2,116,255,182,255,1,114,250,179,255,1,62,143,100,255,2,0,0,0,255,1,95,67,117,255,1,78,54,96,255,2,0,0,0,255,1,68,156,110,255,1,110,243,173,255,7,0,0,0,255,1,86,194,137,255,1,35,86,59,255,4,0,0,0,255,1,51,121,85,255,1,104,231,165,255,2,116,255,182,255,1,67,153,108,255,1,0,0,0,151,20,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,106,1,40,98,68,255,3,116,255,182,255,1,20,57,38,255,1,122,87,149,255,1,137,97,166,255,1,95,66,116,255,5,0,0,0,255,1,43,103,72,255,1,93,208,148,255,5,116,255,182,255,1,97,215,153,255,1,66,151,106,255,1,46,110,76,255,1,51,121,84,255,1,53,125,87,255,1,79,179,127,255,1,90,200,142,255,1,105,231,165,255,2,116,255,182,255,1,114,250,179,255,1,104,229,163,255,1,110,242,172,255,1,116,255,182,255,1,114,251,179,255,1,104,231,165,255,1,99,221,157,255,1,89,199,142,255,1,81,183,130,255,1,73,165,117,255,1,52,122,85,255,1,10,34,22,255,3,0,0,0,255,1,41,100,69,255,3,116,255,182,255,1,71,162,114,255,9,0,0,0,255,1,47,113,78,255,1,72,164,116,255,1,92,204,145,255,5,116,255,182,255,1,89,199,141,255,1,59,138,96,255,1,18,53,35,255,2,0,0,0,255,1,20,56,37,255,1,106,233,166,255,1,116,255,182,255,1,64,147,104,255,5,0,0,0,255,1,51,121,84,255,1,108,239,171,255,1,109,241,172,255,1,85,190,135,255,1,75,170,120,255,1,88,198,140,255,1,109,241,172,255,4,116,255,182,255,1,75,170,120,255,1,0,0,0,153,20,0,0,0,0,}, + {27,0,0,0,0,1,0,0,0,29,1,0,0,0,255,1,104,230,164,255,2,116,255,182,255,1,91,202,144,255,6,0,0,0,255,1,52,123,86,255,1,91,204,145,255,1,116,254,181,255,3,116,255,182,255,1,86,192,136,255,1,75,170,120,255,1,102,226,161,255,10,116,255,182,255,1,69,158,111,255,1,0,0,0,255,1,12,39,25,255,1,81,182,129,255,1,114,250,179,255,7,116,255,182,255,1,111,245,175,255,1,105,232,165,255,1,104,231,164,255,1,109,241,172,255,3,116,255,182,255,1,110,243,174,255,1,62,143,100,255,1,40,97,67,255,2,0,0,0,255,1,12,39,25,255,1,53,124,86,255,1,76,172,122,255,1,99,220,156,255,1,114,250,178,255,10,116,255,182,255,1,115,253,180,255,1,104,231,165,255,1,104,231,164,255,1,111,245,175,255,3,116,255,182,255,4,104,231,165,255,1,111,245,175,255,11,116,255,182,255,1,71,161,114,255,1,0,0,0,153,20,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,194,1,75,170,120,255,3,116,255,182,255,1,113,248,177,255,1,97,215,153,255,1,91,203,144,255,1,92,205,146,255,1,104,231,164,255,1,109,241,172,255,1,116,255,182,255,1,91,203,144,255,1,58,134,94,255,1,108,238,170,255,1,116,255,182,255,1,94,209,149,255,1,0,0,0,245,1,0,0,0,174,1,0,0,0,255,1,94,209,149,255,2,116,255,182,255,1,114,252,180,255,1,44,105,73,255,1,0,0,0,255,1,67,154,108,255,1,114,250,179,255,2,116,255,182,255,1,52,123,86,255,1,0,0,0,102,1,0,0,0,89,1,0,0,0,233,1,58,134,94,255,1,115,252,180,255,11,116,255,182,255,1,104,231,165,255,1,107,236,169,255,15,116,255,182,255,1,107,237,169,255,1,114,251,179,255,25,116,255,182,255,1,102,226,161,255,1,15,46,30,255,1,0,0,0,82,20,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,124,1,20,56,37,255,1,110,242,172,255,7,116,255,182,255,1,115,253,181,255,1,81,183,130,255,1,0,0,0,255,1,0,0,0,221,1,52,123,86,255,1,114,250,179,255,1,35,87,60,255,1,0,0,0,134,1,0,0,0,0,1,0,0,0,133,1,43,104,72,255,2,116,255,182,255,1,99,220,157,255,1,0,0,0,255,1,0,0,0,72,1,0,0,0,225,1,77,173,123,255,2,116,255,182,255,1,21,59,39,255,1,0,0,0,76,1,0,0,0,0,1,0,0,0,31,1,0,0,0,236,1,78,177,125,255,4,116,255,182,255,1,98,217,154,255,1,91,203,144,255,1,98,217,154,255,3,116,255,182,255,1,95,212,151,255,2,0,0,0,255,1,77,173,123,255,1,114,250,179,255,4,116,255,182,255,1,112,246,176,255,1,81,182,129,255,1,90,201,142,255,1,104,231,165,255,1,105,231,165,255,2,116,255,182,255,1,112,247,176,255,1,74,168,119,255,1,0,0,0,255,1,23,61,41,255,1,92,206,146,255,1,109,241,172,255,3,116,255,182,255,1,101,223,159,255,1,98,218,155,255,1,111,245,175,255,3,116,255,182,255,1,78,175,124,255,1,39,95,66,255,1,61,141,99,255,1,91,204,145,255,1,109,241,172,255,4,116,255,182,255,1,115,254,181,255,1,108,237,169,255,1,112,247,176,255,1,116,255,182,255,1,112,247,176,255,1,34,85,58,255,1,0,0,0,153,21,0,0,0,0,}, + {28,0,0,0,0,1,0,0,0,18,1,0,0,0,224,1,56,131,91,255,1,106,235,168,255,6,116,255,182,255,1,68,155,109,255,1,0,0,0,237,1,0,0,0,77,1,0,0,0,31,1,0,0,0,237,1,32,80,55,255,1,0,0,0,224,1,0,0,0,10,1,0,0,0,0,1,0,0,0,53,1,0,0,0,255,2,116,255,182,255,1,91,203,144,255,1,0,0,0,213,1,0,0,0,0,1,0,0,0,74,1,15,47,30,255,1,109,240,171,255,1,115,253,181,255,1,0,0,0,255,1,0,0,0,52,2,0,0,0,0,1,0,0,0,71,1,0,0,0,254,1,77,174,123,255,1,114,250,179,255,1,116,255,182,255,1,93,207,147,255,1,0,0,0,251,1,0,0,0,235,1,0,0,0,255,1,93,207,147,255,2,116,255,182,255,1,90,202,143,255,1,0,0,0,204,1,0,0,0,76,1,0,0,0,227,1,43,103,71,255,2,75,170,120,255,1,85,191,136,255,1,54,126,88,255,1,0,1,0,255,1,0,0,0,225,1,0,0,0,228,2,0,0,0,255,1,51,120,83,255,1,88,197,140,255,1,40,97,67,255,1,0,0,0,215,1,0,0,0,77,1,0,0,0,125,1,0,0,0,239,1,19,54,36,255,1,107,236,168,255,1,116,255,182,255,1,106,235,168,255,2,0,0,0,255,1,23,61,41,255,1,102,226,161,255,1,116,255,182,255,1,100,222,158,255,1,0,0,0,255,1,0,0,0,114,1,0,0,0,148,1,0,0,0,236,1,21,59,39,255,1,100,222,158,255,2,116,255,182,255,1,89,199,142,255,1,20,56,37,255,1,0,0,0,255,1,17,50,33,255,1,77,175,124,255,1,58,135,94,255,1,0,0,0,210,1,0,0,0,7,21,0,0,0,0,}, + {29,0,0,0,0,1,0,0,0,31,1,0,0,0,185,1,20,56,37,255,1,84,188,133,255,2,104,231,165,255,1,99,220,157,255,1,82,185,131,255,1,71,162,115,255,1,0,0,0,255,1,0,0,0,53,2,0,0,0,0,1,0,0,0,55,1,0,0,0,111,1,0,0,0,32,2,0,0,0,0,1,0,0,0,51,1,0,0,0,255,2,116,255,182,255,1,91,203,144,255,1,0,0,0,204,2,0,0,0,0,1,0,0,0,212,1,78,177,125,255,1,113,249,178,255,1,0,0,0,255,1,0,0,0,51,3,0,0,0,0,1,0,0,0,62,1,0,0,0,236,1,34,85,58,255,1,72,165,116,255,1,47,111,77,255,1,0,0,0,155,1,0,0,0,0,1,0,0,0,141,1,60,139,97,255,2,116,255,182,255,1,90,202,143,255,1,0,0,0,204,1,0,0,0,0,1,0,0,0,10,1,0,0,0,140,1,0,0,0,175,1,0,0,0,167,1,0,0,0,203,1,0,0,0,174,1,0,0,0,80,1,0,0,0,1,2,0,0,0,0,1,0,0,0,27,1,0,0,0,157,1,0,0,0,204,1,0,0,0,144,1,0,0,0,10,3,0,0,0,0,1,0,0,0,194,1,78,177,125,255,1,116,255,182,255,1,91,203,144,255,1,0,0,0,223,1,0,0,0,1,1,0,0,0,153,1,14,43,27,255,1,52,123,86,255,1,9,34,21,255,1,0,0,0,144,3,0,0,0,0,1,0,0,0,151,1,0,0,0,255,1,58,135,94,255,1,73,166,117,255,1,0,0,0,255,1,0,0,0,113,1,0,0,0,49,1,0,0,0,92,1,0,0,0,191,1,0,0,0,189,1,0,0,0,31,22,0,0,0,0,}, + {31,0,0,0,0,1,0,0,0,108,1,0,0,0,226,3,0,0,0,255,1,0,0,0,210,1,0,0,0,187,1,0,0,0,103,8,0,0,0,0,1,0,0,0,51,1,0,0,0,255,2,116,255,182,255,1,91,203,144,255,1,0,0,0,204,2,0,0,0,0,1,0,0,0,76,1,0,0,0,255,1,57,133,94,255,1,0,0,0,241,1,0,0,0,3,4,0,0,0,0,1,0,0,0,11,1,0,0,0,104,2,0,0,0,153,1,0,0,0,40,1,0,0,0,0,1,0,0,0,69,1,6,24,14,255,1,112,246,176,255,1,116,255,182,255,1,91,203,144,255,1,0,0,0,204,19,0,0,0,0,1,0,0,0,153,1,75,170,120,255,1,116,255,182,255,1,91,203,144,255,1,0,0,0,210,2,0,0,0,0,1,0,0,0,85,1,0,0,0,118,1,0,0,0,67,5,0,0,0,0,1,0,0,0,42,1,0,0,0,137,1,0,0,0,153,1,0,0,0,104,28,0,0,0,0,}, + {33,0,0,0,0,1,0,0,0,1,13,0,0,0,0,1,0,0,0,51,1,0,0,0,255,2,116,255,182,255,1,91,203,144,255,1,0,0,0,204,3,0,0,0,0,1,0,0,0,70,1,0,0,0,153,1,0,0,0,82,12,0,0,0,0,1,0,0,0,239,1,95,212,151,255,1,116,255,182,255,1,91,203,144,255,1,0,0,0,204,19,0,0,0,0,1,0,0,0,161,1,75,170,120,255,1,116,255,182,255,1,103,229,163,255,1,0,0,0,255,42,0,0,0,0,}, + {47,0,0,0,0,1,0,0,0,51,1,0,0,0,255,2,116,255,182,255,1,91,204,145,255,1,0,0,0,216,18,0,0,0,0,1,0,0,0,165,1,60,140,98,255,1,116,255,182,255,1,89,199,141,255,1,0,0,0,208,19,0,0,0,0,1,0,0,0,196,1,76,172,122,255,1,116,255,182,255,1,111,244,174,255,1,0,0,0,255,1,0,0,0,41,41,0,0,0,0,}, + {47,0,0,0,0,1,0,0,0,51,1,0,0,0,255,2,116,255,182,255,1,98,217,155,255,1,0,0,0,254,18,0,0,0,0,1,0,0,0,41,1,0,0,0,253,1,86,193,137,255,1,40,98,67,255,1,0,0,0,146,19,0,0,0,0,1,0,0,0,194,1,76,172,122,255,2,116,255,182,255,1,0,0,0,255,1,0,0,0,52,41,0,0,0,0,}, + {47,0,0,0,0,1,0,0,0,51,1,0,0,0,255,2,116,255,182,255,1,98,217,155,255,1,0,0,0,254,19,0,0,0,0,1,0,0,0,102,1,0,0,0,255,1,0,0,0,195,1,0,0,0,10,19,0,0,0,0,1,0,0,0,164,1,75,170,120,255,2,116,255,182,255,1,0,0,0,255,1,0,0,0,59,41,0,0,0,0,}, + {47,0,0,0,0,1,0,0,0,47,1,0,0,0,255,1,112,246,176,255,1,116,255,182,255,1,100,221,157,255,1,0,0,0,254,20,0,0,0,0,1,0,0,0,34,1,0,0,0,6,20,0,0,0,0,1,0,0,0,129,1,57,133,93,255,1,116,255,182,255,1,110,243,173,255,1,0,0,0,255,1,0,0,0,40,41,0,0,0,0,}, + {47,0,0,0,0,1,0,0,0,2,1,0,0,0,255,1,104,231,165,255,1,116,255,182,255,1,104,231,165,255,1,0,0,0,255,42,0,0,0,0,1,0,0,0,67,1,0,0,0,255,1,112,246,176,255,1,97,216,153,255,1,0,0,0,244,42,0,0,0,0,}, + {48,0,0,0,0,1,0,0,0,234,1,94,210,150,255,1,116,255,182,255,1,105,232,166,255,1,0,0,0,255,1,0,0,0,19,42,0,0,0,0,1,0,0,0,215,1,71,161,114,255,1,67,153,108,255,1,0,0,0,182,42,0,0,0,0,}, + {48,0,0,0,0,1,0,0,0,180,1,78,176,124,255,2,116,255,182,255,1,0,0,0,255,1,0,0,0,51,42,0,0,0,0,1,0,0,0,61,1,0,0,0,231,1,0,0,0,243,1,0,0,0,52,42,0,0,0,0,}, + {48,0,0,0,0,1,0,0,0,105,1,39,96,66,255,2,116,255,182,255,1,0,0,0,255,1,0,0,0,51,88,0,0,0,0,}, + {48,0,0,0,0,1,0,0,0,21,1,0,0,0,250,1,101,223,159,255,1,104,230,164,255,1,0,0,0,255,1,0,0,0,28,88,0,0,0,0,}, + {49,0,0,0,0,1,0,0,0,161,1,38,94,65,255,1,51,120,84,255,1,0,0,0,182,89,0,0,0,0,}, + {49,0,0,0,0,1,0,0,0,10,1,0,0,0,119,1,0,0,0,138,1,0,0,0,27,89,0,0,0,0,}, + {142,0,0,0,0,}, + }, + bytes = table.concat({ +string.char(137,80,78,71,13,10,26,10,0,0,0,13,73,72,68,82,0,0,0,142,0,0,0,100,8,6,0,0,0,80,34,8,236,0,0,58,237,73,68,65,84,120,156,237,125,5,92,85,89,247,246,67,119,35,165,40,22,54,6,216,221,173,88,40,131,237,88,99,140,58,142,221,237,24,99,119,119,119,143,157,168,216,133,40,24,72,136,210,221,223,94,251,220,115,27,65,37,230,157,239), +string.char(255,248,219,158,203,169,123,238,57,235,236,189,226,89,107,107,226,255,240,127,248,1,104,22,244,5,100,3,103,214,58,177,54,142,181,245,172,93,96,237,62,107,62,146,118,152,181,9,172,181,100,205,176,128,174,241,255,75,252,27,5,167,33,107,155,89,11,97,237,181,109,195,242,71,220,38,123,44,108,177,117,212,160,14,199,166,52,247,184,182,208,205,235,254,50,87,79,239,165,174), +string.char(237,14,78,236,92,107,150,215,124,93,123,179,115,108,223,120,214,30,179,182,148,53,119,214,236,191,243,123,181,88,235,202,218,70,214,94,178,150,249,19,45,148,181,187,172,173,101,205,147,53,131,108,190,155,132,222,149,181,254,172,253,1,225,37,57,10,225,229,120,35,119,222,135,16,94,164,2,199,191,69,112,232,166,45,131,112,195,175,54,92,62,168,127,239,39,171,236,70,68,31), +string.char(66,175,171,139,209,104,78,95,184,244,109,1,231,142,117,80,180,65,37,216,187,57,195,186,92,81,20,111,225,138,122,83,189,48,252,227,46,12,250,176,13,76,168,42,55,88,54,112,116,225,150,85,232,166,7,179,118,130,181,186,217,124,55,61,180,81,172,189,47,220,170,202,193,166,235,135,255,234,121,123,73,185,65,31,183,99,72,200,110,149,246,171,255,102,105,27,28,184,67,117), +string.char(159,224,93,232,253,116,181,109,151,127,230,212,104,178,110,216,144,98,29,170,239,97,231,126,197,154,169,220,119,182,98,109,48,107,127,179,118,138,181,40,214,124,202,15,106,177,185,250,20,143,197,173,118,141,29,196,94,10,247,30,183,22,187,178,151,164,148,248,125,29,142,79,173,10,77,141,35,108,223,209,185,121,243,127,4,5,45,56,245,89,187,202,154,79,253,37,3,126,247,242), +string.char(89,110,59,42,225,40,170,143,116,135,141,75,9,232,153,42,142,62,105,137,201,240,59,126,7,7,221,103,96,165,89,87,222,110,204,218,133,164,200,56,152,22,181,225,66,85,227,247,78,240,60,55,15,131,2,183,163,229,246,49,237,217,97,55,89,219,207,90,113,165,239,166,147,79,99,237,131,83,167,154,127,247,184,181,168,136,199,177,153,168,58,168,13,10,215,46,7,83,199,66), +string.char(48,182,179,80,105,230,37,236,165,205,164,136,181,234,62,246,150,176,169,84,28,197,155,85,67,181,193,109,209,229,208,84,184,78,232,90,76,242,59,55,177,22,162,239,100,117,150,245,162,235,26,173,24,60,170,227,201,233,109,7,188,221,172,51,38,237,20,218,172,31,133,134,179,251,162,162,87,19,148,237,90,31,69,234,148,231,47,137,248,125,206,29,106,163,223,75,234,140,120,175), +string.char(234,150,119,143,37,123,20,164,224,44,96,237,122,203,157,127,52,28,246,117,31,106,142,233,2,123,215,210,208,54,208,83,187,51,9,205,217,223,86,162,228,254,85,56,234,17,141,203,235,74,225,246,22,103,12,72,185,134,3,29,166,225,235,203,143,10,251,155,22,41,132,74,189,155,225,183,207,123,80,103,126,111,15,8,195,207,114,214,138,64,24,146,30,149,242,172,63,147,189,209), +string.char(214,157,15,76,97,15,169,2,180,244,116,114,253,71,106,234,104,163,238,100,79,222,91,180,61,48,97,64,247,155,139,236,6,191,220,196,123,81,183,17,29,81,186,93,77,88,148,180,135,166,86,206,30,133,85,153,34,244,66,208,199,21,185,126,177,223,129,130,18,156,165,54,245,202,142,255,53,96,11,42,245,108,10,3,43,211,108,15,120,127,249,9,154,197,61,192,194,81,197,81), +string.char(181,172,49,140,13,181,96,103,173,131,129,157,237,112,160,111,38,78,13,88,130,132,175,209,42,199,25,218,152,163,206,132,30,232,251,98,173,126,153,190,77,70,178,85,129,86,53,75,31,100,250,146,115,135,237,227,248,27,173,169,173,149,7,63,81,6,93,99,3,222,91,148,235,214,0,142,117,43,64,39,139,151,35,167,40,205,206,197,64,255,213,207,141,235,251,17,20,132,224,116), +string.char(53,173,80,100,180,251,158,137,48,47,110,151,229,78,25,169,105,248,242,252,61,158,237,184,136,107,83,183,225,244,192,165,232,211,193,22,58,218,26,72,77,203,196,200,133,254,8,254,146,130,49,139,3,208,100,200,91,124,245,126,131,7,107,78,34,41,42,78,237,249,172,203,23,67,219,77,99,184,30,210,243,242,95,92,95,210,212,81,47,48,201,49,9,8,241,241,67,224,205,231), +string.char(248,112,229,9,124,15,221,192,171,131,215,241,242,192,53,190,20,27,173,255,116,235,5,223,247,243,163,183,136,13,250,138,180,164,148,92,185,73,223,130,190,185,17,154,111,166,119,0,35,243,252,203,178,64,65,8,206,204,198,243,251,115,29,66,25,169,9,201,248,112,245,9,174,76,218,130,165,186,29,112,172,218,20,148,218,162,3,247,103,245,144,26,18,131,146,69,4,227,228,133), +string.char(127,60,50,50,0,191,15,137,240,125,220,8,13,75,182,228,235,239,78,223,131,85,22,30,184,183,236,8,194,125,3,145,153,145,169,112,126,26,14,72,15,209,49,84,125,227,147,34,98,225,119,226,14,46,79,216,204,117,39,223,78,251,97,63,61,6,133,231,196,195,126,89,34,50,102,188,134,241,194,207,72,158,242,12,62,125,119,192,110,89,2,238,247,217,142,125,245,254,196,238), +string.char(234,163,176,179,218,72,172,47,210,27,203,12,220,177,88,163,13,206,12,90,134,143,215,158,50,91,72,238,26,216,231,248,176,40,196,133,70,114,33,11,125,248,150,11,93,208,157,87,92,16,95,236,187,138,155,115,246,224,210,216,13,252,28,98,187,61,127,31,82,98,19,20,174,183,68,75,174,226,208,144,91,250,167,158,198,15,34,191,5,167,38,107,229,157,154,86,85,217,16,253), +string.char(49,12,187,155,142,67,230,232,151,24,31,62,24,119,39,249,227,201,156,32,76,237,184,0,77,202,183,134,129,174,6,44,205,180,249,190,79,223,196,99,64,39,91,60,103,2,52,188,249,56,236,31,118,14,79,166,7,99,181,231,46,52,46,213,10,215,71,111,194,214,114,131,217,195,251,27,95,94,124,200,246,162,72,104,78,253,250,55,62,14,57,15,15,191,166,184,61,209,15,39), +string.char(71,223,196,132,118,179,121,155,251,203,82,236,248,253,16,86,244,220,138,93,67,78,241,235,154,219,117,57,95,190,157,31,131,167,51,66,120,187,62,238,5,206,143,186,143,195,67,175,224,229,198,11,56,208,104,2,162,222,133,242,239,200,72,77,103,195,233,82,172,181,253,5,235,236,189,184,144,237,114,29,201,133,110,111,157,63,112,218,99,1,206,122,254,133,82,39,12,209,222,183), +string.char(46,54,244,58,192,219,138,30,219,113,119,242,110,220,154,183,23,153,233,25,210,107,54,41,108,141,58,243,122,211,199,17,185,241,96,190,23,249,45,56,157,235,44,232,163,242,198,167,39,167,242,55,116,78,169,217,88,217,123,59,26,148,109,134,98,214,37,160,175,163,207,183,199,38,197,160,103,27,19,233,254,212,211,148,43,110,128,165,187,35,97,111,86,132,175,179,53,179,71,151), +string.char(234,94,216,57,248,36,46,140,126,128,110,85,123,227,213,230,139,216,94,113,40,30,109,56,163,242,198,202,195,123,201,97,116,140,106,138,163,163,174,160,111,253,161,40,81,168,52,179,122,101,183,198,208,41,9,214,245,162,96,211,60,18,86,117,99,96,89,39,6,102,213,152,37,87,57,30,118,149,181,225,84,217,2,69,203,89,162,66,249,50,168,94,173,10,234,149,109,136,22,101), +string.char(59,240,99,19,153,80,18,194,158,191,131,239,214,75,216,59,240,28,23,46,161,249,176,107,245,193,189,201,1,92,240,2,23,37,99,177,231,122,252,214,116,44,58,84,237,198,155,71,141,222,88,234,177,25,15,22,28,102,122,222,35,133,235,150,232,58,228,251,49,254,209,7,242,163,200,111,193,169,98,229,92,68,101,101,98,100,44,130,206,63,70,171,74,29,160,193,254,169,108,79), +string.char(73,128,173,165,204,226,185,251,44,14,134,6,130,126,162,163,37,172,215,212,205,132,142,101,58,51,225,53,225,226,88,13,139,122,172,195,223,221,54,243,109,151,6,175,194,133,223,215,114,221,69,25,209,239,63,195,103,222,1,212,117,110,4,3,29,245,206,231,132,119,250,136,125,99,132,244,4,45,104,27,167,65,199,36,13,250,133,82,96,96,155,12,163,98,137,48,113,142,135,89), +string.char(197,56,88,86,143,134,133,107,12,116,172,210,225,23,250,82,225,28,97,207,222,113,97,110,92,174,37,42,23,117,147,52,87,118,173,174,40,106,85,28,54,166,118,236,183,232,170,253,254,122,206,141,133,223,189,244,136,194,250,66,21,138,161,152,123,13,35,246,177,155,218,3,243,16,249,45,56,230,234,44,152,216,160,112,232,106,233,177,30,70,189,131,53,37,77,81,225,140,138,77), +string.char(151,138,87,114,90,18,95,234,218,166,242,7,71,61,3,245,10,22,37,181,224,89,187,63,246,15,186,192,183,211,219,254,207,168,181,124,200,144,71,100,64,8,95,154,25,152,103,121,209,25,41,26,72,8,208,67,248,45,83,124,189,105,142,216,183,70,72,137,100,2,171,42,227,28,233,113,154,120,31,245,86,97,221,251,139,143,80,179,100,189,44,191,227,91,40,108,81,148,11,221), +string.char(167,115,143,152,126,20,161,176,173,162,103,19,90,116,254,161,19,255,4,10,218,1,200,145,145,146,134,202,14,89,251,179,52,148,30,144,174,142,6,247,191,143,237,105,137,240,184,47,124,93,82,160,46,239,21,50,51,52,120,175,96,90,46,14,22,53,99,209,168,98,51,108,239,119,156,239,67,194,243,238,226,67,133,115,197,127,142,228,75,19,253,236,93,2,132,244,120,77,36,248), +string.char(235,33,242,158,9,34,124,76,145,26,173,232,251,73,75,212,66,88,88,184,244,111,3,107,225,188,126,59,175,162,120,161,82,57,250,14,117,168,95,166,41,95,198,48,93,80,30,22,165,28,104,81,246,135,79,252,131,248,87,8,78,108,240,87,60,248,116,39,203,237,153,138,198,17,220,27,89,226,67,112,18,10,219,232,34,56,242,147,116,61,245,10,17,247,77,153,206,36,252,44), +string.char(93,243,84,88,84,143,69,171,170,237,177,84,50,108,113,11,37,46,81,122,76,164,191,208,227,24,231,80,112,228,145,26,174,141,8,111,19,68,63,55,230,2,75,200,72,212,68,82,106,146,116,31,99,91,11,174,124,19,138,88,22,83,255,251,114,240,93,230,134,22,124,25,253,65,81,112,140,29,172,104,65,18,153,69,255,151,55,200,111,193,9,34,69,88,25,100,54,187,187,120), +string.char(102,121,16,233,49,177,9,233,8,139,72,197,173,199,49,220,151,243,49,52,25,175,223,39,34,53,93,241,124,105,81,90,136,148,19,30,29,83,214,251,184,196,163,91,141,94,104,83,190,11,62,223,120,133,128,11,15,100,251,39,255,188,223,37,41,72,23,225,222,102,72,97,189,79,194,39,125,132,68,9,194,236,50,178,29,247,132,167,177,30,149,96,109,108,163,246,248,172,158,120), +string.char(122,70,154,244,179,131,185,35,95,138,130,46,66,223,204,72,252,88,249,135,127,192,15,32,191,5,231,75,166,114,247,1,193,127,67,10,98,86,48,49,48,197,242,125,81,176,109,254,20,135,46,126,133,141,149,14,28,10,233,226,115,120,10,206,63,59,193,30,84,144,194,254,52,156,68,63,49,145,246,2,164,200,26,177,30,125,84,203,137,252,239,187,204,138,74,151,60,76,50,113), +string.char(173,244,109,96,101,172,234,87,250,30,164,199,106,34,146,245,62,201,33,58,248,20,33,184,0,116,77,4,101,59,133,41,229,149,236,170,193,72,79,213,248,137,74,136,68,48,19,180,207,209,33,10,130,66,208,210,212,150,126,38,229,153,112,103,250,46,133,125,180,13,164,10,245,207,185,163,191,19,249,62,84,165,198,39,169,172,139,124,27,156,229,254,52,20,253,185,119,8,255,220), +string.char(169,161,1,98,227,211,224,53,37,16,119,158,198,98,168,135,3,58,118,190,138,223,182,247,68,76,98,148,226,247,68,106,241,183,95,132,97,145,100,148,119,112,65,13,199,122,248,114,251,53,34,252,132,94,225,237,73,111,180,117,233,204,30,82,238,133,29,124,67,94,240,37,133,26,8,137,17,49,104,90,190,181,210,239,10,68,191,141,157,209,255,76,59,244,58,220,10,227,188,251), +string.char(161,255,97,119,92,122,121,22,105,233,105,42,231,52,147,12,85,153,41,233,72,142,142,87,216,86,110,64,51,90,84,202,181,31,144,3,228,183,224,132,69,168,17,18,234,133,28,213,140,255,116,3,215,92,90,132,135,193,222,152,63,204,6,19,250,59,98,104,55,7,188,58,92,30,45,106,91,96,220,223,239,81,198,201,16,125,189,252,176,226,194,2,149,227,227,253,244,165,67,150), +string.char(150,126,6,180,217,144,215,218,197,157,255,157,18,39,8,112,244,243,79,176,52,182,202,205,223,136,87,193,207,248,210,162,132,64,9,138,249,248,133,89,140,50,83,63,49,53,17,227,14,15,65,144,142,63,44,45,45,97,98,98,2,35,35,35,124,77,9,197,170,23,179,113,232,254,46,149,115,234,105,235,161,168,89,9,254,89,249,229,51,178,183,164,69,190,62,203,252,22,156,40), +string.char(181,107,53,200,170,49,83,89,253,36,208,7,155,110,175,128,207,206,50,92,104,106,84,52,129,107,121,99,148,117,50,64,235,186,22,216,56,189,20,106,246,241,99,159,45,241,48,98,41,222,124,246,85,56,62,51,157,153,209,129,66,175,163,109,148,6,13,237,76,20,182,44,202,255,142,9,12,99,67,153,224,137,213,214,202,189,168,120,18,19,138,243,190,130,21,103,90,84,24,254), +string.char(190,188,252,128,50,246,229,165,251,220,242,187,140,160,204,119,112,114,114,130,185,185,57,202,149,43,135,162,69,139,114,33,178,183,183,199,218,39,11,164,214,162,60,186,184,121,241,165,232,66,80,130,145,186,149,121,133,252,22,156,216,172,117,127,85,221,231,89,224,35,76,31,88,136,11,139,58,184,177,245,211,126,181,230,33,136,65,93,108,225,253,246,134,202,62,201,161,18,29,128), +string.char(125,175,182,105,6,244,116,100,195,87,194,215,24,190,252,150,126,245,61,72,99,58,202,209,7,123,165,127,19,5,130,16,246,52,0,118,102,14,210,245,123,239,108,133,131,131,3,239,105,223,189,123,135,224,224,96,4,4,4,32,33,33,129,175,175,90,181,42,94,6,61,85,57,191,179,157,32,124,113,33,138,190,28,45,29,46,248,142,185,242,35,114,136,124,215,113,210,152,34,172), +string.char(12,13,238,168,81,149,168,208,232,96,100,71,83,33,225,121,254,54,30,21,74,26,34,40,242,163,202,118,82,148,211,18,100,250,139,150,92,40,33,53,65,232,242,245,181,179,99,118,178,30,42,49,26,79,62,250,72,219,163,15,247,240,224,157,55,174,249,254,131,227,15,15,96,217,249,185,112,158,96,129,209,7,6,240,253,171,252,225,14,125,11,19,102,181,165,226,253,209,187,48), +string.char(212,149,117,8,222,254,55,160,195,30,182,157,157,29,239,105,168,149,40,81,2,214,214,214,72,73,73,225,66,68,223,167,12,209,215,164,204,0,200,72,227,58,81,184,202,1,121,136,252,22,156,151,143,151,30,87,189,8,45,45,100,100,102,168,217,157,245,24,169,234,215,139,48,54,212,102,10,51,121,131,179,118,99,164,198,8,214,73,90,180,166,204,209,199,58,184,68,73,143,83), +string.char(68,50,124,125,11,251,239,110,71,203,101,213,165,173,245,242,154,104,187,178,54,186,111,104,129,193,187,186,99,193,249,41,72,72,19,30,168,69,85,39,84,253,85,80,134,19,190,8,163,179,133,145,76,143,26,216,104,36,194,194,194,224,235,235,139,47,95,190,32,52,52,148,11,11,245,62,241,241,241,208,98,247,67,126,127,17,214,38,130,57,31,173,228,4,148,88,170,170,99,91), +string.char(30,162,64,116,28,81,49,21,65,186,134,166,178,123,152,129,204,215,185,91,194,145,146,154,181,139,236,222,243,24,84,46,99,140,184,132,180,172,45,163,76,13,46,60,164,243,124,137,21,110,186,105,49,27,136,62,37,93,237,236,45,89,81,231,40,221,179,33,170,253,217,153,183,42,99,58,194,109,82,55,184,77,246,224,252,24,143,171,11,56,247,185,223,189,149,176,42,43,140,28), +string.char(20,78,33,152,25,202,66,26,205,43,182,67,68,68,4,244,244,244,80,186,116,105,232,234,234,242,30,199,209,209,17,225,225,225,176,53,140,133,165,26,193,177,149,152,228,100,9,22,52,242,91,112,158,211,127,242,158,91,130,134,38,93,134,170,224,80,16,144,64,254,26,181,39,123,155,128,9,171,194,80,165,140,17,150,238,12,66,163,114,45,212,238,151,78,222,220,48,65,215,137), +string.char(140,23,30,164,161,181,41,98,131,133,207,20,89,207,14,164,244,18,26,207,237,135,38,127,253,202,91,179,37,131,209,136,253,77,52,208,202,253,91,161,104,67,23,206,125,150,143,199,197,51,125,164,153,115,91,102,21,201,116,43,242,217,244,107,87,30,41,145,254,8,10,10,194,135,15,31,120,11,246,127,136,206,204,90,44,102,83,156,89,122,214,42,215,96,103,86,152,47,163,158), +string.char(124,84,235,214,200,79,20,72,200,33,45,65,241,71,75,226,45,42,168,85,178,62,26,148,104,142,243,183,35,213,110,63,115,51,2,99,188,44,48,103,227,71,84,50,158,6,215,98,53,213,238,23,255,70,31,9,254,172,165,196,99,241,217,153,48,175,92,20,102,197,108,145,33,225,183,20,50,177,205,246,154,79,61,62,204,151,154,186,223,103,129,69,190,11,65,133,194,138,78,93), +string.char(183,226,181,161,17,87,6,61,59,212,98,194,174,141,1,109,75,162,89,5,29,12,238,222,0,215,239,251,161,135,211,82,169,195,79,30,164,11,182,42,39,184,19,136,8,86,144,40,8,193,121,37,114,84,68,168,11,67,16,104,8,89,214,115,11,214,110,117,196,148,213,239,113,241,110,20,124,223,39,32,148,245,64,254,129,73,48,208,211,228,156,156,150,14,7,48,172,217,56,104), +string.char(102,227,196,123,29,242,2,159,19,130,81,101,64,107,222,43,68,188,14,204,209,5,71,37,68,224,83,236,123,254,153,72,88,196,111,190,58,117,27,174,77,219,142,187,75,15,115,246,30,177,7,131,239,189,230,204,67,249,223,147,24,206,12,73,13,197,219,108,106,96,134,222,229,230,98,241,138,39,136,51,170,130,175,218,21,145,110,83,15,230,229,58,225,229,67,29,149,96,168,182), +string.char(105,58,143,248,211,105,202,218,87,20,174,233,253,231,28,93,123,94,161,32,4,39,46,37,86,113,168,202,72,79,207,98,87,138,209,20,193,254,225,231,80,81,107,55,206,31,233,5,175,17,102,176,111,241,12,245,189,162,17,237,55,1,79,102,4,115,175,108,118,158,223,180,244,84,28,241,17,76,101,9,1,10,233,105,233,104,86,166,109,182,23,76,60,157,141,189,14,98,97,231), +string.char(181,24,213,100,50,156,238,152,192,237,154,35,34,87,63,193,141,63,54,115,246,222,137,142,179,177,167,230,104,206,60,220,215,102,178,244,216,175,47,63,160,176,133,170,165,76,20,146,126,245,127,67,201,146,37,209,180,105,83,184,185,185,113,71,96,173,82,13,84,246,37,221,140,243,128,172,210,56,197,130,64,180,83,17,148,209,1,33,55,45,223,80,16,130,19,171,220,227,36,124), +string.char(137,150,152,228,234,65,206,65,34,64,141,109,51,29,103,198,178,55,123,73,58,30,205,14,228,214,137,173,105,246,250,9,9,205,154,75,75,176,241,214,50,52,93,63,156,13,83,130,117,146,26,159,204,134,145,42,217,30,79,190,159,246,85,187,162,79,189,33,152,208,110,14,182,13,60,138,25,157,22,227,230,148,87,8,93,154,137,160,197,169,120,54,51,20,151,254,120,204,247,15), +string.char(185,204,85,57,30,15,35,83,220,201,186,164,218,243,86,114,116,197,163,71,50,86,31,41,198,230,134,170,188,32,114,41,16,180,12,211,153,101,37,113,42,62,123,39,221,110,83,209,137,22,249,154,241,80,16,130,163,34,33,169,137,201,40,100,154,189,158,33,66,83,35,231,151,253,57,38,4,127,157,153,129,121,231,38,162,108,191,166,60,215,74,196,203,157,23,213,50,14,191,23), +string.char(20,140,36,61,73,236,245,74,123,9,189,134,200,245,161,161,73,29,200,155,124,236,216,49,233,223,228,0,116,178,46,197,253,87,17,241,138,110,153,232,151,198,60,10,47,246,56,111,246,92,151,146,241,37,217,34,229,88,83,31,126,207,3,252,43,248,56,132,148,52,85,199,224,143,130,168,166,20,47,34,167,92,229,25,14,88,113,101,30,28,219,185,162,249,223,67,160,173,47,163), +string.char(103,166,126,141,87,171,132,254,40,66,163,132,56,156,109,21,161,135,17,173,54,21,38,154,4,228,83,210,142,50,192,231,207,130,190,66,62,157,39,129,15,176,252,194,124,149,151,131,136,106,153,105,26,124,232,150,254,206,112,193,15,69,156,28,51,23,46,80,249,70,173,248,215,8,206,183,134,170,236,64,94,214,213,23,23,97,193,169,169,152,116,112,4,138,79,48,66,227,197,46), +string.char(220,41,71,104,240,247,175,112,223,53,1,122,50,238,138,84,129,181,254,73,58,133,60,124,67,132,33,138,44,54,126,93,18,71,93,86,148,141,168,248,8,248,133,191,196,214,173,91,177,113,227,70,156,57,115,6,79,63,62,192,164,246,115,165,196,45,17,6,69,83,96,228,156,4,43,19,217,185,228,57,212,206,204,66,131,106,154,115,158,161,32,4,71,83,221,232,96,162,247,253), +string.char(12,60,17,36,116,111,66,95,97,217,229,57,216,114,103,21,28,219,84,67,141,233,158,104,127,116,10,6,188,221,140,26,163,58,43,8,13,65,170,103,253,132,192,42,227,201,71,129,32,38,230,140,137,206,63,83,53,1,220,253,119,183,161,236,84,43,100,100,166,227,228,137,83,216,190,109,7,94,250,188,70,9,155,210,72,207,80,53,22,52,117,51,56,175,136,134,214,254,181,133), +string.char(140,152,164,72,153,174,152,19,22,97,110,162,32,4,199,216,216,206,82,101,165,50,147,239,123,64,93,254,236,174,203,48,180,193,88,225,11,236,45,81,107,76,23,148,113,175,195,243,178,213,65,188,233,63,75,224,18,145,156,154,132,163,79,247,240,207,162,224,36,134,11,241,38,117,58,78,243,10,237,120,74,12,41,213,33,75,211,153,194,159,134,231,243,66,241,71,235,105,184,243), +string.char(230,26,110,191,185,170,176,63,89,86,34,68,175,114,204,39,153,47,71,210,99,231,46,63,228,27,40,144,161,74,75,87,91,101,157,232,153,205,10,196,227,77,72,201,58,55,138,132,103,66,219,217,232,225,218,159,231,83,93,28,187,65,197,67,45,143,84,73,176,213,88,13,43,79,30,196,157,137,75,138,249,230,62,132,144,104,129,133,168,107,103,10,19,129,7,204,185,210,191,214), +string.char(249,93,237,254,228,25,38,253,74,222,249,72,20,89,90,215,186,178,187,52,18,46,15,234,117,8,206,118,229,248,50,90,206,151,35,209,221,178,15,186,229,18,254,21,58,14,17,186,35,19,34,190,185,207,107,166,63,44,56,53,229,155,251,144,217,60,187,203,223,232,94,173,47,207,164,188,247,247,17,158,131,174,22,146,23,56,59,221,106,231,205,13,40,63,197,6,155,174,173,228), +string.char(17,241,184,164,88,181,251,133,197,8,25,155,53,88,79,39,14,127,79,183,158,71,85,167,26,223,60,127,86,16,3,154,242,16,5,71,188,248,184,96,153,229,37,41,184,144,111,35,86,65,8,142,131,114,178,191,17,19,28,117,92,100,101,108,186,181,66,33,131,64,29,168,231,153,210,113,1,92,11,215,134,247,180,221,184,179,248,160,74,46,21,135,228,235,44,140,84,99,66,242), +string.char(56,246,112,31,82,50,146,49,229,248,72,30,17,47,53,201,20,99,247,14,198,129,187,219,113,242,209,65,206,121,62,193,150,20,61,231,63,174,166,144,169,66,58,84,74,112,52,236,37,241,165,220,2,37,30,26,234,9,250,218,235,163,183,164,235,37,202,254,147,92,253,178,111,93,71,126,125,145,28,236,245,76,21,21,85,18,154,236,252,41,212,133,147,34,249,50,40,251,123,67), +string.char(221,255,198,254,251,225,96,92,20,119,38,237,196,131,117,39,85,246,17,57,45,122,217,68,198,31,6,9,145,232,174,151,231,161,237,129,9,168,187,160,15,142,70,30,199,200,253,125,49,112,167,7,250,108,237,136,65,108,185,251,254,70,190,159,72,23,21,123,131,239,241,79,229,4,90,198,233,176,55,23,132,49,254,237,103,233,11,167,165,195,135,255,255,116,143,163,130,88,166,228), +string.char(17,7,229,91,16,187,238,149,255,44,100,138,116,246,41,45,14,22,142,216,53,88,16,152,107,35,55,224,221,63,138,137,120,84,197,139,240,173,161,42,67,206,186,57,212,100,18,222,156,244,230,49,174,90,227,186,193,170,70,41,232,218,152,192,168,148,173,116,104,170,53,167,39,175,210,69,136,9,20,104,24,86,217,244,104,57,133,182,137,112,45,20,183,146,103,44,198,127,22,248), +string.char(62,98,70,69,126,225,95,33,56,148,59,158,221,13,166,188,234,206,149,189,112,246,213,81,220,244,187,146,163,243,150,47,236,130,221,191,158,225,159,207,13,91,137,232,15,50,101,82,172,99,163,142,48,37,130,130,147,135,134,92,194,134,158,7,48,171,195,50,180,9,170,139,138,199,45,81,105,159,25,154,166,213,67,74,88,44,127,235,73,11,46,63,176,5,74,183,145,69,231,201), +string.char(249,231,98,231,166,150,30,241,67,208,16,58,19,77,189,76,62,28,211,189,32,36,69,8,138,123,78,134,250,220,196,191,66,112,168,131,141,77,86,175,116,202,67,204,132,24,182,163,87,182,202,180,136,38,229,90,97,74,155,133,136,127,243,25,199,189,22,74,223,208,112,191,79,232,82,165,39,180,53,85,45,60,17,212,27,213,115,110,130,14,213,186,97,80,163,223,121,156,106,122), +string.char(167,69,124,73,37,79,222,204,139,198,163,105,159,112,122,164,55,138,222,48,196,206,106,35,164,222,220,136,55,65,252,184,156,130,188,221,225,113,217,83,37,180,244,133,158,135,170,121,16,168,214,142,112,177,57,254,170,92,65,129,8,142,64,220,146,65,91,95,7,201,217,152,227,4,177,40,65,68,242,23,108,186,154,179,18,120,244,240,135,52,30,141,17,141,38,34,236,150,47,188), +string.char(23,29,224,235,73,153,116,180,116,250,190,11,87,2,189,249,164,111,184,58,213,68,53,137,245,148,78,138,56,123,251,31,45,61,134,114,14,217,167,58,249,135,189,134,253,24,77,238,237,246,92,211,42,219,158,131,148,99,130,72,126,87,34,174,255,199,253,56,74,69,26,173,202,56,34,50,94,61,89,75,30,37,108,156,165,159,151,92,156,137,123,1,183,190,177,183,12,148,254,50), +string.char(162,249,56,24,104,25,225,209,18,33,168,248,124,251,197,28,23,26,200,9,68,31,147,174,177,62,87,188,51,153,0,21,179,42,145,237,113,31,195,223,179,14,87,16,134,167,161,15,240,53,46,76,237,126,98,232,74,199,84,112,148,90,24,10,50,242,245,149,64,208,55,113,224,67,162,250,48,124,30,32,191,5,135,151,29,83,46,172,68,10,103,104,116,144,218,3,228,33,242,118), +string.char(29,154,185,112,190,239,31,123,7,113,26,102,78,96,106,96,142,54,149,58,65,67,146,54,145,202,244,147,156,144,212,115,10,49,195,130,178,55,69,78,117,78,244,27,11,67,69,47,186,232,15,82,134,182,161,48,68,105,106,103,66,203,48,3,6,122,130,50,252,233,166,144,53,106,96,197,11,79,169,146,121,242,8,249,45,56,252,21,215,51,81,74,71,97,195,73,78,148,59,49), +string.char(240,23,124,241,41,234,79,239,133,232,178,233,152,114,248,247,108,189,206,132,136,248,175,56,254,116,63,154,109,28,33,117,10,42,51,243,126,20,84,49,236,224,163,29,208,182,18,30,38,213,249,211,209,212,85,75,56,87,134,104,213,217,53,170,192,151,242,213,55,228,161,169,39,203,246,208,52,200,128,141,196,227,28,122,237,5,23,84,35,27,126,111,232,197,116,250,225,31,242,29), +string.char(200,247,194,74,186,14,102,42,129,69,29,125,221,44,211,99,228,33,95,252,136,138,103,119,220,53,1,55,245,124,176,254,202,50,181,249,214,34,162,19,163,48,235,216,56,152,185,21,67,217,78,245,56,157,147,144,91,67,149,216,219,184,142,20,248,192,68,76,243,170,49,48,71,199,138,175,139,75,159,230,40,233,81,151,231,92,169,3,21,114,74,12,213,227,141,74,169,200,123,150), +string.char(201,90,180,44,45,229,109,47,98,141,190,60,79,203,187,229,183,224,148,172,212,187,185,202,202,228,111,196,148,228,161,92,237,193,204,201,22,238,187,39,98,99,212,86,108,185,177,154,103,82,42,131,120,57,253,55,118,193,69,157,155,76,208,198,67,207,220,72,250,176,168,220,90,124,178,250,242,182,223,131,251,1,183,249,210,186,172,48,244,17,93,180,98,145,156,81,99,196,161,73), +string.char(199,72,31,197,26,85,198,234,107,11,213,190,4,209,143,140,16,243,68,104,233,9,154,10,110,132,168,128,16,94,76,178,231,195,21,40,222,165,54,85,34,221,0,33,163,68,125,22,64,46,32,191,5,199,66,67,77,106,166,57,19,128,184,28,152,227,234,108,78,74,179,165,202,232,155,50,119,97,201,217,89,92,80,72,239,33,138,195,198,171,203,57,47,39,99,96,97,120,158,159), +string.char(199,222,74,193,227,170,111,102,200,217,128,211,79,142,65,183,85,205,113,241,197,233,44,117,139,236,64,195,212,142,91,27,248,103,74,141,33,80,249,89,203,28,58,254,136,147,67,160,176,75,33,129,2,138,27,126,151,112,211,239,50,246,221,221,134,133,167,167,97,175,247,86,92,120,126,18,247,223,221,230,159,15,220,219,129,127,158,159,146,158,35,226,173,160,31,218,85,45,133,78), +string.char(123,39,129,230,161,168,61,183,23,249,46,54,255,208,143,202,1,242,91,112,74,136,15,79,30,196,149,217,124,107,5,110,249,93,97,214,85,132,90,125,135,44,143,143,225,2,207,182,120,87,129,108,158,150,40,56,241,136,198,208,126,219,159,120,254,75,12,154,174,113,69,229,153,14,104,185,204,13,91,116,247,227,151,187,75,81,119,146,39,228,195,28,84,236,168,225,156,190,208,50), +string.char(209,227,33,133,158,155,219,193,101,134,61,70,237,234,207,201,96,228,157,166,180,222,19,15,15,226,242,203,115,248,135,9,22,9,135,184,238,226,139,51,56,249,232,16,255,60,239,228,100,60,11,21,120,56,52,92,16,157,51,224,208,109,124,137,253,140,199,31,125,112,235,205,85,158,50,236,243,238,14,79,23,190,254,250,34,255,219,251,237,117,118,190,253,56,245,68,72,187,33,131), +string.char(193,82,146,107,238,185,177,21,186,174,107,138,81,251,251,225,239,75,179,49,250,64,127,244,222,210,1,237,87,214,229,159,71,238,235,131,193,187,122,72,127,143,232,5,39,80,28,144,238,7,77,154,2,97,178,145,60,161,147,230,183,224,20,53,176,52,81,89,73,194,212,98,199,104,12,122,56,16,229,166,90,161,236,68,107,244,94,223,17,125,54,184,163,250,140,18,176,27,163,193), +string.char(125,29,237,86,214,65,41,207,250,104,190,116,8,47,63,235,127,246,30,175,190,78,160,243,86,238,223,18,163,99,143,98,88,248,126,208,252,16,93,14,78,133,67,13,213,242,120,196,113,38,234,67,31,159,149,156,188,110,219,64,160,48,236,123,184,149,147,193,230,158,157,192,211,122,7,237,242,192,47,155,90,163,23,19,172,113,135,7,75,215,245,220,220,22,3,119,118,227,159,183), +string.char(222,89,197,143,109,188,122,40,159,90,128,178,82,27,44,27,136,241,71,134,162,213,178,234,232,178,182,49,79,25,166,107,167,116,97,143,245,205,249,223,238,107,26,242,135,127,193,247,4,63,158,104,17,70,54,230,168,183,168,255,119,223,84,211,34,170,189,155,72,237,64,30,21,35,200,247,114,181,102,78,234,57,190,149,122,53,67,175,107,75,48,52,100,55,58,220,155,9,179,181), +string.char(181,161,49,181,52,74,47,108,13,47,159,229,232,245,120,21,134,71,30,0,149,242,167,55,42,200,219,23,58,204,244,53,41,172,104,185,144,115,145,132,40,171,249,33,40,134,20,204,142,165,138,230,150,206,69,248,108,49,189,174,45,230,130,54,36,100,23,255,254,62,207,215,194,235,254,50,222,60,174,46,68,151,11,115,164,205,211,123,41,95,63,240,253,54,233,84,67,191,199,29), +string.char(129,235,111,237,249,249,201,181,80,125,68,71,116,189,56,151,7,69,169,53,219,52,18,213,167,120,192,141,53,10,77,56,247,110,204,91,217,254,205,248,186,134,203,7,193,92,66,56,171,53,182,43,198,102,158,145,181,140,211,24,149,120,76,237,20,72,212,70,68,29,100,191,65,72,241,81,158,14,192,185,87,35,90,148,251,241,199,149,53,242,83,112,122,22,170,237,108,39,167,253), +string.char(171,130,13,81,81,239,67,225,179,226,56,14,54,153,136,115,94,139,113,121,232,26,236,118,251,157,23,208,38,26,129,72,249,164,41,134,104,127,93,99,67,30,192,164,121,22,110,204,220,137,115,191,173,196,213,201,91,121,194,28,77,73,196,19,231,88,123,123,230,30,39,118,209,176,72,113,164,164,168,120,105,22,2,129,4,141,152,137,70,118,22,188,126,48,77,14,66,173,104,195), +string.char(74,40,222,188,154,180,21,174,89,150,175,167,20,27,113,170,33,82,108,229,65,194,75,213,227,105,210,15,106,85,6,180,226,211,9,53,98,173,205,134,81,232,176,253,79,222,218,109,30,131,58,227,187,243,185,183,110,207,219,135,39,155,207,169,36,39,146,105,79,21,184,104,40,35,222,146,252,20,71,84,255,143,10,85,209,125,57,59,100,57,118,53,254,19,65,119,101,53,130,36), +string.char(105,64,170,116,203,92,64,126,9,14,37,47,173,107,56,187,143,66,150,129,60,168,160,227,78,246,195,63,15,186,133,17,6,127,224,220,156,119,184,178,48,68,218,54,180,56,140,114,71,109,177,161,252,64,188,58,112,157,155,244,52,159,211,171,67,215,17,48,248,2,10,111,209,65,43,191,198,232,151,202,116,128,148,81,240,252,236,129,110,31,221,209,57,176,29,170,250,148,65,252), +string.char(164,23,216,219,98,34,175,0,90,178,149,27,247,240,138,229,242,9,20,99,162,26,194,148,90,75,217,152,113,33,225,188,196,28,245,76,180,31,197,132,168,201,235,95,196,34,164,99,62,63,241,199,251,75,143,240,114,223,85,188,216,123,5,247,87,28,227,65,213,19,189,22,226,112,231,89,120,115,194,91,45,39,40,194,47,8,71,60,230,34,118,220,83,244,252,226,137,212,101,31), +string.char(113,125,198,14,94,141,130,230,163,216,90,99,4,214,218,121,241,70,243,75,28,103,231,35,83,159,64,147,147,236,106,242,39,34,6,221,69,201,237,38,168,226,91,150,79,132,242,246,148,172,32,129,100,10,167,220,37,4,73,144,31,130,67,218,253,165,54,251,198,27,169,155,195,129,186,87,239,69,7,241,102,224,105,172,111,118,8,203,135,29,70,155,250,205,225,230,106,15,87,55), +string.char(51,212,168,101,138,42,85,205,81,219,173,28,134,119,29,143,253,76,176,52,23,135,224,238,146,67,156,154,233,189,248,16,188,154,12,199,200,78,179,209,181,193,64,180,112,235,138,170,165,234,160,133,107,23,180,172,222,13,237,107,247,194,8,247,89,248,107,16,27,82,74,79,198,81,207,121,252,123,137,190,90,184,86,57,158,52,119,170,255,18,172,182,238,129,117,246,61,249,28,11), +string.char(148,141,185,206,161,23,54,151,254,149,207,181,176,169,68,127,158,250,75,141,202,221,82,194,63,125,255,114,163,78,252,24,255,238,167,97,189,36,5,141,47,186,162,233,37,166,191,188,104,137,41,166,179,48,198,124,34,222,29,245,198,241,142,179,120,101,117,121,80,13,66,234,33,198,21,155,142,133,3,119,163,73,213,142,104,91,179,7,47,189,127,117,210,22,20,59,97,130,213,77), +string.char(247,225,212,76,63,254,226,28,158,242,4,111,118,95,195,139,125,2,51,128,122,228,197,117,54,97,253,196,237,24,215,119,12,106,149,23,234,32,203,135,115,36,134,72,214,149,191,127,2,249,33,56,27,234,46,236,107,89,190,123,67,149,13,164,72,210,131,176,62,172,129,245,163,207,161,90,133,138,112,118,137,71,177,210,9,176,178,73,129,137,121,26,12,141,211,97,110,149,10,123), +string.char(199,36,148,174,24,143,242,37,42,96,116,215,249,184,61,105,7,94,236,185,140,200,135,239,160,173,173,24,251,202,204,204,64,208,215,247,136,74,125,133,50,85,162,80,170,98,2,44,172,53,208,190,150,23,90,27,180,227,211,19,217,85,19,38,93,161,106,89,84,56,251,246,210,72,248,109,79,135,223,182,116,4,30,74,197,135,253,169,240,223,149,134,119,123,211,240,254,80,44,158), +string.char(111,74,196,142,177,183,112,103,242,78,126,252,141,177,91,240,207,252,64,60,92,157,194,174,253,44,70,119,159,134,206,77,186,192,179,173,59,188,58,182,67,205,202,85,81,198,177,178,194,111,21,65,67,207,201,190,139,177,184,195,14,52,174,210,1,218,90,66,132,222,177,144,16,106,122,179,251,58,92,74,212,66,5,39,55,20,179,45,13,107,51,59,56,217,149,225,219,136,91,36), +string.char(58,162,202,150,114,130,99,137,68,88,20,74,69,33,177,226,134,156,65,42,17,162,60,137,95,229,181,224,212,129,6,90,80,146,191,58,80,23,174,119,48,22,227,123,252,13,167,226,134,112,114,78,128,174,94,6,82,146,53,241,245,179,46,66,2,245,17,202,90,108,180,112,99,19,226,181,144,154,172,1,115,73,177,199,43,195,214,242,165,124,222,248,187,80,95,204,55,222,137,37), +string.char(85,110,96,94,133,23,24,240,108,15,190,68,127,64,177,82,137,48,99,163,125,239,22,99,56,165,52,252,181,224,218,167,228,253,86,85,123,192,196,208,28,230,214,105,188,21,178,79,70,81,182,127,137,114,9,252,154,138,21,207,132,173,131,6,226,37,164,117,125,137,101,104,103,81,4,69,75,100,176,107,206,100,194,206,246,47,155,0,135,98,73,40,100,151,12,43,91,245,100,51), +string.char(26,178,174,76,220,130,95,10,245,99,66,211,94,97,155,141,133,108,84,9,137,80,172,46,70,5,24,172,141,36,17,241,207,130,239,71,83,67,149,252,150,33,55,195,140,196,130,253,159,236,113,60,154,172,29,38,6,224,20,64,250,195,137,78,115,48,214,99,17,236,29,12,121,143,66,117,137,73,88,252,158,26,33,244,163,30,194,67,117,240,149,181,15,126,6,248,18,162,199,132), +string.char(72,8,142,190,249,36,36,190,137,41,176,98,214,227,139,176,199,88,84,252,2,26,104,107,96,164,150,5,134,38,233,177,135,10,140,140,190,137,47,145,95,97,109,151,194,31,182,123,205,254,8,56,239,195,143,73,142,138,67,141,178,194,36,27,159,3,117,241,249,147,46,222,60,51,194,115,31,19,188,123,109,200,175,39,33,78,11,177,81,218,136,79,20,20,243,140,180,116,88,25), +string.char(217,75,174,65,3,84,73,237,227,91,3,124,122,111,192,5,158,4,61,38,82,155,11,163,8,241,90,223,48,29,196,232,98,26,126,105,58,92,229,158,196,37,202,178,41,162,212,112,115,26,184,8,214,147,232,183,73,136,85,125,124,242,5,28,178,210,39,115,3,121,45,56,54,234,252,54,132,199,155,206,98,0,83,86,75,216,151,99,66,147,204,123,153,183,47,13,185,176,168,3), +string.char(61,208,132,88,225,13,187,248,240,40,95,186,78,236,202,151,228,162,143,142,143,192,106,135,203,24,101,85,20,173,27,52,68,37,151,74,60,165,182,179,157,19,210,156,236,112,206,215,27,90,218,194,195,115,46,82,137,43,177,4,138,243,208,80,160,2,182,107,124,140,22,191,158,128,87,134,252,115,80,248,123,190,137,34,224,237,107,245,228,159,67,153,176,101,164,107,32,57,81,19), +string.char(81,95,180,185,192,147,160,71,126,209,81,40,84,153,158,154,198,45,164,147,157,231,226,207,238,139,97,168,38,45,71,222,47,30,173,134,102,98,37,225,47,147,142,165,101,172,139,240,152,175,72,77,201,250,17,202,207,111,149,219,200,107,193,81,123,229,148,72,246,96,225,97,52,175,214,133,255,29,21,206,122,149,55,6,72,73,202,254,114,252,62,61,195,198,243,115,248,231,10,61), +string.char(132,158,130,42,146,7,132,248,194,222,216,24,85,43,87,198,163,135,143,176,108,201,50,148,118,46,205,235,7,59,4,134,33,192,32,13,233,105,194,163,41,100,110,207,45,16,122,0,225,204,178,177,181,80,157,10,73,29,164,61,2,211,202,117,114,80,254,141,80,185,152,48,123,53,69,193,105,42,198,145,237,231,161,76,17,23,181,251,234,233,202,88,3,105,106,19,20,101,10,76), +string.char(177,86,174,72,77,19,94,56,1,170,225,24,73,194,222,59,149,13,185,128,2,33,114,69,75,204,96,39,59,103,20,47,155,200,222,26,225,141,205,14,188,92,201,137,153,252,115,133,33,173,96,227,82,28,166,21,138,32,131,41,158,225,49,161,112,178,178,134,166,166,38,2,63,6,162,125,135,246,136,8,143,96,138,179,54,76,227,146,144,148,170,201,134,59,161,235,254,18,37,112), +string.char(120,168,234,104,192,129,91,176,179,204,153,224,36,73,200,90,90,186,58,112,178,205,217,140,134,149,138,87,231,203,4,214,219,144,78,214,170,70,247,44,247,141,145,244,50,68,128,191,249,252,172,202,118,11,57,126,143,101,25,193,33,76,247,78,135,231,91,73,132,74,110,58,73,73,209,131,172,203,214,255,4,242,90,112,212,86,76,34,223,135,103,131,17,48,210,55,129,150,86,166), +string.char(180,39,144,135,127,240,75,156,191,127,144,183,39,254,222,252,161,93,127,122,6,151,158,10,177,157,42,146,170,158,228,48,163,55,213,206,178,40,238,51,61,38,41,41,9,141,155,54,230,85,60,169,199,177,102,194,244,198,68,23,230,145,154,136,139,214,226,20,213,115,62,7,97,90,209,81,26,235,50,207,1,111,38,33,57,14,59,175,44,69,197,223,90,243,184,153,94,22,115,107), +string.char(137,136,79,138,197,217,123,251,240,232,173,16,57,39,127,83,187,234,189,153,229,148,53,43,80,204,222,40,217,166,58,194,227,85,131,174,133,204,5,229,152,252,60,162,149,150,206,134,73,77,77,153,176,104,104,202,238,37,9,43,67,206,200,217,223,137,188,22,156,40,242,208,42,131,222,88,115,201,219,243,246,133,33,87,36,229,65,195,145,251,172,10,24,187,217,131,183,158,139,106), +string.char(163,250,72,35,252,190,222,93,186,143,109,101,225,1,196,251,135,113,29,165,168,77,73,118,94,93,236,56,123,154,167,218,52,111,217,156,189,141,169,216,117,255,22,146,204,140,81,37,179,26,223,255,210,195,99,120,250,225,54,42,246,108,194,29,123,61,234,15,87,41,1,119,237,233,105,220,122,113,1,47,222,251,72,219,105,111,33,47,188,80,5,39,30,182,80,118,1,16,168,84), +string.char(11,93,251,193,235,27,209,105,122,101,140,219,226,137,23,159,238,241,109,100,98,183,171,229,245,205,155,21,18,46,148,150,35,174,17,33,73,41,229,89,154,123,198,122,21,145,183,77,61,181,60,189,41,67,85,175,201,147,244,135,188,22,28,159,192,155,207,84,86,166,179,7,106,164,175,158,103,148,154,150,130,153,59,133,73,63,44,170,57,161,234,159,170,147,191,209,132,237,98,101), +string.char(79,77,67,109,174,115,152,26,90,160,255,19,23,220,45,108,142,225,207,239,96,238,131,27,24,27,240,24,119,116,51,209,231,106,41,20,182,44,131,187,175,46,99,252,214,95,248,113,37,90,184,34,244,145,63,106,150,107,162,112,110,82,178,135,175,105,135,33,43,91,162,199,130,234,210,54,107,239,96,190,157,4,150,56,197,54,230,50,211,57,41,37,17,203,14,79,130,235,112,125), +string.char(116,153,227,130,89,123,6,33,36,250,29,44,221,74,112,198,33,197,173,8,46,37,212,23,183,20,127,247,157,151,255,240,207,230,146,120,158,188,149,165,2,137,176,36,198,107,113,71,168,40,60,143,215,158,202,250,152,92,68,94,11,206,117,191,29,87,21,194,254,4,170,38,225,39,49,169,149,241,37,58,132,247,8,154,6,218,232,125,125,41,154,254,245,43,70,37,28,67,79,159), +string.char(229,168,57,221,19,174,227,187,160,116,251,90,178,115,81,60,70,114,215,42,217,86,195,204,0,119,120,93,43,142,242,175,237,209,245,31,59,204,254,208,5,206,14,21,113,244,214,86,252,186,188,41,239,202,93,126,111,15,91,151,18,120,178,245,60,74,21,174,160,240,253,41,146,20,227,74,195,218,240,26,198,213,167,116,231,214,27,5,41,91,108,27,13,251,234,101,248,27,47,95), +string.char(151,153,252,59,155,255,153,207,63,83,16,179,211,153,153,24,240,102,19,250,222,89,206,135,84,195,66,230,168,226,84,15,38,89,84,230,10,139,10,230,47,203,193,91,235,248,223,133,42,57,193,190,113,69,21,203,202,80,238,101,35,93,142,4,43,49,94,19,254,47,141,164,147,193,165,71,37,201,250,24,225,18,115,111,162,10,57,228,181,224,208,196,77,175,2,111,189,80,88,73), +string.char(65,189,19,247,182,34,50,86,181,24,56,197,130,42,57,214,66,70,98,26,62,63,9,224,33,9,154,147,201,206,181,52,234,207,232,133,198,11,6,64,223,66,118,3,99,124,131,21,236,9,51,35,75,238,151,105,102,210,20,85,74,214,97,58,198,45,12,95,233,142,105,59,5,186,66,201,30,245,81,127,90,79,158,4,24,253,244,35,172,149,42,114,133,73,170,106,217,48,193,114), +string.char(27,225,206,134,72,119,212,24,217,137,115,156,137,222,73,161,10,229,225,192,74,46,205,151,2,161,228,63,249,116,231,21,110,204,20,130,172,15,215,159,150,250,138,148,241,53,58,20,163,86,119,193,241,187,91,165,235,72,208,28,235,87,100,215,162,72,224,183,144,43,170,68,145,253,168,184,112,158,253,64,78,71,93,61,217,93,72,23,74,244,67,162,38,168,79,155,248,73,228,135), +string.char(85,181,248,201,150,115,10,33,127,138,68,215,152,214,3,135,110,108,86,49,59,11,91,59,97,235,184,43,152,217,115,51,206,53,251,11,171,29,189,112,127,229,49,177,26,131,10,180,205,244,153,190,161,232,232,162,161,99,227,233,249,168,53,202,20,195,216,176,227,237,39,76,232,74,243,116,183,89,55,146,123,84,137,224,173,165,161,13,99,3,69,250,69,178,164,199,161,153,131,41), +string.char(54,181,198,198,19,107,169,68,109,255,37,210,158,83,157,127,164,113,69,65,255,162,10,164,177,99,159,162,198,153,210,168,112,167,24,124,230,28,192,243,53,103,80,68,141,82,76,47,201,234,227,51,240,44,80,8,76,90,215,42,205,115,212,169,166,31,145,186,148,157,128,242,67,151,88,42,134,168,218,228,206,48,212,149,57,27,57,167,154,157,251,217,74,62,108,249,168,189,113,63), +string.char(137,252,16,156,125,239,143,222,125,26,112,254,129,194,202,106,67,218,97,253,195,191,176,233,204,2,254,230,200,131,28,103,157,235,245,199,149,197,65,56,50,226,33,92,254,113,194,154,66,158,120,190,235,18,247,218,202,35,45,50,17,86,74,37,65,200,114,90,113,114,146,244,111,18,152,62,207,214,160,206,196,30,210,28,235,68,38,136,253,154,143,83,185,216,106,165,235,193,103,101), +string.char(2,188,151,69,227,241,234,84,92,91,20,134,233,191,108,224,147,177,62,92,39,232,15,202,215,64,152,224,185,12,251,38,220,199,157,191,163,177,104,208,30,120,52,26,140,6,46,109,164,219,229,39,1,17,241,254,179,31,14,221,94,207,63,147,46,228,117,113,33,156,26,11,85,80,141,88,175,227,31,252,74,97,255,104,201,125,210,54,212,227,243,95,124,137,17,220,10,17,97,108), +string.char(52,202,84,10,63,200,134,210,159,159,59,82,13,242,67,112,200,52,152,120,194,125,118,230,167,219,47,165,102,36,233,57,125,110,47,195,205,210,62,168,63,214,26,123,46,173,194,103,165,212,16,10,254,57,217,58,99,80,219,137,56,63,247,61,50,150,124,226,209,112,145,162,144,85,74,13,13,87,162,5,210,238,224,68,46,48,34,159,87,4,89,70,54,230,234,185,65,100,106,27), +string.char(233,155,50,235,76,27,150,108,120,136,148,188,249,58,146,25,239,94,237,187,10,3,61,69,65,112,176,42,198,131,146,242,61,152,188,231,88,93,138,110,96,152,63,95,86,29,235,206,117,33,121,94,15,13,199,137,41,170,22,41,33,200,251,21,94,31,189,137,136,24,217,80,175,171,208,235,230,125,30,121,126,57,0,41,243,191,239,190,186,99,19,14,117,153,205,249,46,4,98,242), +string.char(17,13,180,207,211,53,184,92,201,27,205,38,58,226,200,205,45,42,61,16,129,30,204,100,175,149,120,191,240,6,158,239,185,204,215,137,121,218,186,58,170,94,220,78,181,133,233,127,178,186,133,36,124,150,106,138,23,169,195,171,15,66,45,98,235,178,130,163,48,37,52,134,91,113,217,65,62,111,75,93,25,151,4,73,134,133,179,123,93,213,99,153,18,127,233,209,49,133,117,150), +string.char(166,130,142,115,127,246,126,180,142,105,141,110,13,100,41,56,10,57,98,25,255,29,193,33,236,96,205,246,195,177,187,179,182,85,24,34,45,3,66,247,147,172,136,250,83,189,208,207,119,3,206,148,190,132,49,107,61,216,120,174,170,211,208,219,79,220,154,11,125,255,230,73,253,25,105,66,239,165,238,33,90,137,74,111,22,189,18,197,141,114,82,233,52,33,41,22,255,60,57,200), +string.char(63,235,155,171,143,187,41,131,28,132,212,75,197,196,203,124,111,234,122,71,177,119,74,137,81,45,81,103,96,101,134,160,72,127,133,117,213,203,52,226,220,156,251,43,18,240,107,155,241,176,183,82,159,137,42,78,106,111,80,130,11,90,238,149,85,149,67,126,135,28,232,21,155,158,153,145,185,254,206,162,131,42,27,41,213,133,50,18,210,220,77,113,255,245,53,181,39,40,227,40), +string.char(196,121,252,207,222,103,74,106,214,165,252,179,67,144,247,75,56,218,168,82,85,200,202,17,91,120,204,103,132,201,165,24,83,78,86,86,243,78,220,122,126,30,115,118,13,195,108,214,92,134,104,162,193,216,66,24,178,170,149,116,59,85,163,48,181,76,227,220,32,17,85,74,214,230,22,228,227,205,231,56,211,144,120,62,98,109,100,145,144,165,92,140,129,156,157,250,186,170,94,107), +string.char(83,35,217,203,35,70,226,171,9,92,228,58,217,220,138,31,66,65,149,57,217,40,209,248,85,64,142,61,74,180,35,135,152,58,136,30,103,98,13,82,175,209,164,146,170,131,144,240,173,190,132,44,60,242,228,22,82,154,110,136,76,247,198,227,237,165,173,209,56,59,180,159,94,70,186,157,166,42,74,142,17,244,14,229,2,214,211,182,15,196,254,155,107,112,128,53,130,142,181,17), +string.char(202,246,109,74,249,77,104,178,246,55,118,238,155,72,74,208,132,190,65,58,12,37,69,146,140,13,204,176,120,200,62,116,201,236,138,175,131,238,32,115,109,8,110,205,217,35,157,218,154,64,22,162,8,18,34,101,61,80,4,241,117,220,74,54,226,159,69,193,145,48,0,179,159,115,224,7,80,80,130,51,202,101,100,123,181,27,136,64,254,140,89,79,69,109,75,169,221,158,146,42), +string.char(84,11,77,254,20,201,205,226,146,14,170,213,57,9,202,165,84,228,65,165,65,136,4,101,105,170,168,227,80,184,128,64,41,51,148,166,226,38,201,76,160,214,100,205,111,124,170,68,241,161,200,59,227,40,52,16,22,43,232,109,67,67,119,227,143,180,83,248,253,203,65,180,92,51,156,19,213,203,118,169,135,35,119,54,33,130,153,243,105,169,26,156,70,34,130,116,183,62,45,198), +string.char(96,241,224,125,232,215,242,15,60,95,123,150,207,70,35,142,162,242,213,199,174,62,62,197,245,192,74,67,52,48,122,77,55,220,124,126,78,225,250,171,150,170,199,151,98,175,37,201,0,249,159,229,28,203,131,204,134,83,69,90,87,235,89,103,130,44,74,76,230,45,149,94,165,76,4,162,84,246,117,24,130,178,142,234,95,148,224,112,97,50,120,202,175,250,22,228,169,154,202,32), +string.char(66,122,159,166,99,21,122,13,210,65,142,221,218,198,63,127,245,253,200,149,83,114,253,23,169,85,30,110,191,117,64,181,161,237,36,251,73,126,136,174,172,116,218,231,72,153,163,238,253,165,199,184,62,115,39,118,55,27,143,229,134,157,56,209,252,249,174,203,112,234,92,11,111,131,95,32,58,66,7,6,70,233,40,228,160,218,163,90,73,4,217,103,213,113,145,189,167,160,27,217), +string.char(89,202,82,164,162,59,106,225,209,27,245,101,94,18,37,62,47,67,27,238,219,81,37,122,231,2,242,91,112,102,148,250,165,65,91,247,221,19,184,57,78,150,13,205,243,116,168,243,76,108,44,222,15,153,179,62,96,70,153,197,232,213,92,125,109,96,114,22,30,191,45,84,247,36,79,242,183,10,92,136,155,148,75,170,16,168,199,113,86,226,196,144,162,188,116,200,1,236,29,127), +string.char(15,3,18,7,160,213,211,134,40,117,216,2,233,51,223,241,235,147,206,203,160,230,75,205,140,45,49,185,199,26,12,105,61,29,141,46,85,195,144,216,33,88,91,119,31,39,152,19,174,143,217,132,42,253,91,114,193,12,13,210,66,70,134,6,108,11,39,195,177,84,18,76,45,210,160,169,149,9,29,221,76,238,2,32,16,87,136,122,76,171,234,37,241,53,74,166,99,201,83,75), +string.char(109,42,21,87,201,149,215,209,214,225,2,47,206,101,37,201,114,80,63,83,238,79,34,191,5,103,4,41,191,98,200,224,237,217,123,72,152,250,18,115,203,174,192,189,229,113,152,211,111,11,15,58,106,101,81,94,237,225,155,155,82,215,188,99,93,138,49,101,111,118,82,78,182,50,40,251,211,214,66,181,7,39,43,165,98,241,234,156,64,222,161,118,47,12,235,56,3,94,77,135), +string.char(243,7,41,95,83,88,25,68,203,232,209,104,40,223,191,77,141,30,168,81,182,17,31,66,229,205,125,167,38,85,225,109,114,15,247,94,222,68,240,71,193,95,99,102,145,202,185,205,229,171,197,193,169,140,82,36,156,9,128,69,73,7,102,89,203,126,163,142,220,252,232,244,210,41,151,238,39,255,85,166,156,41,46,87,136,60,215,139,15,228,167,224,208,119,25,202,103,88,146,133), +string.char(82,183,98,43,110,93,40,59,212,148,65,81,107,138,64,19,138,182,119,131,189,171,51,127,152,217,78,196,170,198,228,166,88,147,97,54,223,39,34,82,153,251,251,131,239,46,13,57,245,167,247,196,168,53,93,225,251,58,4,97,193,122,74,219,53,20,75,189,176,235,126,187,255,6,239,205,212,94,87,64,8,222,124,82,100,30,88,73,106,230,136,204,1,57,218,110,238,77,145,35), +string.char(65,126,10,142,138,210,65,188,28,81,103,249,22,98,19,163,177,112,223,24,105,76,167,225,172,62,60,185,158,142,87,71,234,166,50,179,164,79,16,148,159,115,90,98,50,30,173,60,206,205,99,117,147,109,40,67,36,168,75,125,62,63,232,91,35,154,106,145,218,229,81,115,77,79,140,223,216,19,87,111,63,71,96,128,1,210,36,36,54,90,134,73,116,37,155,186,101,165,223,165), +string.char(47,71,24,147,247,59,81,22,231,237,215,130,114,76,67,56,17,223,30,50,203,141,96,100,35,139,91,105,26,115,143,178,250,176,252,79,32,191,135,170,79,241,161,50,167,24,37,203,251,248,169,247,215,136,32,243,115,246,174,223,112,242,190,160,219,180,221,63,65,58,31,148,1,51,143,119,95,91,206,153,118,68,188,186,224,115,8,151,30,30,197,194,253,99,112,229,153,64,104,215), +string.char(85,170,254,69,166,110,70,124,10,250,45,109,200,222,254,46,56,113,103,167,148,105,120,249,209,113,78,218,10,137,8,228,62,28,226,50,139,86,141,145,189,240,230,139,243,67,36,170,153,87,130,242,185,18,147,227,17,147,16,137,79,95,2,112,215,247,178,202,62,52,209,154,193,224,226,232,62,223,21,219,78,236,196,189,91,201,240,125,108,12,127,223,76,92,127,38,208,69,139,54), +string.char(146,233,95,159,190,190,227,47,7,249,149,34,98,100,129,110,195,66,130,44,204,223,59,10,93,102,86,229,196,183,61,215,132,130,154,242,73,121,85,135,119,160,133,122,146,243,79,32,191,5,231,89,244,7,217,143,167,135,241,232,221,13,133,135,64,93,58,221,164,23,31,30,48,101,114,59,55,63,207,62,216,3,109,115,125,110,38,151,243,144,149,185,51,47,110,207,35,202,196,180), +string.char(35,226,213,31,155,186,97,212,134,206,210,27,72,219,168,224,144,60,40,119,188,211,233,153,48,40,94,8,87,159,31,199,228,237,189,165,76,67,98,24,18,105,171,197,164,162,220,135,83,229,55,29,76,220,38,100,51,136,229,231,72,169,167,97,100,252,6,47,28,188,182,1,87,30,159,192,185,251,7,120,148,187,253,148,242,168,241,187,49,234,142,177,68,235,169,37,49,97,171,140), +string.char(241,71,66,78,160,97,164,230,152,174,124,14,45,162,122,212,29,99,141,5,123,198,99,218,150,97,88,123,102,186,228,119,217,73,5,180,215,162,58,168,63,182,16,247,43,117,158,35,123,254,198,204,212,166,10,94,123,216,139,19,16,38,163,173,184,78,236,134,82,109,101,132,49,201,236,121,50,103,84,46,33,191,5,231,20,89,81,34,168,20,135,117,109,103,110,109,156,185,187,23), +string.char(139,15,252,9,151,161,154,252,38,245,152,239,134,169,59,251,242,253,42,12,110,5,207,171,139,121,101,9,121,144,67,174,253,214,177,220,193,230,50,178,157,64,184,154,218,29,29,79,78,199,47,222,75,209,237,232,12,105,224,144,148,73,210,137,168,60,74,137,150,110,24,234,183,133,87,123,24,224,191,25,158,119,150,162,211,217,89,104,189,251,79,78,222,50,44,161,232,223,41,247), +string.char(107,115,105,84,157,82,99,90,110,31,141,43,207,143,113,86,224,200,117,29,241,231,230,238,88,119,118,38,62,132,191,86,56,174,242,232,14,104,181,107,44,39,161,201,15,51,52,204,210,28,90,158,119,150,160,238,95,125,113,50,230,8,78,220,219,198,183,145,176,59,183,175,197,191,167,209,202,33,106,111,34,253,78,155,74,37,208,102,227,104,116,187,60,159,87,196,232,113,123,49), +string.char(70,68,31,66,227,121,253,120,65,2,17,18,67,36,215,159,115,126,11,206,73,223,45,23,21,106,243,182,92,254,27,230,237,31,198,41,157,219,47,47,86,216,217,109,178,7,47,113,210,122,237,8,41,199,152,240,241,218,83,238,36,163,120,151,85,89,71,78,209,104,193,206,211,120,94,127,174,255,148,110,87,147,79,198,97,36,119,3,159,237,186,136,117,133,123,97,75,245,17,60), +string.char(92,65,250,3,221,96,154,123,161,112,173,178,188,16,65,133,95,26,243,9,232,127,243,223,134,223,227,143,114,103,30,149,62,161,137,233,229,81,166,83,61,222,99,80,85,47,17,198,101,236,57,75,176,221,161,73,24,240,118,19,198,164,156,224,1,220,138,94,77,184,235,64,29,40,119,189,246,159,30,232,231,189,92,82,182,100,23,188,46,253,37,250,95,224,198,134,25,229,178,38), +string.char(116,93,244,59,245,205,141,56,55,185,88,227,202,188,34,6,233,79,34,87,89,30,34,33,63,183,145,223,130,67,238,213,27,129,55,101,180,81,251,234,206,24,157,116,28,253,94,173,231,117,103,122,62,88,193,111,32,189,61,141,230,244,229,126,147,219,11,246,75,73,84,97,79,223,225,64,163,9,9,167,61,22,28,221,80,180,15,255,91,4,89,75,84,230,100,177,70,27,60,218), +string.char(120,70,161,66,68,164,63,247,135,12,142,123,29,242,203,209,54,211,195,188,23,31,252,102,194,26,249,127,140,120,89,17,75,233,3,161,106,21,68,146,162,74,23,212,99,180,219,242,135,180,142,205,16,223,205,188,148,9,121,137,201,140,214,212,81,239,82,160,50,45,202,124,30,242,217,8,165,75,44,85,252,78,242,101,77,120,73,21,53,126,169,172,64,47,150,247,146,67,244,241), +string.char(124,142,15,202,33,10,34,228,112,129,38,121,167,33,67,4,41,115,212,115,80,221,25,187,106,165,248,13,164,174,218,255,156,15,175,22,113,123,210,142,0,186,1,36,8,183,230,241,185,195,151,179,70,65,170,121,15,214,202,102,134,33,239,51,185,236,25,90,95,26,180,234,35,77,218,33,226,195,85,238,140,35,94,40,157,160,6,59,103,248,155,211,119,85,46,142,170,72,80,192), +string.char(145,150,242,248,242,226,3,103,4,174,182,238,142,215,71,110,169,45,91,66,111,55,245,102,193,247,124,185,144,197,135,202,56,195,20,31,123,185,255,26,86,91,117,199,238,38,227,16,250,232,173,194,156,154,226,245,83,212,95,156,132,77,132,223,177,219,220,163,78,247,45,228,193,27,149,227,8,164,27,82,9,23,145,105,73,203,139,127,172,39,122,44,21,40,60,162,114,192,79,162), +string.char(32,4,103,205,87,239,55,31,238,45,63,138,132,175,89,179,248,137,110,122,180,245,52,26,211,200,44,104,226,51,247,64,196,17,143,57,240,223,127,147,236,247,57,146,221,142,189,88,39,139,215,72,166,75,124,201,26,173,28,122,125,214,78,94,25,130,156,98,95,110,113,253,67,76,86,162,115,120,157,232,56,59,147,138,18,137,66,240,249,177,63,182,148,25,148,198,132,245,43,91), +string.char(242,191,69,80,242,30,3,209,24,219,159,236,50,247,211,249,17,171,248,112,73,237,195,213,167,184,62,125,7,150,25,185,131,245,102,31,247,212,28,19,68,66,70,148,211,43,19,55,131,8,108,212,19,158,233,177,144,170,30,117,248,124,227,213,180,93,213,70,70,237,105,62,65,202,41,162,123,65,222,243,45,206,3,177,175,213,36,41,237,132,82,120,40,199,254,245,246,203,127,220), +string.char(248,99,243,234,221,110,191,135,239,107,51,89,161,182,15,85,39,219,94,103,20,86,49,161,220,94,119,52,94,51,65,163,250,58,1,7,111,147,163,103,228,143,60,164,236,80,16,130,67,194,208,208,123,234,238,11,7,218,79,229,5,137,196,162,69,52,156,80,129,165,103,59,47,225,226,192,149,180,47,241,18,168,75,161,7,221,230,253,177,187,163,216,178,17,4,86,33,65,221,88), +string.char(35,134,147,207,68,63,249,248,226,3,211,135,228,134,6,121,47,12,117,223,141,78,118,158,27,126,204,107,62,39,135,61,221,198,185,201,219,32,112,88,150,190,187,168,56,85,145,4,20,214,47,243,98,253,249,37,108,184,60,198,218,190,131,141,39,248,220,155,181,239,40,59,123,39,8,5,170,137,241,69,218,112,145,7,11,14,111,217,87,119,236,3,223,173,151,168,151,116,149,252), +string.char(158,217,172,89,132,223,123,123,67,36,242,71,191,231,130,240,158,53,77,38,88,27,246,181,158,132,171,83,182,241,152,23,132,30,118,41,107,84,169,192,154,189,4,235,168,102,207,237,5,251,120,14,252,222,218,99,200,187,61,134,109,171,22,249,240,93,175,147,157,230,92,189,61,113,7,249,2,40,146,156,123,243,114,203,161,160,162,227,36,8,45,217,143,29,113,168,217,228,64,177), +string.char(104,209,230,82,3,112,164,229,212,103,231,123,47,185,200,182,147,73,113,95,238,24,26,87,232,6,190,255,142,239,153,192,122,21,78,55,101,160,87,251,177,210,246,235,172,21,99,111,230,208,115,94,139,55,60,89,126,146,186,245,37,146,109,231,238,45,57,44,85,46,149,72,95,36,184,52,115,44,9,138,39,107,148,231,75,67,39,81,246,228,133,147,60,122,68,69,116,99,109,20), +string.char(100,2,47,226,149,18,191,39,86,114,252,224,152,23,65,30,172,151,29,159,24,240,165,153,228,88,121,12,101,173,2,19,142,191,206,254,178,104,131,228,239,191,89,35,170,226,46,214,40,165,130,52,247,236,189,171,63,136,130,158,118,136,74,118,22,149,92,135,24,140,35,103,5,85,209,94,159,11,231,167,222,161,229,157,201,59,232,230,246,131,250,148,100,34,216,80,66,211,96,73), +string.char(19,139,232,253,147,18,26,19,69,174,253,60,132,148,100,108,44,84,9,149,159,110,134,152,110,127,177,118,41,139,99,105,72,166,238,136,174,121,93,30,93,95,150,40,104,193,17,145,59,36,89,161,87,80,78,17,165,241,135,110,238,143,40,136,103,252,207,9,157,158,100,184,203,122,114,171,31,67,73,145,55,164,47,155,23,61,79,170,132,230,54,254,45,130,147,43,208,19,156,116), +string.char(206,217,237,247,29,152,115,115,236,150,140,143,215,159,113,243,30,194,208,150,155,144,234,104,114,25,14,57,139,190,22,48,254,83,130,147,7,160,196,166,161,7,26,142,135,239,150,75,20,65,156,144,203,231,255,168,134,112,150,115,71,77,1,226,127,93,112,56,147,73,84,96,53,181,243,228,231,144,126,68,81,85,82,54,85,29,40,63,7,127,242,219,16,82,100,19,161,100,77,252), +string.char(249,23,225,127,93,112,56,33,37,41,90,32,144,107,27,242,238,62,47,102,167,12,64,222,100,68,94,167,42,166,36,248,68,103,133,160,168,251,102,115,204,191,2,255,235,130,67,253,252,23,177,104,180,156,201,156,111,115,83,254,36,200,98,242,37,39,226,199,107,252,29,200,109,29,42,207,240,191,46,56,132,123,81,1,66,133,9,35,91,41,129,201,169,160,46,230,7,48,232,124), +string.char(159,165,161,151,6,175,162,96,220,180,130,190,152,156,226,191,32,56,167,201,197,254,63,12,154,18,143,18,188,136,231,153,39,149,37,242,2,255,5,193,57,244,122,219,229,84,10,16,202,209,53,62,127,235,128,255,195,207,227,191,32,56,20,13,220,123,109,250,14,62,249,6,3,133,199,213,167,59,254,31,114,13,255,5,193,33,12,245,219,113,101,194,131,5,135,196,137,76,255,15), +string.char(121,140,255,138,224,144,127,101,33,107,84,41,41,251,9,204,255,15,63,141,255,7,104,194,255,34,35,114,240,96,0,0,0,0,73,69,78,68,174,66,96,130), + }) +} diff --git a/reascripts/ReaSpeech/resources/images/reaspeech-logo-small.png b/reascripts/ReaSpeech/resources/images/reaspeech-logo-small.png new file mode 100644 index 0000000000000000000000000000000000000000..f96beda0f6712ba6cd4a6cdfbf102d9293124728 Binary files /dev/null and b/reascripts/ReaSpeech/resources/images/reaspeech-logo-small.png differ diff --git a/reascripts/ReaSpeech/resources/images/reaspeech-logo.lua b/reascripts/ReaSpeech/resources/images/reaspeech-logo.lua new file mode 100644 index 0000000000000000000000000000000000000000..e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 diff --git a/reascripts/ReaSpeech/resources/images/reaspeech-logo.png b/reascripts/ReaSpeech/resources/images/reaspeech-logo.png new file mode 100644 index 0000000000000000000000000000000000000000..e3fcba10b5cf8ab2e77dba8ebe2e8794f125fca1 Binary files /dev/null and b/reascripts/ReaSpeech/resources/images/reaspeech-logo.png differ diff --git a/reascripts/ReaSpeech/source/AlertPopup.lua b/reascripts/ReaSpeech/source/AlertPopup.lua new file mode 100644 index 0000000000000000000000000000000000000000..b289e1b862557a4035b6ccbf11812b8111091f12 --- /dev/null +++ b/reascripts/ReaSpeech/source/AlertPopup.lua @@ -0,0 +1,59 @@ +--[[ + + AlertPopup.lua - Alert popup UI + +]]-- + +AlertPopup = Polo { + WIDTH = 400, + HEIGHT = 200, + MIN_CONTENT_WIDTH = 375, + BUTTON_WIDTH = 120, + DEFAULT_TITLE = 'Alert', +} + +function AlertPopup:init() + self.title = self.title or self.DEFAULT_TITLE + self.msg = '' +end + +function AlertPopup:show(msg) + self.msg = msg + self:open() +end + +function AlertPopup:render() + local center = {ImGui.Viewport_GetCenter(ImGui.GetWindowViewport(ctx))} + ImGui.SetNextWindowPos(ctx, center[1], center[2], ImGui.Cond_Appearing(), 0.5, 0.5) + ImGui.SetNextWindowSize(ctx, self.WIDTH, self.HEIGHT, ImGui.Cond_FirstUseEver()) + + if ImGui.BeginPopupModal(ctx, self.title, true, ImGui.WindowFlags_AlwaysAutoResize()) then + app:trap(function () self:render_content() end) + ImGui.EndPopup(ctx) + else + self:close() + end +end + +function AlertPopup:render_content() + ImGui.Text(ctx, self.msg) + self:render_separator() + if ImGui.Button(ctx, 'OK', self.BUTTON_WIDTH, 0) then + self:close() + end +end + +function AlertPopup:render_separator() + ImGui.Dummy(ctx, self.MIN_CONTENT_WIDTH, 0) + ImGui.Separator(ctx) + ImGui.Dummy(ctx, 0, 0) +end + +function AlertPopup:open() + ImGui.OpenPopup(ctx, self.title) +end + +function AlertPopup:close() + ImGui.CloseCurrentPopup(ctx) + if self.onclose then self.onclose() end +end diff --git a/reascripts/ReaSpeech/source/CSVWriter.lua b/reascripts/ReaSpeech/source/CSVWriter.lua new file mode 100644 index 0000000000000000000000000000000000000000..31ee8e3cce4658bb66f35585c54aaf2c211a820e --- /dev/null +++ b/reascripts/ReaSpeech/source/CSVWriter.lua @@ -0,0 +1,85 @@ +--[[ + + CSVWriter.lua - Write CSV files + +]]-- + +CSVWriter = Polo { + TIME_FORMAT = '%02d:%02d:%02d,%03d', + DELIMITERS = { + { char = ',', name = 'Comma' }, + { char = ';', name = 'Semicolon' }, + { char = '\t', name = 'Tab' }, + }, + + new = function(options) + options = options or {} + return { + file = options.file, + delimiter = options.delimiter or ',', + include_header_row = options.include_header_row or false, + } + end, +} + +function CSVWriter:init() + assert(self.file, 'missing file') +end + +CSVWriter.format_time = function (time) + local milliseconds = math.floor(time * 1000) % 1000 + local seconds = math.floor(time) % 60 + local minutes = math.floor(time / 60) % 60 + local hours = math.floor(time / 3600) + return string.format(CSVWriter.TIME_FORMAT, hours, minutes, seconds, milliseconds) + end + +function CSVWriter:write(transcript) + if self.include_header_row then + self:write_header_row() + end + + local sequence_number = 1 + for _, segment in pairs(transcript:get_segments()) do + self:write_segment(segment, sequence_number) + sequence_number = sequence_number + 1 + end +end + +function CSVWriter:write_header_row() + local fields = { + CSVWriter._quoted('Sequence Number'), + CSVWriter._quoted('Start Time'), + CSVWriter._quoted('End Time'), + CSVWriter._quoted('Text'), + CSVWriter._quoted('File'), + } + + self.file:write(table.concat(fields, self.delimiter)) + self.file:write('\n') +end + +function CSVWriter:write_segment(segment, sequence_number) + local start = segment:get('start') + local end_ = segment:get('end') + local text = segment:get('text') + local file = segment:get_file_with_extension() + self:write_line(text, sequence_number, start, end_, file) +end + +function CSVWriter:write_line(line, sequence_number, start, end_, file) + local fields = { + sequence_number, + CSVWriter._quoted(CSVWriter.format_time(start)), + CSVWriter._quoted(CSVWriter.format_time(end_)), + CSVWriter._quoted(line), + CSVWriter._quoted(file), + } + + self.file:write(table.concat(fields, self.delimiter)) + self.file:write('\n') +end + +function CSVWriter._quoted(input_string) + return table.concat({'"', input_string:gsub('"', '""'), '"'}) +end diff --git a/reascripts/ReaSpeech/source/ExecProcess.lua b/reascripts/ReaSpeech/source/ExecProcess.lua new file mode 100644 index 0000000000000000000000000000000000000000..fefe885288f45ffabfba122a7e1764dbb6b8db71 --- /dev/null +++ b/reascripts/ReaSpeech/source/ExecProcess.lua @@ -0,0 +1,52 @@ +--[[ + + ExecProcess.lua - wrapper for process execution in REAPER + +]]-- + +ExecProcess = Polo { + WAIT_FOREVER = 0, + NO_WAIT = -1, + BACKGROUND = -2, + + new = function(command_string_list) + return { + commands = command_string_list + } + end +} + +ExecProcess.command_prefix = function() + if reaper.GetOS():match('Win') then + return 'cmd /c ' + end + + return '' +end + +ExecProcess.command_separator = function() + if reaper.GetOS():match('Win') then + return ' & ' + end + + return ' ; ' +end + +function ExecProcess:run(timeout) + local command_prefix = self.command_prefix() + local compound_command = command_prefix .. table.concat(self.commands, self.command_separator()) + + return reaper.ExecProcess(compound_command, timeout) +end + +function ExecProcess:wait(timeout) + return self:run(timeout or self.WAIT_FOREVER) +end + +function ExecProcess:no_wait() + return self:run(self.NO_WAIT) +end + +function ExecProcess:background() + return self:run(self.BACKGROUND) +end \ No newline at end of file diff --git a/reascripts/ReaSpeech/source/Fonts.lua b/reascripts/ReaSpeech/source/Fonts.lua new file mode 100644 index 0000000000000000000000000000000000000000..a2827e524df4e683e8255f4ddd8e1be0b3800553 --- /dev/null +++ b/reascripts/ReaSpeech/source/Fonts.lua @@ -0,0 +1,52 @@ +--[[ + + Fonts.lua - Font configuration and loader + +]]-- + +Fonts = { + SIZE = 15, + ICON = { + pencil = 'a', + cog = 'b', + play = 'c', + stop = 'd', + }, + LOCAL_FILE = nil, +} + +function Fonts:load() + self.main = ImGui.CreateFont('sans-serif', self.SIZE) + ImGui.Attach(ctx, self.main) + + if self.LOCAL_FILE then + self.icons = ImGui.CreateFont(self.LOCAL_FILE, self.SIZE) + ImGui.Attach(ctx, self.icons) + return + end + + if not Script or not Script.host or Script.host == '' then + return + end + + local icons_url = 'http://' .. Script.host .. '/static/reascripts/ReaSpeech/icons.ttf' + local icons_file = Tempfile:name() + + local curl = "curl" + if not reaper.GetOS():find("Win") then + curl = "/usr/bin/curl" + end + local command = ( + curl + .. ' "' .. icons_url .. '"' + .. ' -o "' .. icons_file .. '"' + ) + + if reaper.ExecProcess(command, 5000) then + self.icons = ImGui.CreateFont(icons_file, self.SIZE) + ImGui.Attach(ctx, self.icons) + else + self.icons = self.main + end + +end diff --git a/reascripts/ReaSpeech/source/IntervalFunction.lua b/reascripts/ReaSpeech/source/IntervalFunction.lua new file mode 100644 index 0000000000000000000000000000000000000000..60b2febb415c2a306f5b4c58b70af58ef8ea18a5 --- /dev/null +++ b/reascripts/ReaSpeech/source/IntervalFunction.lua @@ -0,0 +1,45 @@ +--[[ + + IntervalFunction.lua - Interval functions that run every x seconds or ticks + + Examples: + + IntervalFunction.new(5, function() + -- run no more often than once every 5 seconds + -- chill interval to check on states that don't + -- need to feel so snappy + end), + + IntervalFunction.new(-15, function () + -- run every 15 ticks, ~0.5 seconds + -- maybe a good interval for updating some states + -- in a way that feels responsive, like selections + end) + +]]-- + +IntervalFunction = Polo { + new = function(interval, f) + return { + interval = interval, + f = f, + last = 0 + } + end +} + +function IntervalFunction:react(time) + if self.interval >= 0 then + if time - self.last >= self.interval then + self.f() + self.last = time + end + else + self.last = self.last - 1 + + if self.last < self.interval then + self.f() + self.last = 0 + end + end +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechAPI.lua b/reascripts/ReaSpeech/source/ReaSpeechAPI.lua new file mode 100644 index 0000000000000000000000000000000000000000..a1dd3b4de326db3b388dc7a1e6e8a93ba1ce5e0c --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechAPI.lua @@ -0,0 +1,249 @@ +--[[ + + ReaSpeechAPI.lua - ReaSpeech API client + +]]-- + +ReaSpeechAPI = { + CURL_TIMEOUT_SECONDS = 5, + base_url = nil, +} + +-- Initialize the module with the given base URL +-- Example: "http://localhost:9000" +function ReaSpeechAPI:init(base_url) + self.base_url = base_url +end + +function ReaSpeechAPI:get_api_url(remote_path) + local remote_path_no_leading_slash = remote_path:gsub("^/+", "") + return ("%s/%s"):format(self.base_url, url.quote(remote_path_no_leading_slash)) +end + +function ReaSpeechAPI:get_curl_cmd() + local curl = "curl" + if not reaper.GetOS():find("Win") then + curl = "/usr/bin/curl" + end + return curl +end + +-- Fetch simple JSON responses. Will block until result or curl timeout. +-- For large amounts of data, use fetch_large instead. +function ReaSpeechAPI:fetch_json(url_path, http_method, error_handler) + http_method = http_method or 'GET' + + local curl = self:get_curl_cmd() + local api_url = self:get_api_url(url_path) + + local http_method_argument = "" + if http_method ~= 'GET' then + http_method_argument = " -X " .. http_method + end + + local command = table.concat({ + curl, + ' "', api_url, '"', + ' -H "accept: application/json"', + http_method_argument, + ' -m ', self.CURL_TIMEOUT_SECONDS, + ' -s', + ' -i', + }) + + app:debug('Fetch JSON: ' .. command) + + local exec_result = (ExecProcess.new { command }):wait() + + if exec_result == nil then + local msg = "Unable to run curl" + app:log(msg) + error_handler(msg) + return nil + end + + local status, output = exec_result:match("(%d+)\n(.*)") + + if tonumber(status) ~= 0 then + local msg = "Curl failed with status " .. status + app:debug(msg) + error_handler(msg) + return nil + end + + local response_status, response_body = self.http_status_and_body(output) + + if response_status >= 400 then + local msg = "Request failed with status " .. response_status + app:log(msg) + error_handler(msg) + return nil + end + + local response_json = nil + if app:trap(function() + response_json = json.decode(response_body) + end) then + return response_json + else + app:log("JSON parse error") + app:log(output) + return nil + end +end + +-- Requests data that may be large or time-consuming. +-- This method is non-blocking, and does not give any indication that it has +-- completed. The path to the output file is returned. +function ReaSpeechAPI:fetch_large(url_path, http_method) + http_method = http_method or 'GET' + + local curl = self:get_curl_cmd() + local api_url = self:get_api_url(url_path) + + local http_method_argument = "" + if http_method ~= 'GET' then + http_method_argument = " -X " .. http_method + end + + local output_file = Tempfile:name() + local sentinel_file = Tempfile:name() + + local command = table.concat({ + curl, + ' "', api_url, '"', + ' -H "accept: application/json"', + http_method_argument, + ' -i ', + ' -o "', output_file, '"', + }) + + app:debug('Fetch large: ' .. command) + + local executor = ExecProcess.new { command, self.touch_cmd(sentinel_file) } + + if executor:background() then + return output_file, sentinel_file + else + app:log("Unable to run curl") + return nil + end +end + +ReaSpeechAPI.touch_cmd = function(filename) + if reaper.GetOS():find("Win") then + return 'echo. > "' .. filename .. '"' + else + return 'touch "' .. filename .. '"' + end +end + +-- Uploads a file to start a request for processing. +-- This method is non-blocking, and does not give any indication that it has +-- completed. The path to the output file is returned. +function ReaSpeechAPI:post_request(url_path, data, file_path) + local curl = self:get_curl_cmd() + local api_url = self:get_api_url(url_path) + + local query = {} + for k, v in pairs(data) do + table.insert(query, k .. '=' .. url.quote(v)) + end + + local output_file = Tempfile:name() + local sentinel_file = Tempfile:name() + + local command = table.concat({ + curl, + ' "', api_url, '?', table.concat(query, '&'), '"', + ' -H "accept: application/json"', + ' -H "Content-Type: multipart/form-data"', + ' -F ', self:_maybe_quote('audio_file=@"' .. file_path .. '"'), + ' -i ', + ' -o "', output_file, '"', + }) + + app:log(file_path) + app:debug('Post request: ' .. command) + + local executor = ExecProcess.new { command, self.touch_cmd(sentinel_file) } + + if executor:background() then + return output_file, sentinel_file + else + app:log("Unable to run curl") + return nil + end +end + +function ReaSpeechAPI:_maybe_quote(arg) + if reaper.GetOS():find("Win") then + return arg + else + return "'" .. arg .. "'" + end +end + +function ReaSpeechAPI.http_status_and_body(response) + local headers, content = ReaSpeechAPI._split_curl_response(response) + local last_status_line = headers[#headers] and headers[#headers][1] or '' + + local status = last_status_line:match("^HTTP/%d%.%d%s+(%d+)") + if not status then + return -1, 'Unable to parse response' + end + + local body = {} + for _, chunk in pairs(content) do + table.insert(body, table.concat(chunk, "\n")) + end + + return tonumber(status), table.concat(body, "\n") +end + +function ReaSpeechAPI._split_curl_response(input) + local line_iterator = ReaSpeechAPI._line_iterator(input) + local chunk_iterator = ReaSpeechAPI._chunk_iterator(line_iterator) + local header_chunks = {} + local content_chunks = {} + local in_header = true + for chunk in chunk_iterator do + if in_header and chunk[1] and chunk[1]:match("^HTTP/%d%.%d") then + table.insert(header_chunks, chunk) + else + in_header = false + table.insert(content_chunks, chunk) + end + end + return header_chunks, content_chunks +end + +function ReaSpeechAPI._line_iterator(input) + if type(input) == 'string' then + local i = 1 + local lines = {} + for line in input:gmatch("([^\n]*)\n?") do + table.insert(lines, line) + end + return function () + local line = lines[i] + i = i + 1 + return line + end + else + return input:lines() + end +end + +function ReaSpeechAPI._chunk_iterator(line_iterator) + return function () + local chunk = nil + while true do + local line = line_iterator() + if line == nil or line:match("^%s*$") then break end + chunk = chunk or {} + table.insert(chunk, line) + end + return chunk + end +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechActionsUI.lua b/reascripts/ReaSpeech/source/ReaSpeechActionsUI.lua new file mode 100644 index 0000000000000000000000000000000000000000..6786b6bca97b12e7fcfe23b4065faf0410d78f4a --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechActionsUI.lua @@ -0,0 +1,128 @@ +--[[ + +ReaSpeechActionsUI.lua - Main action bar UI in ReaSpeech + +]]-- + +ReaSpeechActionsUI = Polo {} + +function ReaSpeechActionsUI:init() + self.disabler = ReaUtil.disabler(ctx) +end + +function ReaSpeechActionsUI.pluralizer(count, suffix) + if count == 0 then + return '', suffix + elseif count == 1 then + return '1', '' + else + return '', suffix + end +end + +function ReaSpeechActionsUI:render() + local disable_if = self.disabler + local progress = self.worker:progress() + + disable_if(progress, function() + local selected_track_count = reaper.CountSelectedTracks(ReaUtil.ACTIVE_PROJECT) + disable_if(selected_track_count == 0, function() + local button_text = ("Process %s Selected Track%s") + :format(self.pluralizer(selected_track_count, 's')) + + if ImGui.Button(ctx, button_text) then + self:process_jobs(self.jobs_for_selected_tracks) + end + end) + + ImGui.SameLine(ctx) + + local selected_item_count = reaper.CountSelectedMediaItems(ReaUtil.ACTIVE_PROJECT) + disable_if(selected_item_count == 0, function() + local button_text = ("Process %s Selected Item%s") + :format(self.pluralizer(selected_item_count, 's')) + + if ImGui.Button(ctx, button_text) then + self:process_jobs(self.jobs_for_selected_items) + end + end) + + ImGui.SameLine(ctx) + if ImGui.Button(ctx, "Process All Items") then + self:process_jobs(self.jobs_for_all_items) + end + end) + + if progress then + ImGui.SameLine(ctx) + if ImGui.Button(ctx, "Cancel") then + self.worker:cancel() + end + + ImGui.SameLine(ctx) + ImGui.ProgressBar(ctx, progress) + end + ImGui.Dummy(ctx,0, 5) +end + +function ReaSpeechActionsUI.make_job(media_item, take) + local path = ReaSpeechUI.get_source_path(take) + + if path then + return {item = media_item, take = take, path = path} + else + return nil + end +end + +function ReaSpeechActionsUI.jobs_for_selected_tracks() + local jobs = {} + for track in ReaIter.each_selected_track() do + for item in ReaIter.each_track_item(track) do + for take in ReaIter.each_take(item) do + local job = ReaSpeechActionsUI.make_job(item, take) + if job then + table.insert(jobs, job) + end + end + end + end + return jobs +end + +function ReaSpeechActionsUI.jobs_for_selected_items() + local jobs = {} + for item in ReaIter.each_selected_media_item() do + for take in ReaIter.each_take(item) do + local job = ReaSpeechActionsUI.make_job(item, take) + if job then + table.insert(jobs, job) + end + end + end + return jobs +end + +function ReaSpeechActionsUI.jobs_for_all_items() + local jobs = {} + for item in ReaIter.each_media_item() do + for take in ReaIter.each_take(item) do + local job = ReaSpeechActionsUI.make_job(item, take) + if job then + table.insert(jobs, job) + end + end + end + return jobs +end + +function ReaSpeechActionsUI:process_jobs(job_generator) + local jobs = job_generator() + + if #jobs == 0 then + reaper.MB("No media found to process.", "No media", 0) + return + end + + app:new_jobs(jobs) +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechControlsUI.lua b/reascripts/ReaSpeech/source/ReaSpeechControlsUI.lua new file mode 100644 index 0000000000000000000000000000000000000000..e7d59f3a19619bc2c5b63842f6765069f2b3863e --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechControlsUI.lua @@ -0,0 +1,184 @@ +--[[ + +ReaSpeechControlsUI.lua - UI elements for configuring ASR services + +]]-- + +ReaSpeechControlsUI = Polo { + -- Copied from whisper.tokenizer.LANGUAGES + LANGUAGES = { + en = 'english', zh = 'chinese', de = 'german', + es = 'spanish', ru = 'russian', ko = 'korean', + fr = 'french', ja = 'japanese', pt = 'portuguese', + tr = 'turkish', pl = 'polish', ca = 'catalan', + nl = 'dutch', ar = 'arabic', sv = 'swedish', + it = 'italian', id = 'indonesian', hi = 'hindi', + fi = 'finnish', vi = 'vietnamese', he = 'hebrew', + uk = 'ukrainian', el = 'greek', ms = 'malay', + cs = 'czech', ro = 'romanian', da = 'danish', + hu = 'hungarian', ta = 'tamil', no = 'norwegian', + th = 'thai', ur = 'urdu', hr = 'croatian', + bg = 'bulgarian', lt = 'lithuanian', la = 'latin', + mi = 'maori', ml = 'malayalam', cy = 'welsh', + sk = 'slovak', te = 'telugu', fa = 'persian', + lv = 'latvian', bn = 'bengali', sr = 'serbian', + az = 'azerbaijani', sl = 'slovenian', kn = 'kannada', + et = 'estonian', mk = 'macedonian', br = 'breton', + eu = 'basque', is = 'icelandic', hy = 'armenian', + ne = 'nepali', mn = 'mongolian', bs = 'bosnian', + kk = 'kazakh', sq = 'albanian', sw = 'swahili', + gl = 'galician', mr = 'marathi', pa = 'punjabi', + si = 'sinhala', km = 'khmer', sn = 'shona', + yo = 'yoruba', so = 'somali', af = 'afrikaans', + oc = 'occitan', ka = 'georgian', be = 'belarusian', + tg = 'tajik', sd = 'sindhi', gu = 'gujarati', + am = 'amharic', yi = 'yiddish', lo = 'lao', + uz = 'uzbek', fo = 'faroese', ht = 'haitian creole', + ps = 'pashto', tk = 'turkmen', nn = 'nynorsk', + mt = 'maltese', sa = 'sanskrit', lb = 'luxembourgish', + my = 'myanmar', bo = 'tibetan', tl = 'tagalog', + mg = 'malagasy', as = 'assamese', tt = 'tatar', + haw = 'hawaiian', ln = 'lingala', ha = 'hausa', + ba = 'bashkir', jw = 'javanese', su = 'sundanese' + }, + LANGUAGE_CODES = {}, + DEFAULT_LANGUAGE = 'en', + + ITEM_WIDTH = 125, + LARGE_ITEM_WIDTH = 375, +} + +function ReaSpeechControlsUI:init() + self.log_enable = false + self.log_debug = false + + self.language = self.DEFAULT_LANGUAGE + self.translate = false + self.initial_prompt = '' + self.model_name = nil +end + +function ReaSpeechControlsUI:get_request_data() + return { + language = self.language, + translate = self.translate, + initial_prompt = self.initial_prompt, + model_name = self.model_name, + } +end + +ReaSpeechControlsUI._init_languages = function () + for code, _ in pairs(ReaSpeechControlsUI.LANGUAGES) do + table.insert(ReaSpeechControlsUI.LANGUAGE_CODES, code) + end + + table.sort(ReaSpeechControlsUI.LANGUAGE_CODES, function (a, b) + return ReaSpeechControlsUI.LANGUAGES[a] < ReaSpeechControlsUI.LANGUAGES[b] + end) + + table.insert(ReaSpeechControlsUI.LANGUAGE_CODES, 1, '') + ReaSpeechControlsUI.LANGUAGES[''] = 'detect' +end + +ReaSpeechControlsUI._init_languages() + +function ReaSpeechControlsUI:render() + --start input table so logo and inputs sit side-by-side + if ImGui.BeginTable(ctx, 'InputTable', 2) then + app:trap(function() + --column settings + ImGui.TableSetupColumn(ctx, 'Logo', ImGui.TableColumnFlags_WidthFixed()) + ImGui.TableSetupColumn(ctx, 'Inputs', ImGui.TableColumnFlags_WidthFixed()) + -- first column + ImGui.TableNextColumn(ctx) + ImGui.SameLine(ctx, -10) + app.png_from_bytes('reaspeech-logo-small') + -- second column + ImGui.TableNextColumn(ctx) + -- start language selection + self:render_language_controls() + ImGui.Dummy(ctx,0, 10) + self:render_advanced_controls() + end) + ImGui.EndTable(ctx) + end + -- end input table + ImGui.SameLine(ctx, ImGui.GetWindowWidth(ctx) - self.ITEM_WIDTH + 65) + app.png_from_bytes('heading-logo-tech-audio') +end + +function ReaSpeechControlsUI:render_language_controls() + if ImGui.TreeNode(ctx, 'Language Options', ImGui.TreeNodeFlags_DefaultOpen()) then + app:trap(function() + ImGui.Dummy(ctx, 0, 25) + ImGui.SameLine(ctx) + if ImGui.BeginCombo(ctx, "language", self.LANGUAGES[self.language]) then + app:trap(function() + local combo_items = self.LANGUAGE_CODES + for _, combo_item in pairs(combo_items) do + local is_selected = (combo_item == self.language) + if ImGui.Selectable(ctx, self.LANGUAGES[combo_item], is_selected) then + self.language = combo_item + end + end + end) + ImGui.EndCombo(ctx) + end + local rv, value + ImGui.SameLine(ctx) + rv, value = ImGui.Checkbox(ctx, "translate", self.translate) + if rv then + self.translate = value + end + end) + + ImGui.TreePop(ctx) + end +end + +function ReaSpeechControlsUI:render_advanced_controls() + local rv, value + + if ImGui.TreeNode(ctx, 'Advanced Options') then + app:trap(function() + ImGui.Dummy(ctx, 0, 25) + + ImGui.SameLine(ctx) + ImGui.PushItemWidth(ctx, self.LARGE_ITEM_WIDTH) + app:trap(function () + rv, value = ImGui.InputText(ctx, 'initial prompt', self.initial_prompt) + if rv then + self.initial_prompt = value + end + end) + ImGui.PopItemWidth(ctx) + + ImGui.SameLine(ctx) + ImGui.PushItemWidth(ctx, 100) + app:trap(function () + rv, value = ImGui.InputTextWithHint(ctx, 'model name', self.model_name or "") + if rv then + self.model_name = value + end + end) + ImGui.PopItemWidth(ctx) + + ImGui.SameLine(ctx) + rv, value = ImGui.Checkbox(ctx, "log", self.log_enable) + if rv then + self.log_enable = value + end + + if self.log_enable then + ImGui.SameLine(ctx) + rv, value = ImGui.Checkbox(ctx, "debug", self.log_debug) + if rv then + self.log_debug = value + end + end + end) + + ImGui.TreePop(ctx) + ImGui.Spacing(ctx) + end +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechEULAContent.lua b/reascripts/ReaSpeech/source/ReaSpeechEULAContent.lua new file mode 100644 index 0000000000000000000000000000000000000000..dd1d01724b18dd903da8e52f74e51dee6e06b015 --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechEULAContent.lua @@ -0,0 +1,233 @@ +ReaSpeechEULAContent = [=[teamaud.io LLC + +End User License Agreement + +Important – Use of this Software is subject to license restrictions. +Carefully read this license agreement before using the software. + +This End User License Agreement (this “Agreement”) is a legal +agreement between you, either individually or as an authorized +representative of the company or organization acquiring the license, and +teamaud.io LLC (“Team Audio”). THE DOWNLOAD AND USE OF THE SOFTWARE +INDICATES YOUR COMPLETE AND UNCONDITIONAL ACCEPTANCE OF THE TERMS AND +CONDITIONS SET FORTH IN THIS AGREEMENT. If you do not agree to these +terms and conditions, do not download or otherwise use the Software. + + + +1. License Grant + +a. The software programs you are installing, downloading, or have +acquired with this Agreement, including any audio plug-in, updates, +upgrades, modifications, revisions, copies, online materials and design +data (the “Software”) are copyrighted, proprietary information of +Team Audio and its affiliates, who maintain exclusive title to all +Software and retain all rights not expressly granted by this Agreement. +Team Audio grants to you, subject to your continued compliance with the +terms and conditions set forth in this Agreement, a non-exclusive, +non-transferable, revocable license to use a single copy of the Software +owned or distributed by Team Audio on up to three (3) computers owned or +controlled by you in machine readable, object-code form for your +personal home entertainment and commercial use on a compatible +electronic device, or for such use otherwise authorized by Team Audio. + +2. Restrictions and Intellectual Property Ownership + +a. You may not, without Team Audio’s prior written consent, (i) +decompile, disassemble or reverse engineer the Software or otherwise +attempt to gain access to its source code, except to the extent that +such restrictions are expressly prohibited by law; (ii) copy, offer for +public display, translate, adapt, modify or otherwise alter the +Software, or create derivative works thereof, including derivative works +with respect to artwork contained in the Software, except to the extent +that such restrictions are expressly prohibited by law; (iii) rent, +lease, loan, sublicense or distribute the Software, or offer it on a +pay-per-play, coin-op or other for charge (or free) basis; (iv) use the +Software to infringe the copyrights or other intellectual property +rights of others in any way; (v) remove, circumvent, disable, damage or +otherwise interfere with security-related or limiting features of the +Software, and/or (vi) modify or delete the copyright and other +proprietary rights notices on or in the Software. + +b. The Software, which is copyrighted, including any modifications, +upgrades, or updates thereto, is the sole and exclusive property of Team +Audio and is a valuable asset and trade secret of Team Audio. Team +Audio retains all ownership and intellectual property rights to the +Software and to any modifications, upgrades, or updates thereto. Except +for the rights granted herein, you shall have no right, title, or +interest of any kind in or to the Software. + +3. Additional Restrictions for Demo Version + +If the Software was provided to you for demo use for a limited period of +time and/or number of uses, you agree not to use the Software beyond the +expiration or termination of the demo period. You acknowledge and agree +that the Software may include code designed to prevent you from +exceeding these limits, and that such code may remain on your computer +or device after deletion of the Software to prevent you from installing +another copy of the Software and repeating the demo. + +4. Term and Termination. + +a. This Agreement remains effective until expiration or termination. +This Agreement will immediately terminate upon notice if you exceed the +scope of the license granted or otherwise fail to comply with the +provisions in sections 1, 2, and 3 above. For any other material breach +of the Agreement, Team Audio may terminate this Agreement if you are in +breach and fail to cure the breach within thirty (30) days of written +notification. If Software is provided for a limited term use, this +Agreement will automatically expire at the end of the authorized term. + +b. Upon termination of this Agreement for any reason, you agree to +delete from any permanent machine storage (i.e., hard disk) previously +loaded copies of the Software in all forms. Upon request of Team Audio, +you shall certify in writing that all copies of the Software and +associated documentation have been destroyed or returned to Team Audio. +The indemnity and limitation of liability obligations hereunder, as well +as your obligations with respect to confidential treatment of the +Software and Team Audio’s trade secrets, other intellectual property, +and proprietary information, shall survive the termination of this +Agreement. + +5. Disclaimer of Warranties. + +TEAM AUDIO DISCLAIMS ALL WARRANTIES, WHETHER EXPRESS OR IMPLIED, +INCLUDING THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A +PARTICULAR PURPOSE. TEAM AUDIO MAKES NO WARRANTY THAT ANY SOFTWARE WILL +PERFORM ERROR-FREE OR UNINTERRUPTED, OR THAT ALL ERRORS THEREIN CAN OR +WILL BE CORRECTED. TEAM AUDIO FURTHER DISCLAIMS ANY IMPLIED WARRANTIES +OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, TITLE AND +NON-INFRINGEMENT AND ANY WARRANTIES ARISING FROM COURSE OF PERFORMANCE, +COURSE OF DEALING OR USAGE OF TRADE. IN ADDITION, YOU ACKNOWLEDGE THAT +TEAM AUDIO IS NOT RESPONSIBLE FOR THE INTERNET OR WHETHER IT SHOULD +CONTINUE TO EXIST IN ITS PRESENT FORM OR WHETHER OR NOT A GOVERNMENT OR +GOVERNMENTAL AGENCY, EITHER FOREIGN OR DOMESTIC, WILL CONTROL, REGULATE +OR DISBAND THE INTERNET. USE OF THE INTERNET IS AT YOUR SOLE RISK. + +6. Indemnity; Responsibility + +a. In the event that some or all of the Software is held or is believed +by Team Audio to infringe upon patent or copyrights of third parties, +Team Audio shall have the option, at its expense: (i) to modify the +Software to be non-infringing; or (ii) to obtain for you a license to +continue using the Software. If it is not commercially feasible to +perform either of the above options, then Team Audio may require from +you return of the infringing Software and all rights thereto. Upon +return of the infringing Software to Team Audio, you may terminate the +Agreement with ten (10) days’ written notice. This subsection sets +forth Team Audio’s entire liability and exclusive remedy for +infringement. + +b. You will defend and indemnify Team Audio and its affiliates against +any claim incurred by, borne by or asserted against Team Audio or its +affiliates that relates to or results from (i) your use of the Software, +(ii) any intentional or willful conduct or negligence by you or (iii) +any breach of an applicable representation, covenant or warranty +contained herein. + +c. In no event shall Team Audio be held responsible for any damage in +the event your passwords are disclosed, including but limited to any +damage that occurs to your account, your characters or your scores. +Further, Team Audio shall not be liable for any financial or emotional +damage or distress you may suffer and/or for subsequent loss or damage +to your Account, characters, scores as a result of the disclosure of +your passwords to a third party. You shall not disclose your passwords +to any third party or allow anyone to use your password to access your +account, or to use the Software. It is also prohibited to obtain, +attempt to obtain, use, or attempt to use the password of anyone else. + +d. You are solely responsible for any and all telecommunications or +other connectivity charges incurred through your use of the application. + +7. Confidentiality + +You agree that you will not, directly or indirectly, copy the structure, +sequence, or organization of the Software, nor will you copy any portion +of the Software or related documentation to produce software programs +that are substantially similar to the Software. + +8. LIMITATION OF LIABILITY + +EXCEPT WHERE THIS EXCLUSION OR RESTRICTION OF LIABILITY WOULD BE VOID OR +INEFFECTIVE UNDER APPLICABLE LAW, IN NO EVENT WILL TEAM AUDIO BE LIABLE +FOR ANY INDIRECT, INCIDENTAL, SPECIAL, OR CONSEQUENTIAL DAMAGES, OR +DAMAGES FOR LOSS OF PROFITS, REVENUE, DATA, OR USE, INCURRED BY YOU OR +ANY THIRD PARTY, WHETHER IN AN ACTION IN CONTRACT OR TORT, EVEN IF TEAM +AUDIO OR ANY OTHER PERSON HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH +DAMAGES. TEAM AUDIO’S LIABILITY FOR DAMAGES HEREUNDER SHALL IN NO +EVENT EXCEED THE AMOUNT OF FEES PAID BY YOU FOR THE SOFTWARE OR SERVICE +GIVING RISE TO THE CLAIM. IN THE CASE WHERE NO AMOUNT WAS PAID, TEAM +AUDIO SHALL HAVE NO LIABILITY FOR ANY DAMAGES WHATSOEVER. + +9. Assignment; Jurisdiction. + +This Agreement will be binding upon, and will inure to the benefit of, +the permitted successors and assigns of each party hereto. You may not +assign, delegate, transfer, or otherwise convey this Agreement, or any +of its rights hereunder, to any entity without the prior written consent +of Team Audio, which consent Team Audio may give or withhold in its sole +discretion, and any attempted assignment or delegation without such +consent shall be void. This Agreement, and all matters arising out of +or relating to this Agreement, shall be governed by the laws of the +State of Texas, United States of America. Any legal action or +proceeding relating to this Agreement shall be instituted in any state +or federal court in Travis County, Texas, United States of America. +Team Audio and you agree to submit to the jurisdiction of, and agree +that venue is proper in, the aforesaid courts in any such legal action +or proceeding. + +10. Severability; Waiver + +In the event any provision of this Agreement is held to be invalid or +unenforceable, the remaining provisions of this Agreement will remain in +full force. The waiver by either party of any default or breach of this +Agreement shall not constitute a waiver of any other or subsequent +default or breach. Except for actions for nonpayment or breach of +either party’s intellectual property rights, no action, regardless of +form, arising out of this Agreement may be brought by either party more +than two years after the cause of action has accrued. The headings +appearing in this Agreement are inserted for convenience only, and will +not be used to define, limit or enlarge the scope of this Agreement or +any of the obligations herein. + +11. Interpretation; Compliance. + +This Agreement constitutes the complete agreement between the Parties +and supersedes all previous and contemporaneous agreements, proposals, +or representations, written or oral, concerning the subject matter of +this Agreement. This Agreement may not be modified or amended except in +a writing signed by you and Team Audio; no other act, document, usage, +or custom shall be deemed to amend or modify this Agreement. It is +expressly agreed that the terms and conditions of this Agreement +supersede the terms of any purchase order. Each party agrees to comply +with all relevant export laws and regulations of the United States and +the country or territory in which the Services are provided (“Export +Laws”) to assure that neither any deliverable, if any, nor any direct +product thereof is (a) exported, directly or indirectly, in violation of +Export Laws or (b) intended to be used for any purposes prohibited by +the Export Laws, including without limitation nuclear, chemical, or +biological weapons proliferation. Each party agrees to comply with all +federal, state and local laws and regulations applicable to this +Agreement. Each party represents and warrants that it is qualified to +do business in the geographies in which it will perform its obligations +under this Agreement, and will obtain all necessary licenses and +permits, and satisfy any other legal, regulatory and administrative +requirements, necessary to its performance hereunder. + +13. Notices; How to Contact Team Audio + +The Software is made available to you by Team Audio, LLC located at 611 +Cloud Ct., Round Rock, TX 78681 (or at an updated address posted online +at HYPERLINK "https://teamaud.io/" https://teamaud.io/ ). If you +have any questions about the Software, you may contact Team Audio at +support@techaud.io . + +YOU ACKNOWLEDGE THAT YOU HAVE READ THIS AGREEMENT, YOU UNDERSTAND THIS +AGREEMENT, AND YOU UNDERSTAND THAT, BY CONTINUING THE DOWNLOAD OR +INSTALLATION OF THE SOFTWARE, BY LOADING OR RUNNING THE SOFTWARE, OR BY +PLACING OR COPYING THE SOFTWARE ONTO YOUR COMPUTER OR MOBILE DEVICE HARD +DRIVE, COMPUTER RAM, OR OTHER STORAGE, YOU AGREE TO BE BOUND BY THE +TERMS AND CONDITIONS OF THIS AGREEMENT. + +Last updated: June 2022 +]=] diff --git a/reascripts/ReaSpeech/source/ReaSpeechMain.lua b/reascripts/ReaSpeech/source/ReaSpeechMain.lua new file mode 100644 index 0000000000000000000000000000000000000000..5b0c10f0fb16eda3f96f2114755a4c6bb1533d34 --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechMain.lua @@ -0,0 +1,70 @@ +--[[ + + ReaSpeechMain.lua - ReaSpeech main class + +]]-- + +ctx = nil +app = nil + +ReaSpeechMain = {} + +function ReaSpeechMain:main() + if not self:check_imgui() then return end + reaper.atexit(function () self:onexit() end) + + ctx = ImGui.CreateContext(ReaSpeechUI.TITLE, ReaSpeechUI.config_flags()) + Fonts:load() + app = ReaSpeechUI.new() + reaper.defer(self:loop()) +end + +function ReaSpeechMain:loop() + local visible, open = false, false + + return function() + ImGui.PushFont(ctx, Fonts.main) + app:trap(function() + Theme():push(ctx) + app:trap(function() + if ReaSpeechUI.METRICS then + ImGui.ShowMetricsWindow(ctx) + end + + ImGui.SetNextWindowSize(ctx, app.WIDTH, app.HEIGHT, ImGui.Cond_FirstUseEver()) + visible, open = ImGui.Begin(ctx, ReaSpeechUI.TITLE, true) + + if visible then + app:trap(function() + app:react() + end) + ImGui.End(ctx) + end + end) + Theme():pop(ctx) + end) + ImGui.PopFont(ctx) + + if open then + reaper.defer(self:loop()) + end + end +end + +function ReaSpeechMain:check_imgui() + if ImGui.CreateContext then + return true + else + reaper.MB( + "This script requires the ReaImGui API, which can be installed from:\n\n" + .. "Extensions > ReaPack > Browse packages...", + "ReaImGui required", + 0 + ) + return false + end +end + +function ReaSpeechMain:onexit() + Tempfile:remove_all() +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechProductActivation.lua b/reascripts/ReaSpeech/source/ReaSpeechProductActivation.lua new file mode 100644 index 0000000000000000000000000000000000000000..da675e97cef1c4cc6df82032aa526a956472931e --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechProductActivation.lua @@ -0,0 +1,151 @@ +--[[ + + ReaSpeechProductActivation.lua - Product key entry and activation checks + +]]-- + +ReaSpeechProductActivation = Polo { + ACTIVATION_URL = "https://techaud.io/ProductActivationDeveloper.php", + + PRODUCT_ID = 79004057, + PRODUCT_CHECK_COUNT = 4, + PRODUCT_NAME = "ReaSpeech", + + config = nil, + + -- nil = not activated + -- 'pending' = activation in process + -- 'activated' = activation complete + state = nil, + activation_message ="" +} + +function ReaSpeechProductActivation:init() + self:init_config() + self:activation_state_check() +end + +function ReaSpeechProductActivation:init_config() + self.config = OptionsConfig:new { + section = 'ReaSpeech', + options = { + product_run_check_count = {'number', 0}, + product_license = {'string', ''}, + product_license_value = {'string', ''}, + eula_signed = {'boolean', false}, + } + } +end + +function ReaSpeechProductActivation:is_activated() + return self.state == 'activated' and self.config:get('eula_signed') +end + +function ReaSpeechProductActivation:activation_state_check() + local has_l = self.config:exists('product_license') + local has_lv = self.config:exists('product_license_value') + + if has_l and has_lv then + local count = self.config:get('product_run_check_count') + + if count > self.PRODUCT_CHECK_COUNT then + self.state = 'pending' + self:handle_product_activation_recheck() + else + self.state = 'activated' + self.config:set('product_run_check_count', count + 1) + end + else + self.state = nil + end +end + +function ReaSpeechProductActivation:handle_product_activation(product_key) + product_key = string.gsub(product_key, "%s+", "") + if #product_key == 0 then + self.state = nil + return + end + + local process_result = self:send_activation_request(product_key, false) + + if process_result then + self:process_activation_reply(product_key, process_result) + end + +end + +function ReaSpeechProductActivation:handle_product_activation_recheck() + local process_result = self:send_activation_request(self.config:get('product_license'), true) + + if process_result then + if string.find(process_result, "SUCCESS") then + self.state = 'activated' + self.config:set('product_run_check_count', 0) + elseif string.find(process_result, "FAILURE") then + self.state = nil + self.config:delete('product_license') + self.config:delete('product_license_value') + else + -- Connection failed, silently ignore + self.state = 'activated' + end + else + -- Command failed, silently ignore + self.state = 'activated' + end +end + +function ReaSpeechProductActivation:send_activation_request(product_key, is_recheck) + local curl = "curl" + if not reaper.GetOS():find("Win") then + curl = "/usr/bin/curl" + end + + local cmd_data_id = "user_product_id=" .. self.PRODUCT_ID + local cmd_data_license = "user_license=" .. product_key + local cmd_data_p_n = "user_product_name=" .. self.PRODUCT_NAME + local cmd_data_p_v = "user_product_version=" .. ReaSpeechUI.VERSION + local cmd_data_recheck = "recheck=" .. tostring(is_recheck) + + local cmd_args = ( + curl.." -X POST" + .. " -d " .. cmd_data_id + .. " -d " .. cmd_data_license + .. " -d " .. cmd_data_p_n + .. " -d " .. cmd_data_p_v + .. " -d " .. cmd_data_recheck + .. " \"" .. self.ACTIVATION_URL .. "\"" + ) + + local process_result = reaper.ExecProcess(cmd_args, 8000) + if process_result then + return process_result + else + self.activation_message = "Activation failed: Connection Error" + reaper.ShowConsoleMsg("Failed CURL at activation request" .. '\n') + return nil + end + +end + +function ReaSpeechProductActivation:process_activation_reply(product_key, process_result) + process_result = string.gsub(process_result, "%s+", "") + + if string.find(process_result, "SUCCESS") then + self.config:set('product_run_check_count', 1) + self.config:set('product_license', product_key) + self.config:set('product_license_value', process_result) + + self.state = 'activated' + self.activation_message = "Thanks for your support! Enjoy :)" + elseif string.find(process_result, "FAILURE") then + self.state = nil + if string.find(process_result, "Invalid_License") then + self.activation_message = "Activation failed: Sorry, we didn't find a valid license :(" + else + self.activation_message = "Activation failed: Sorry, you are out of activations :(" + end + + end +end \ No newline at end of file diff --git a/reascripts/ReaSpeech/source/ReaSpeechProductActivationUI.lua b/reascripts/ReaSpeech/source/ReaSpeechProductActivationUI.lua new file mode 100644 index 0000000000000000000000000000000000000000..5cf51bbe3f2bf754f3aa8ff9b52ba04a67f88e4d --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechProductActivationUI.lua @@ -0,0 +1,69 @@ +--[[ + + ReaSpeechProductActivationUI.lua - ReaSpeech Product Activation UI + +]]-- + +ReaSpeechProductActivationUI = Polo { + LARGE_ITEM_WIDTH = 375, +} + +function ReaSpeechProductActivationUI:init() + assert(self.product_activation, 'missing product activation') + self.license_input = '' +end + +function ReaSpeechProductActivationUI:render() + if self.product_activation.state ~= "activated" then + self:render_activation_inputs() + return + end + + if not self.product_activation.config:get('eula_signed') then + self:render_EULA_inputs() + return + end +end + +function ReaSpeechProductActivationUI:render_activation_inputs() + ImGui.Text(ctx, ('Welcome to ReaSpeech by Tech Audio')) + ImGui.Dummy(ctx, self.LARGE_ITEM_WIDTH, 25) + ImGui.Text(ctx, ('Please enter your license key to get started')) + ImGui.Dummy(ctx, self.LARGE_ITEM_WIDTH, 5) + ImGui.PushItemWidth(ctx, self.LARGE_ITEM_WIDTH) + app:trap(function () + local rv, value = ImGui.InputText(ctx, '##', self.license_input) + if rv then + self.license_input = value + end + if self.product_activation.activation_message ~= "" then + --Possibly make this ColorText with and change depending on message + ImGui.SameLine(ctx) + ImGui.Text(ctx, self.product_activation.activation_message) + end + end) + ImGui.PopItemWidth(ctx) + ImGui.Dummy(ctx, self.LARGE_ITEM_WIDTH, 30) + if ImGui.Button(ctx, "Submit") then + self:handle_product_activation(self.license_input) + end +end + +function ReaSpeechProductActivationUI:render_EULA_inputs() + ImGui.PushItemWidth(ctx, self.LARGE_ITEM_WIDTH) + app:trap(function () + ImGui.Text(ctx, 'EULA') + ImGui.Dummy(ctx, self.LARGE_ITEM_WIDTH, 25) + ImGui.TextWrapped(ctx, ReaSpeechEULAContent) + ImGui.Dummy(ctx, self.LARGE_ITEM_WIDTH, 25) + if ImGui.Button(ctx, "Agree") then + self.product_activation.config:set('eula_signed', true) + end + end) + ImGui.PopItemWidth(ctx) +end + +function ReaSpeechProductActivationUI:handle_product_activation(input_license) + --reaper.ShowConsoleMsg(tostring(input_license) .. '\n') + self.product_activation:handle_product_activation(input_license) +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechUI.lua b/reascripts/ReaSpeech/source/ReaSpeechUI.lua new file mode 100644 index 0000000000000000000000000000000000000000..7016fce6c4923141470ed9571dcd18388aa4eb37 --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechUI.lua @@ -0,0 +1,225 @@ +--[[ + + ReaSpeechUI.lua - ReaSpeech user interface + +]]-- + +ReaSpeechUI = Polo { + VERSION = "unknown (development)", + -- Set to show ImGui Metrics/Debugger window + METRICS = false, + + TITLE = 'ReaSpeech', + WIDTH = 1000, + HEIGHT = 600, + + ITEM_WIDTH = 125, +} + +function ReaSpeechUI:init() + self.onerror = function (e) + self:log(e) + end + + self.requests = {} + self.responses = {} + self.logs = {} + + ReaSpeechAPI:init('http://' .. Script.host) + + self.worker = ReaSpeechWorker.new({ + requests = self.requests, + responses = self.responses, + logs = self.logs, + }) + + self.product_activation = ReaSpeechProductActivation.new() + self.product_activation_ui = ReaSpeechProductActivationUI.new { + product_activation = self.product_activation + } + + self.controls_ui = ReaSpeechControlsUI.new() + + self.actions_ui = ReaSpeechActionsUI.new({ + worker = self.worker + }) + + self.transcript = Transcript.new() + self.transcript_ui = TranscriptUI.new { transcript = self.transcript } + + self.failure = AlertPopup.new { title = 'Transcription Failed' } + + self.react_handlers = self:get_react_handlers() +end + +ReaSpeechUI.config_flags = function () + return ImGui.ConfigFlags_DockingEnable() +end + +ReaSpeechUI.log_time = function () + return os.date('%Y-%m-%d %H:%M:%S') +end + +function ReaSpeechUI:log(msg) + table.insert(self.logs, {msg, false}) +end + +function ReaSpeechUI:debug(msg) + table.insert(self.logs, {msg, true}) +end + +function ReaSpeechUI:trap(f) + return xpcall(f, self.onerror) +end + +function ReaSpeechUI:has_js_ReaScriptAPI() + if reaper.JS_Dialog_BrowseForSaveFile then + return true + end + return false +end + +function ReaSpeechUI:show_file_dialog(options) + local title = options.title or 'Save file' + local folder = options.folder or '' + local file = options.file or '' + local ext = options.ext or '' + local save = options.save or false + local multi = options.multi or false + if self:has_js_ReaScriptAPI() then + if save then + return reaper.JS_Dialog_BrowseForSaveFile(title, folder, file, ext) + else + return reaper.JS_Dialog_BrowseForOpenFiles(title, folder, file, ext, multi) + end + else + return nil + end +end + +function ReaSpeechUI:tooltip(text) + if not ImGui.IsItemHovered(ctx, ImGui.HoveredFlags_DelayNormal()) or + not ImGui.BeginTooltip(ctx) + then return end + + self:trap(function() + ImGui.PushTextWrapPos(ctx, ImGui.GetFontSize(ctx) * 42) + self:trap(function() + ImGui.Text(ctx, text) + end) + ImGui.PopTextWrapPos(ctx) + end) + + ImGui.EndTooltip(ctx) +end + +function ReaSpeechUI:react() + for _, handler in pairs(self.react_handlers) do + self:trap(handler) + end +end + +function ReaSpeechUI:get_react_handlers() + return { + function() self:react_to_worker_response() end, + function() self:react_to_logging() end, + function() self.worker:react() end, + function() self:render() end + } +end + +function ReaSpeechUI:react_to_worker_response() + local response = table.remove(self.responses, 1) + + if not response then + return + end + + self:debug('Response: ' .. dump(response)) + + if response.error then + self.failure:show(response.error) + self.worker:cancel() + return + end + + if not response.segments then + return + end + + for _, segment in pairs(response.segments) do + for _, s in pairs( + TranscriptSegment.from_whisper(segment, response._job.item, response._job.take) + ) do + if s:get('text') then + self.transcript:add_segment(s) + end + end + end + + self.transcript:update() +end + +function ReaSpeechUI:react_to_logging() + for _, log in pairs(self.logs) do + local msg, dbg = table.unpack(log) + if dbg and self.controls_ui.log_enable and self.controls_ui.log_debug then + reaper.ShowConsoleMsg(self:log_time() .. ' [DBG] ' .. tostring(msg) .. '\n') + elseif not dbg and self.controls_ui.log_enable then + reaper.ShowConsoleMsg(self:log_time() .. ' [LOG] ' .. tostring(msg) .. '\n') + end + end + + self.logs = {} +end + +function ReaSpeechUI:render() + ImGui.PushItemWidth(ctx, self.ITEM_WIDTH) + + self:trap(function () + if not self.product_activation:is_activated() then + self.product_activation_ui:render() + return + end + + self.controls_ui:render() + self.actions_ui:render() + self.transcript_ui:render() + self.failure:render() + end) + + ImGui.PopItemWidth(ctx) +end + +function ReaSpeechUI:new_jobs(jobs) + local request = self.controls_ui:get_request_data() + request.jobs = jobs + self:debug('Request: ' .. dump(request)) + + self.transcript:clear() + + table.insert(self.requests, request) +end + +function ReaSpeechUI.png_from_bytes(image_key) + if not IMAGES[image_key] or not IMAGES[image_key].bytes then + return + end + + local image = IMAGES[image_key] + + if not ImGui.ValidatePtr(image.imgui_image, 'ImGui_Image*') then + image.imgui_image = ImGui.CreateImageFromMem(image.bytes) + end + + ImGui.Image(ctx, image.imgui_image, image.width, image.height) +end + +function ReaSpeechUI.get_source_path(take) + local source = reaper.GetMediaItemTake_Source(take) + if source then + local source_path = reaper.GetMediaSourceFileName(source) + return source_path + end + return nil +end diff --git a/reascripts/ReaSpeech/source/ReaSpeechWorker.lua b/reascripts/ReaSpeech/source/ReaSpeechWorker.lua new file mode 100644 index 0000000000000000000000000000000000000000..b404de0f904c4e71cc27019800ecb96b30a3b92d --- /dev/null +++ b/reascripts/ReaSpeech/source/ReaSpeechWorker.lua @@ -0,0 +1,322 @@ +--[[ + + ReaSpeechWorker.lua - Speech transcription worker + +]]-- + +ReaSpeechWorker = Polo {} + +function ReaSpeechWorker:init() + assert(self.requests, 'missing requests') + assert(self.responses, 'missing responses') + assert(self.logs, 'missing logs') + + self.active_job = nil + self.pending_jobs = {} + self.job_count = 0 +end + +function ReaSpeechWorker:react() + local time = reaper.time_precise() + local fs = self:interval_functions() + for i = 1, #fs do + app:trap(function () + fs[i]:react(time) + end) + end +end + +function ReaSpeechWorker:interval_functions() + if self._interval_functions then + return self._interval_functions + end + + self._interval_functions = { + IntervalFunction.new(0.3, function () self:react_handle_request() end), + IntervalFunction.new(0.5, function () self:react_handle_jobs() end), + } + + return self._interval_functions +end + +-- Handle next request +function ReaSpeechWorker:react_handle_request() + local request = table.remove(self.requests, 1) + if request then + self:handle_request(request) + end +end + +-- Make progress on jobs +function ReaSpeechWorker:react_handle_jobs() + if self.active_job then + self:check_active_job() + return + end + + local pending_job = table.remove(self.pending_jobs, 1) + if pending_job then + self.active_job = pending_job + self:start_active_job() + elseif self.job_count ~= 0 then + app:log('Processing finished') + self.job_count = 0 + end +end + +function ReaSpeechWorker:progress() + local job_count = self.job_count + if job_count == 0 then + return nil + end + + local pending_job_count = #self.pending_jobs + + local active_job_progress = 0 + + -- the active job adds 1 to the total count, and if we can know the progress + -- then we can use that fraction + if self.active_job then + if self.active_job.job and self.active_job.job.progress then + local progress = self.active_job.job.progress + active_job_progress = (progress.current / progress.total) + end + + pending_job_count = pending_job_count + 1 + end + + local completed_job_count = job_count + active_job_progress - pending_job_count + return completed_job_count / job_count +end + +function ReaSpeechWorker:cancel() + if self.active_job then + if self.active_job.job and self.active_job.job.job_id then + self:cancel_job(self.active_job.job.job_id) + end + self.active_job = nil + end + self.pending_jobs = {} + self.job_count = 0 +end + +function ReaSpeechWorker:cancel_job(job_id) + local url_path = "jobs/" .. job_id + ReaSpeechAPI:fetch_json(url_path, 'DELETE', function(error_message) + self:handle_error(self.active_job, error_message) + end) +end + +function ReaSpeechWorker:get_job_status(job_id) + local url_path = "jobs/" .. job_id + return ReaSpeechAPI:fetch_json(url_path, 'GET', function(error_message) + self:handle_error(self.active_job, error_message) + self.active_job = nil + end) +end + +function ReaSpeechWorker:handle_request(request) + app:log('Processing speech...') + self.job_count = #request.jobs + + local data = { + task = request.translate and 'translate' or 'transcribe', + output = 'json', + use_async = 'true', + vad_filter = 'true', + word_timestamps = 'true', + model_name = request.model_name, + } + + if request.language and request.language ~= '' then + data.language = request.language + end + + if request.initial_prompt and request.initial_prompt ~= '' then + data.initial_prompt = request.initial_prompt + end + + local seen_path = {} + for _, job in pairs(request.jobs) do + if not seen_path[job.path] then + seen_path[job.path] = true + table.insert(self.pending_jobs, {job = job, data = data}) + end + end +end + +-- May return true if the job has completed and should no longer be active +function ReaSpeechWorker:handle_job_status(active_job, response) + app:debug('Active job: ' .. dump(active_job)) + app:debug('Status: ' .. dump(response)) + + if response.error then + table.insert(self.responses, { error = response.error }) + return true + end + + active_job.job.job_id = response.job_id + + if not response.job_status then + return false + end + + if response.job_status == 'SUCCESS' then + local transcript_url_path = response.job_result.url_path + response._job = active_job.job + active_job.transcript_output_file, active_job.transcript_output_sentinel_file = ReaSpeechAPI:fetch_large(transcript_url_path) + -- Job completion depends on non-blocking download of transcript + return false + elseif response.job_status == 'FAILURE' then + self:handle_error(active_job, response.job_result.error) + return true + end + + if response.job_result and response.job_result.progress then + active_job.job.progress = response.job_result.progress + end + + return false +end + +function ReaSpeechWorker:handle_response(active_job, response) + response._job = active_job.job + table.insert(self.responses, response) +end + +function ReaSpeechWorker:handle_error(_active_job, error_message) + table.insert(self.responses, { error = error_message }) +end + +function ReaSpeechWorker:start_active_job() + if not self.active_job then + return + end + + local active_job = self.active_job + local output_file, sentinel_file = ReaSpeechAPI:post_request('/asr', active_job.data, active_job.job.path) + + if output_file then + active_job.request_output_file = output_file + active_job.request_output_sentinel_file = sentinel_file + else + self.active_job = nil + end +end + +function ReaSpeechWorker:check_active_job() + if not self.active_job then return end + + local active_job = self.active_job + + if active_job.request_output_file then + self:check_active_job_request_output_file() + end + + if active_job.transcript_output_file then + self:check_active_job_transcript_output_file() + else + self:check_active_job_status() + end +end + +function ReaSpeechWorker:check_active_job_status() + if not self.active_job then return end + + local active_job = self.active_job + if not active_job.job.job_id then return end + + local response = self:get_job_status(active_job.job.job_id) + if response then + if self:handle_job_status(active_job, response) then + self.active_job = nil + end + end +end + +ReaSpeechWorker.check_sentinel = function(filename) + local sentinel = io.open(filename, 'r') + + if not sentinel then + return false + end + + sentinel:close() + return true +end + +function ReaSpeechWorker:handle_response_json(output_file, sentinel_file, success_f, fail_f) + if not self.check_sentinel(sentinel_file) then + return + end + + local f = io.open(output_file, 'r') + if not f then + fail_f("Couldn't open output_filename: " .. tostring(output_file)) + return + end + + local http_status, body = ReaSpeechAPI.http_status_and_body(f) + f:close() + + if #body < 1 then + fail_f("Empty response from server.") + return + end + + if http_status ~= 200 then + Tempfile:remove(sentinel_file) + Tempfile:remove(output_file) + local msg = "Server responded with status " .. http_status + fail_f(msg) + app:log(msg) + app:debug(body) + return + end + + local response = nil + if app:trap(function () + response = json.decode(body) + end) then + Tempfile:remove(sentinel_file) + Tempfile:remove(output_file) + success_f(response) + else + app:debug("JSON parse error, trying again later") + end +end + +function ReaSpeechWorker:check_active_job_request_output_file() + local active_job = self.active_job + + self:handle_response_json( + active_job.request_output_file, + active_job.request_output_sentinel_file, + function(response) + if self:handle_job_status(active_job, response) then + self.active_job = nil + end + end, + function(error_message) + self:handle_error(active_job, error_message) + self.active_job = nil + end + ) +end + +function ReaSpeechWorker:check_active_job_transcript_output_file() + local active_job = self.active_job + + self:handle_response_json( + active_job.transcript_output_file, + active_job.transcript_output_sentinel_file, + function(response) + self:handle_response(active_job, response) + self.active_job = nil + end, + function(error_message) + self:handle_error(active_job, error_message) + self.active_job = nil + end + ) +end diff --git a/reascripts/ReaSpeech/source/SRTWriter.lua b/reascripts/ReaSpeech/source/SRTWriter.lua new file mode 100644 index 0000000000000000000000000000000000000000..26e4f2f2bb1ed9a1969371bf2bf0c7aecfc1e52b --- /dev/null +++ b/reascripts/ReaSpeech/source/SRTWriter.lua @@ -0,0 +1,93 @@ +--[[ + + SRTWriter.lua - SRT file writer + +]]-- + +SRTWriter = Polo { + TIME_FORMAT = '%02d:%02d:%02d,%03d', +} + +function SRTWriter:init() + assert(self.file, 'missing file') + + self.options = self.options or {} + + if self.options.coords_x1 then + self.coords_x1 = self.options.coords_x1 + end + if self.options.coords_y1 then + self.coords_y1 = self.options.coords_y1 + end + if self.options.coords_x2 then + self.coords_x2 = self.options.coords_x2 + end + if self.options.coords_y2 then + self.coords_y2 = self.options.coords_y2 + end +end + +SRTWriter.format_time = function (time) + local milliseconds = math.floor(time * 1000) % 1000 + local seconds = math.floor(time) % 60 + local minutes = math.floor(time / 60) % 60 + local hours = math.floor(time / 3600) + return string.format(SRTWriter.TIME_FORMAT, hours, minutes, seconds, milliseconds) +end + +function SRTWriter:write(transcript) + local sequence_number = 1 + for _, segment in pairs(transcript:get_segments()) do + self:write_segment(segment, sequence_number) + sequence_number = sequence_number + 1 + end +end + +function SRTWriter:write_segment(segment, sequence_number) + local start = segment:get('start') + local end_ = segment:get('end') + local text = segment:get('text') + self:write_line(text, sequence_number, start, end_) +end + +function SRTWriter:write_line(line, sequence_number, start, end_) + local sequence_number_str = tostring(sequence_number) + local start_str = SRTWriter.format_time(start) + local end_str = SRTWriter.format_time(end_) + self.file:write(sequence_number_str) + self.file:write('\n') + self.file:write(start_str) + self.file:write(' --> ') + self.file:write(end_str) + self.file:write(self:coords()) + self.file:write('\n') + self.file:write(line) + self.file:write('\n') + self.file:write('\n') +end + +function SRTWriter:coords() + local coords = {} + + if self.coords_x1 then + table.insert(coords, 'X1:' .. self.coords_x1) + end + + if self.coords_x2 then + table.insert(coords, 'X2:' .. self.coords_x2) + end + + if self.coords_y1 then + table.insert(coords, 'Y1:' .. self.coords_y1) + end + + if self.coords_y2 then + table.insert(coords, 'Y2:' .. self.coords_y2) + end + + if #coords == 0 then + return '' + end + + return ' ' .. table.concat(coords, ' ') +end diff --git a/reascripts/ReaSpeech/source/Theme.lua b/reascripts/ReaSpeech/source/Theme.lua new file mode 100644 index 0000000000000000000000000000000000000000..c4de1f0b07f068bc553ae7cc29e4efc38afe635b --- /dev/null +++ b/reascripts/ReaSpeech/source/Theme.lua @@ -0,0 +1,50 @@ +Theme = { + theme = nil, + colors = { + dark_gray_semi_transparent = 0x404040FB, + black_near_transparent = 0x000000E8, + medium_gray_opaque = 0x5C5C5CFF, + dark_gray_translucent = 0x2B2B2B8A, + dark_gray_opaque = 0x404040FF, + dark_blue_gray_opaque = 0x4A5459FF, + pink_opaque = 0xE24097FF, + dark_gray_semi_opaque = 0x404040FB, + } +} +setmetatable(Theme, { __call = function () return Theme.init() end }) + +function Theme.init() + if Theme.theme ~= nil then + return Theme.theme + end + + Theme.theme = ImGuiTheme.new({ + colors = { + { ImGui.Col_WindowBg(), Theme.colors.dark_gray_semi_transparent }, + { ImGui.Col_Border(), Theme.colors.black_near_transparent }, + { ImGui.Col_Button(), Theme.colors.medium_gray_opaque }, + { ImGui.Col_ButtonHovered(), Theme.colors.dark_gray_translucent }, + { ImGui.Col_ButtonActive(), Theme.colors.dark_gray_opaque }, + { ImGui.Col_TitleBg(), Theme.colors.dark_gray_semi_transparent }, + { ImGui.Col_TitleBgActive(), Theme.colors.dark_blue_gray_opaque }, + { ImGui.Col_FrameBg(), Theme.colors.dark_gray_translucent }, + { ImGui.Col_FrameBgHovered(), Theme.colors.dark_gray_translucent }, + { ImGui.Col_FrameBgActive(), Theme.colors.pink_opaque }, + { ImGui.Col_CheckMark(), Theme.colors.pink_opaque }, + { ImGui.Col_HeaderHovered(), Theme.colors.dark_gray_semi_opaque }, + { ImGui.Col_HeaderActive(), Theme.colors.dark_gray_semi_transparent }, + { ImGui.Col_Header(), Theme.colors.dark_gray_semi_opaque } + }, + + styles = { + { ImGui.StyleVar_FramePadding(), 10.0, 6.0 }, + { ImGui.StyleVar_FrameRounding(), 12.0 }, + { ImGui.StyleVar_GrabRounding(), 4.0 }, + { ImGui.StyleVar_FrameBorderSize(), 1.0 }, + { ImGui.StyleVar_WindowBorderSize(), 1.0 }, + { ImGui.StyleVar_PopupBorderSize(), 1.0 } + } + }) + + return Theme.theme +end \ No newline at end of file diff --git a/reascripts/ReaSpeech/source/Transcript.lua b/reascripts/ReaSpeech/source/Transcript.lua new file mode 100644 index 0000000000000000000000000000000000000000..34bb3600b34cc1b8e7b87e97f4ca99589da6a8ed --- /dev/null +++ b/reascripts/ReaSpeech/source/Transcript.lua @@ -0,0 +1,486 @@ +--[[ + + Transcript.lua - Speech transcription data model + +]]-- + +Transcript = Polo { + COLUMN_ORDER = {"id", "seek", "start", "end", "text", "score", "file"}, + DEFAULT_HIDE = { + seek = true, temperature = true, tokens = true, avg_logprob = true, + compression_ratio = true, no_speech_prob = true + }, + + init = function(self) + self:clear() + end +} + +Transcript.calculate_offset = function (item, take) + return ( + reaper.GetMediaItemInfo_Value(item, 'D_POSITION') + - reaper.GetMediaItemTakeInfo_Value(take, 'D_STARTOFFS')) +end + +function Transcript:clear() + self.init_data = {} + self.filtered_data = {} + self.data = {} + self.search = '' +end + +function Transcript:get_columns() + if #self.init_data > 0 then + local columns = {"score"} + local row = self.init_data[1] + for k, _ in pairs(row.data) do + if k:sub(1, 1) ~= '_' then + table.insert(columns, k) + end + end + return self:_sort_columns(columns) + end + return {} +end + +function Transcript:_sort_columns(columns) + local order = self.COLUMN_ORDER + + local column_set = {} + local extra_columns = {} + local order_set = {} + local result = {} + + for _, column in pairs(columns) do + column_set[column] = true + end + + for _, column in pairs(order) do + order_set[column] = true + if column_set[column] then + table.insert(result, column) + end + end + + for _, column in pairs(columns) do + if not order_set[column] then + table.insert(extra_columns, column) + end + end + + table.sort(extra_columns) + for _, column in pairs(extra_columns) do + table.insert(result, column) + end + + return result +end + +function Transcript:add_segment(segment) + table.insert(self.init_data, segment) +end + +function Transcript:has_segments() + return #self.init_data > 0 +end + +function Transcript:get_segments() + return self.data +end + +function Transcript:sort(column, ascending) + self.data = {table.unpack(self.filtered_data)} + table.sort(self.data, function (a, b) + local a_val, b_val = a:get(column), b:get(column) + if a_val == nil then a_val = '' end + if b_val == nil then b_val = '' end + if not ascending then + a_val, b_val = b_val, a_val + end + return a_val < b_val + end) +end + +function Transcript:to_table() + local segments = {} + for _, segment in pairs(self.data) do + table.insert(segments, segment:to_table()) + end + return {segments = segments} +end + +function Transcript:to_json() + return json.encode(self:to_table()) +end + +function Transcript:update() + if #self.init_data == 0 then + self:clear() + return + end + + local columns = self:get_columns() + + if #self.search > 0 then + local search = self.search + local search_lower = search:lower() + local match_case = (search ~= search_lower) + self.filtered_data = {} + + for _, segment in pairs(self.init_data) do + local matching = false + for _, column in pairs(columns) do + if match_case then + if tostring(segment.data[column]):find(search) then + matching = true + break + end + else + if tostring(segment.data[column]):lower():find(search_lower) then + matching = true + break + end + end + end + if matching then + table.insert(self.filtered_data, segment) + end + end + else + self.filtered_data = self.init_data + end + + self.data = self.filtered_data +end + +function Transcript:create_markers(proj, regions, words) + proj = proj or 0 + regions = regions or false + for i, segment in pairs(self.data) do + local offset = self.calculate_offset(segment.item, segment.take) + local want_index = segment:get('id', i) + local color = 0 + if words then + for _, word in pairs(segment.words) do + local start = word.start + offset + local end_ = word.end_ + offset + local name = word.word + reaper.AddProjectMarker2(proj, regions, start, end_, name, want_index, color) + end + else + local start = segment.start + offset + local end_ = segment.end_ + offset + local name = segment.text + reaper.AddProjectMarker2(proj, regions, start, end_, name, want_index, color) + end + end +end + +function Transcript:create_notes_track(words) + local cur_pos = reaper.GetCursorPosition() + local index = 0 + reaper.InsertTrackAtIndex(index, false) + local track = reaper.GetTrack(0, index) + reaper.SetOnlyTrackSelected(track) + reaper.GetSetMediaTrackInfo_String(track, 'P_NAME', 'Speech', true) + for _, segment in pairs(self.data) do + local offset = self.calculate_offset(segment.item, segment.take) + if words then + for _, word in pairs(segment.words) do + local start = word.start + offset + local end_ = word.end_ + offset + local text = word.word + self:_create_note(start, end_, text, false) + end + else + local start = segment.start + offset + local end_ = segment.end_ + offset + local text = segment.text + self:_create_note(start, end_, text, true) + end + end + reaper.SetEditCurPos(cur_pos, true, true) +end + +function Transcript:_create_note(start, end_, text, stretch) + local item = self:_create_empty_item(start, end_) + self:_set_note_text(item, text, stretch) +end + +function Transcript:_create_empty_item(start, end_) + self:_insert_empty_item() + local item = reaper.GetSelectedMediaItem(0, 0) + reaper.SelectAllMediaItems(0, false) + reaper.SetMediaItemPosition(item, start, true) + reaper.SetMediaItemLength(item, end_ - start, true) + return item +end + +function Transcript:_insert_empty_item() + reaper.Main_OnCommand(40142, 0) +end + +function Transcript:_set_note_text(item, text, stretch) + local _, chunk = reaper.GetItemStateChunk(item, "", false) + local notes_chunk = ("\n"):format(text:match("^%s*(.-)%s*$")) + local flags_chunk = (stretch and "IMGRESOURCEFLAGS 11\n" or "") + chunk = chunk:gsub('>', notes_chunk:gsub('%%', '%%%%') .. flags_chunk .. '>') + reaper.SetItemStateChunk(item, chunk, false) +end + +TranscriptSegment = Polo { + _proxy_fields = { + start = 'start', + end_ = 'end', + text = 'text', + } +} + +TranscriptSegment.__index = function(o, key) + local proxy_target = TranscriptSegment._proxy_fields[key] + if proxy_target then + return o.data[proxy_target] + else + return TranscriptSegment[key] + end +end + +function TranscriptSegment:init() + assert(self.data, 'missing data') + assert(self.item, 'missing item') + assert(self.take, 'missing take') + self.data = self._copy(self.data) + self.data['file'] = self:get_file() +end + +TranscriptSegment.from_whisper = function(segment, item, take) + local result = {} + local words = segment.words + + segment = TranscriptSegment._copy(segment) + segment.text = segment.text:match("^%s*(.-)%s*$") + segment.words = nil + + if words then + local transcript_words = {} + for _, word in pairs(words) do + local transcript_word = TranscriptWord.new({ + word = word.word:match("^%s*(.-)%s*$"), + probability = word.probability, + start = word.start, + end_ = word['end'] + }) + table.insert(transcript_words, transcript_word) + end + table.insert(result, TranscriptSegment.new({ + data = segment, + item = item, + take = take, + words = transcript_words + })) + else + table.insert(result, TranscriptSegment.new({ + data = segment, + item = item, + take = take + })) + end + + return result +end + +TranscriptSegment.default_hide = function(column) + return Transcript.DEFAULT_HIDE[column] or false +end + +TranscriptSegment.merge_words = function(words, index1, index2) + local word1 = words[index1] + local word2 = words[index2] + local new_word = TranscriptWord.new { + word = word1.word .. word2.word, + start = word1.start, + end_ = word2.end_, + probability = (word1.probability + word2.probability) / 2 + } + table.remove(words, index2) + table.remove(words, index1) + table.insert(words, index1, new_word) +end + +TranscriptSegment.split_word = function(words, index) + local word = words[index] + local length = utf8.len(word.word) + local half_length = math.floor(length / 2) + local new_word1 = TranscriptWord.new { + word = word.word:sub(1, utf8.offset(word.word, half_length)), + start = word.start, + end_ = word.start + (word.end_ - word.start) / 2, + probability = word.probability + } + local new_word2 = TranscriptWord.new { + word = word.word:sub(utf8.offset(word.word, half_length + 1)), + start = word.start + (word.end_ - word.start) / 2, + end_ = word.end_, + probability = word.probability + } + table.remove(words, index) + table.insert(words, index, new_word2) + table.insert(words, index, new_word1) +end + +TranscriptSegment._copy = function(data) + local result = {} + for k, v in pairs(data) do + result[k] = v + end + return result +end + +function TranscriptSegment:score() + local score = 0.0 + if self.words and #self.words > 0 then + for _, word in pairs(self.words) do + score = score + word:score() + end + return score / #self.words + else + return 0.0 + end +end + +function TranscriptSegment:get(column, default) + if column == 'score' then + return self:score() + elseif self.data[column] then + return self.data[column] + else + return default + end +end + +function TranscriptSegment:set_words(words) + self.words = words + self:update_text() +end + +function TranscriptSegment:update_text() + local text_chunks = {} + for _, word in pairs(self.words) do + table.insert(text_chunks, word.word) + end + self.data['text'] = table.concat(text_chunks, ' ') +end + +function TranscriptSegment:get_file(include_extensions) + include_extensions = include_extensions or false + + local file = '' + app:trap(function () + local source = reaper.GetMediaItemTake_Source(self.take) + if source then + local source_path = reaper.GetMediaSourceFileName(source) + + file = source_path:gsub(".*[\\/](.*)", "%1") + + if not include_extensions then + file = file:gsub("(.*)[.].*", "%1") + end + end + end) + return file +end + +function TranscriptSegment:get_file_with_extension() + return self:get_file(true) +end + +function TranscriptSegment:navigate(word_index, autoplay) + local start = self.start + if word_index then + start = self.words[word_index].start + end + local offset = start - reaper.GetMediaItemTakeInfo_Value(self.take, 'D_STARTOFFS') + self:_navigate_to_media_item(self.item) + reaper.MoveEditCursor(offset, false) + if autoplay and reaper.GetPlayState() & 1 == 0 then + self:_transport_play() + end + if reaper.GetPlayState() & 1 == 1 then + self:_transport_play() + end +end + +function TranscriptSegment:_navigate_to_media_item(item) + reaper.SelectAllMediaItems(0, false) + reaper.SetMediaItemSelected(item, true) + self:_move_cursor_to_start_of_items() +end + +function TranscriptSegment:_move_cursor_to_start_of_items() + reaper.Main_OnCommand(41173, 0) +end + +function TranscriptSegment:_transport_play() + reaper.Main_OnCommand(1007, 0) +end + +function TranscriptSegment:to_json() + return json.encode(self:to_table()) +end + +function TranscriptSegment:to_table() + local result = self._copy(self.data) + if self.words then + result['words'] = {} + for _, word in pairs(self.words) do + table.insert(result['words'], word:to_table()) + end + end + return result +end + +function TranscriptSegment:select_in_timeline(offset) + offset = offset or 0 + local start = self.start + offset + local end_ = self.end_ + offset + + reaper.GetSet_LoopTimeRange(true, true, start, end_, false) +end + +TranscriptWord = Polo {} + +function TranscriptWord:init() + assert(self.word, 'missing word') + assert(self.start, 'missing start') + assert(self.end_, 'missing end_') + assert(self.probability, 'missing probability') +end + +function TranscriptWord:copy() + return TranscriptWord.new { + word = self.word, + start = self.start, + end_ = self.end_, + probability = self.probability, + } +end + +function TranscriptWord:score() + return self.probability +end + +function TranscriptWord:to_table() + return { + word = self.word, + start = self.start, + ['end'] = self.end_, + probability = self.probability, + } +end + +function TranscriptWord:select_in_timeline(offset) + offset = offset or 0 + local start = self.start + offset + local end_ = self.end_ + offset + reaper.GetSet_LoopTimeRange(true, true, start, end_, false) +end diff --git a/reascripts/ReaSpeech/source/TranscriptEditor.lua b/reascripts/ReaSpeech/source/TranscriptEditor.lua new file mode 100644 index 0000000000000000000000000000000000000000..7555ab88ef774bbed7b14471c48f816694f48786 --- /dev/null +++ b/reascripts/ReaSpeech/source/TranscriptEditor.lua @@ -0,0 +1,412 @@ +--[[ + + TranscriptEditor.lua - Transcript editor UI + +]]-- + +TranscriptEditor = Polo { + TITLE = 'Edit Segment', + WIDTH = 500, + HEIGHT = 500, + MIN_CONTENT_WIDTH = 375, + BUTTON_WIDTH = 120, + WORDS_PER_LINE = 5, + + ZOOM_LEVEL = { + NONE = { value = "none", description = "None" }, + WORD = { value = "word", description = "Word" }, + SEGMENT = { value = "segment", description = "Segment" }, + }, +} + +function TranscriptEditor:init() + assert(self.transcript, 'missing transcript') + self.editing = nil + self.is_open = false + self.sync_time_selection = false + self.zoom_level = self.ZOOM_LEVEL.NONE.value +end + +function TranscriptEditor:edit_segment(segment, index) + self.editing = { + segment = segment, + words = {}, + index = index, + text = segment:get('text'), + } + for i, word in pairs(segment.words) do + self.editing.words[i] = word:copy() + end + self:edit_word(self.editing.words[1], 1) +end + +function TranscriptEditor:edit_word(word, index) + self.editing.word = word + self.editing.word_index = index + if self.sync_time_selection then + self:update_time_selection() + self:zoom(self.zoom_level) + end +end + +function TranscriptEditor:render() + if not self.editing then + return + end + + local opening = not self.is_open + if opening then + self:_open() + end + + local center = {ImGui.Viewport_GetCenter(ImGui.GetWindowViewport(ctx))} + ImGui.SetNextWindowPos(ctx, center[1], center[2], ImGui.Cond_Appearing(), 0.5, 0.5) + ImGui.SetNextWindowSize(ctx, self.WIDTH, self.HEIGHT, ImGui.Cond_FirstUseEver()) + + if ImGui.BeginPopupModal(ctx, self.TITLE, true, ImGui.WindowFlags_AlwaysAutoResize()) then + app:trap(function () self:render_content() end) + ImGui.EndPopup(ctx) + else + self:_close() + end +end + +function TranscriptEditor:render_content() + if self.editing.word then + self:render_word_navigation() + self:render_separator() + end + + local edit_requested = self:render_words() + + if self.editing.word then + self:render_separator() + self:render_word_actions() + self:render_word_inputs() + end + + if edit_requested then + self:edit_word(table.unpack(edit_requested)) + end + + self:render_separator() + + if ImGui.Button(ctx, 'Save', self.BUTTON_WIDTH, 0) then + self:handle_save() + self:_close() + end + + ImGui.SameLine(ctx) + if ImGui.Button(ctx, 'Cancel', self.BUTTON_WIDTH, 0) then + self:_close() + end +end + +function TranscriptEditor:render_word_navigation() + local words = self.editing.words + local word_index = self.editing.word_index + local num_words = #words + local spacing = ImGui.GetStyleVar(ctx, ImGui.StyleVar_ItemInnerSpacing()) + local disable_if = ReaUtil.disabler(ctx, app.onerror) + + ImGui.PushButtonRepeat(ctx, true) + app:trap(function () + if ImGui.ArrowButton(ctx, '##left', ImGui.Dir_Left()) then + local index = self.editing.word_index - 1 + if index > 0 then + self:edit_word(words[index], index) + end + end + ImGui.SameLine(ctx, 0, spacing) + if ImGui.ArrowButton(ctx, '##right', ImGui.Dir_Right()) then + local index = self.editing.word_index + 1 + if index <= num_words then + self:edit_word(words[index], index) + end + end + end) + ImGui.PopButtonRepeat(ctx) + + ImGui.SameLine(ctx) + ImGui.AlignTextToFramePadding(ctx) + ImGui.Text(ctx, 'Word ' .. word_index .. ' / ' .. num_words) + + ImGui.SameLine(ctx) + if ImGui.Button(ctx, 'Add') then + self:handle_word_add() + end + app:tooltip('Add word after current word') + + ImGui.SameLine(ctx, 0, spacing) + + disable_if(num_words <= 1, function() + if ImGui.Button(ctx, 'Delete') then + self:handle_word_delete() + end + end) + app:tooltip('Delete current word') + + ImGui.SameLine(ctx, 0, spacing) + if ImGui.Button(ctx, 'Split') then + self:handle_word_split() + end + app:tooltip('Split current word into two words') + + ImGui.SameLine(ctx, 0, spacing) + disable_if(word_index >= num_words, function() + if ImGui.Button(ctx, 'Merge') then + self:handle_word_merge() + end + end) + app:tooltip('Merge current word with next word') +end + +function TranscriptEditor:render_words() + local words = self.editing.words + local num_words = #words + local spacing = ImGui.GetStyleVar(ctx, ImGui.StyleVar_ItemInnerSpacing()) + local edit_requested = nil + + for i, word in pairs(words) do + if self.editing.word_index ~= i then + ImGui.PushStyleColor(ctx, ImGui.Col_Button(), 0xffffff33) + end + app:trap(function() + if ImGui.Button(ctx, word.word .. '##' .. i) then + edit_requested = {word, i} + end + end) + if self.editing.word_index ~= i then + ImGui.PopStyleColor(ctx) + end + + if i < num_words and i % self.WORDS_PER_LINE ~= 0 then + ImGui.SameLine(ctx, 0, spacing) + end + end + + return edit_requested +end + +function TranscriptEditor:render_word_inputs() + self:render_word_input() + + if self.sync_time_selection then + local sel_start, sel_end = reaper.GetSet_LoopTimeRange(false, false, 0, 0, false) + local offset = Transcript.calculate_offset(self.editing.segment.item, self.editing.segment.take) + self.editing.word.start = sel_start - offset + self.editing.word.end_ = sel_end - offset + end + + self:render_time_input('start', self.editing.word.start, function (time) + self.editing.word.start = time + end) + + self:render_time_input('end', self.editing.word.end_, function (time) + self.editing.word.end_ = time + end) + + self:render_score_input() +end + +function TranscriptEditor:render_word_input() + local rv, value = ImGui.InputText(ctx, 'word', self.editing.word.word) + if rv then + value = value:gsub('^%s*(.-)%s*$', '%1') + if #value > 0 then + self.editing.word.word = value + end + end +end + +function TranscriptEditor:render_time_input(label, value, callback) + local value_str = reaper.format_timestr(value, '') + local rv, new_value = ImGui.InputText(ctx, label, value_str) + if rv then + callback(reaper.parse_timestr(new_value)) + end +end + +function TranscriptEditor:render_score_input() + local color = TranscriptUI.score_color(self.editing.word:score()) + if color then + ImGui.PushStyleColor(ctx, ImGui.Col_SliderGrab(), color) + ImGui.PushStyleColor(ctx, ImGui.Col_SliderGrabActive(), color) + end + app:trap(function () + local rv, value = ImGui.SliderDouble(ctx, 'score', self.editing.word.probability, 0, 1) + if rv then + self.editing.word.probability = value + end + end) + if color then + ImGui.PopStyleColor(ctx, 2) + end +end + +function TranscriptEditor:render_icon_button(icon, callback) + ImGui.PushFont(ctx, Fonts.icons) + app:trap(function () + if ImGui.Button(ctx, Fonts.ICON[icon]) then + callback() + end + end) + ImGui.PopFont(ctx) +end + +function TranscriptEditor:render_word_actions() + self:render_icon_button('play', function () + self:update_time_selection() + reaper.Main_OnCommand(1016, 0) -- Transport: Stop + reaper.Main_OnCommand(40630, 0) -- Go to start of time selection + reaper.Main_OnCommand(40044, 0) -- Transport: Play/stop + end) + app:tooltip('Play word') + + local spacing = ImGui.GetStyleVar(ctx, ImGui.StyleVar_ItemInnerSpacing()) + ImGui.SameLine(ctx, 0, spacing) + + self:render_icon_button('stop', function () + reaper.Main_OnCommand(1016, 0) -- Transport: Stop + end) + app:tooltip('Stop') + + ImGui.SameLine(ctx) + local rv, value = ImGui.Checkbox(ctx, 'sync time selection', self.sync_time_selection) + if rv then + self.sync_time_selection = value + if value then + self:update_time_selection() + end + end + + ImGui.SameLine(ctx) + ImGui.Text(ctx, 'and') + ImGui.SameLine(ctx) + self:render_zoom_combo() +end + +function TranscriptEditor:render_zoom_combo() + local disable_if = ReaUtil.disabler(ctx, app.onerror) + + ImGui.SameLine(ctx) + ImGui.Text(ctx, "zoom to") + ImGui.SameLine(ctx) + ImGui.PushItemWidth(ctx, self.BUTTON_WIDTH) + app:trap(function() + disable_if(not self.sync_time_selection, function() + if ImGui.BeginCombo(ctx, "##zoom_level", self.zoom_level) then + app:trap(function() + for _, zoom in pairs(self.ZOOM_LEVEL) do + if ImGui.Selectable(ctx, zoom.description, self.zoom_level == zoom.value) then + self.zoom_level = zoom.value + self:handle_zoom_change() + end + end + end) + ImGui.EndCombo(ctx) + end + end) + end) + ImGui.PopItemWidth(ctx) +end + +function TranscriptEditor:offset() + return Transcript.calculate_offset(self.editing.segment.item, self.editing.segment.take) +end + +function TranscriptEditor:zoom(zoom_level) + -- save current selection + local start, end_ = reaper.GetSet_LoopTimeRange(false, true, 0, 0, false) + + if zoom_level == self.ZOOM_LEVEL.WORD.value then + self.editing.word:select_in_timeline(self:offset()) + elseif zoom_level == self.ZOOM_LEVEL.SEGMENT.value then + self.editing.segment:select_in_timeline(self:offset()) + else + return + end + + -- View: Zoom time selection + reaper.Main_OnCommandEx(40031, 1) + + -- restore selection + reaper.GetSet_LoopTimeRange(true, true, start, end_, false) +end + +function TranscriptEditor:render_separator() + ImGui.Dummy(ctx, self.MIN_CONTENT_WIDTH, 0) + ImGui.Separator(ctx) + ImGui.Dummy(ctx, 0, 0) +end + +function TranscriptEditor:update_time_selection() + if self.editing then + self.editing.word:select_in_timeline(self:offset()) + end +end + +function TranscriptEditor:handle_save() + if self.editing then + local segment = self.editing.segment + segment:set_words(self.editing.words) + self.transcript:update() + end +end + +function TranscriptEditor:handle_word_add() + local words = self.editing.words + local word_index = self.editing.word_index + table.insert(words, word_index + 1, TranscriptWord.new { + word = '...', + start = words[word_index].end_, + end_ = words[word_index].end_, + probability = 1.0 + }) + self:edit_word(words[word_index + 1], word_index + 1) +end + +function TranscriptEditor:handle_word_delete() + local words = self.editing.words + local word_index = self.editing.word_index + table.remove(words, word_index) + local num_words = #words + if word_index > num_words then + word_index = num_words + end + self:edit_word(words[word_index], word_index) +end + +function TranscriptEditor:handle_word_split() + local words = self.editing.words + local word_index = self.editing.word_index + TranscriptSegment.split_word(words, word_index) + self:edit_word(words[word_index], word_index) +end + +function TranscriptEditor:handle_word_merge() + local words = self.editing.words + local word_index = self.editing.word_index + local num_words = #words + if word_index < num_words then + TranscriptSegment.merge_words(words, word_index, word_index + 1) + self:edit_word(words[word_index], word_index) + end +end + +function TranscriptEditor:handle_zoom_change() + if self.sync_time_selection then + self:zoom(self.zoom_level) + end +end + +function TranscriptEditor:_open() + ImGui.OpenPopup(ctx, self.TITLE) + self.is_open = true +end + +function TranscriptEditor:_close() + ImGui.CloseCurrentPopup(ctx) + self.editing = nil + self.is_open = false +end diff --git a/reascripts/ReaSpeech/source/TranscriptExporter.lua b/reascripts/ReaSpeech/source/TranscriptExporter.lua new file mode 100644 index 0000000000000000000000000000000000000000..3efb5912966afaea622eeab560dfdfe9aae00e6e --- /dev/null +++ b/reascripts/ReaSpeech/source/TranscriptExporter.lua @@ -0,0 +1,374 @@ +--[[ + + TranscriptExporter.lua - Transcript export UI + +]]-- + +TranscriptExporter = Polo { + TITLE = 'Export', + WIDTH = 650, + HEIGHT = 200, + BUTTON_WIDTH = 120, + INPUT_WIDTH = 120, + FILE_WIDTH = 500, + +} + +function TranscriptExporter:init() + assert(self.transcript, 'missing transcript') + self.is_open = false + self.export_formats = TranscriptExporterFormats.new { + TranscriptExportFormat.exporter_json(), + TranscriptExportFormat.exporter_srt(), + TranscriptExportFormat.exporter_csv(), + } + self.export_options = {} + self.file = '' + self.success = AlertPopup.new { title = 'Export Successful' } + self.failure = AlertPopup.new { title = 'Export Failed' } +end + +function TranscriptExporter:show_success() + self.success.onclose = function () + self.success.onclose = nil + self:close() + end + self.success:show('Exported ' .. self.export_formats:selected_key() .. ' to: ' .. self.file) +end + +function TranscriptExporter:show_error(msg) + self.failure:show(msg) +end + +function TranscriptExporter:render() + if not self.is_open then + return + end + + local center = {ImGui.Viewport_GetCenter(ImGui.GetWindowViewport(ctx))} + ImGui.SetNextWindowPos(ctx, center[1], center[2], ImGui.Cond_Appearing(), 0.5, 0.5) + ImGui.SetNextWindowSize(ctx, self.WIDTH, self.HEIGHT, ImGui.Cond_FirstUseEver()) + + local flags = ( + 0 + | ImGui.WindowFlags_AlwaysAutoResize() + | ImGui.WindowFlags_NoCollapse() + | ImGui.WindowFlags_NoDocking() + ) + + local visible, open = ImGui.Begin(ctx, self.TITLE, true, flags) + if visible then + app:trap(function () + self:render_content() + self.success:render() + self.failure:render() + end) + ImGui.End(ctx) + end + if not open then + self:close() + end +end + +function TranscriptExporter:render_content() + self.export_formats:render_combo(self.INPUT_WIDTH) + + ImGui.Spacing(ctx) + + self.export_formats:render_format_options(self.export_options) + + ImGui.Spacing(ctx) + + self:render_file_selector() + + self:render_separator() + + self:render_buttons() +end + +-- Display a text input for the output filename, with a Browse button if +-- the js_ReaScriptAPI extension is available. +function TranscriptExporter:render_file_selector() + ImGui.Text(ctx, 'File') + if app:has_js_ReaScriptAPI() then + if ImGui.Button(ctx, 'Choose File', self.BUTTON_WIDTH, 0) then + local rv, file = app:show_file_dialog { + title = 'Save transcript', + file = self.file, + save = true, + ext = self.export_formats:file_selector_spec(), + } + if rv == 1 then + self.file = file + end + end + ImGui.SameLine(ctx) + end + + ImGui.SetNextItemWidth(ctx, self.FILE_WIDTH) + local file_changed, file = ImGui.InputText(ctx, '##file', self.file, 256) + if file_changed then + self.file = file + end + + if not app:has_js_ReaScriptAPI() then + ImGui.Text(ctx, "For a better experience, install js_ReaScriptAPI") + ImGui.Spacing(ctx) + end +end + +function TranscriptExporter:render_buttons() + ReaUtil.disabler(ctx)(self.file == '', function() + if ImGui.Button(ctx, 'Export', self.BUTTON_WIDTH, 0) then + if self:handle_export() then + self:show_success() + end + end + end) + + ImGui.SameLine(ctx) + if ImGui.Button(ctx, 'Cancel', self.BUTTON_WIDTH, 0) then + self:close() + end +end + +function TranscriptExporter:render_separator() + ImGui.Dummy(ctx, 0, 0) + ImGui.Separator(ctx) + ImGui.Dummy(ctx, 0, 0) +end + +function TranscriptExporter:handle_export() + if self.file == '' then + self:show_error('Please specify a file name.') + return false + end + local file = io.open(self.file, 'w') + if not file then + self:show_error('Could not open file: ' .. self.file) + return false + end + self.export_formats:write(self.transcript, file, self.export_options) + file:close() + return true +end + +function TranscriptExporter:open() + self.is_open = true +end + +function TranscriptExporter:close() + self.is_open = false +end + +TranscriptExporterFormats = Polo { + new = function(formatters) + local format_map = {} + + for i, formatter in ipairs(formatters) do + format_map[formatter.key] = i + end + + return { + formatters = formatters, + format_map = format_map, + } + end, +} + +function TranscriptExporterFormats:render_combo(width) + ImGui.Text(ctx, 'Format') + ImGui.SetNextItemWidth(ctx, width) + if ImGui.BeginCombo(ctx, "##format", self.selected_format_key) then + app:trap(function() + for _, format in pairs(self.formatters) do + local is_selected = self.selected_format_key == format.key + if ImGui.Selectable(ctx, format.key, is_selected) then + self.selected_format_key = format.key + end + if is_selected then + ImGui.SetItemDefaultFocus(ctx) + end + end + end) + ImGui.EndCombo(ctx) + end +end + +function TranscriptExporterFormats:selected_key() + return self:selected_format().key +end + +function TranscriptExporterFormats:file_selector_spec() + return self:selected_format():file_selector_spec() +end + +function TranscriptExporterFormats:write(transcript, output_file, options) + return self:selected_format().writer(transcript, output_file, options) +end + +function TranscriptExporterFormats:selected_format() + if not self.selected_format_key then + if not self.formatters or #self.formatters < 1 then + app:debug('no formats to set for default') + return + end + + self.selected_format_key = self.formatters[1].key + end + + local index = self.format_map[self.selected_format_key] + + return self.formatters[index] +end + +function TranscriptExporterFormats:render_format_options(options) + app:trap(function() + local format = self:selected_format() + + if format then + format.option_renderer(options) + end + end) +end + +TranscriptExportFormat = Polo { + OPTIONS_NOOP = function(_options) end, + + new = function (key, extension, option_renderer, writer_f) + return { + key = key, + extension = extension, + option_renderer = option_renderer, + writer = writer_f, + } + end, +} + +function TranscriptExportFormat:file_selector_spec() + local selector_spec = '%s files (*.%s)\0*.%s\0All files (*.*)\0*.*\0\0' + return selector_spec:format(self.key, self.extension, self.extension) +end + +function TranscriptExportFormat.exporter_json() + return TranscriptExportFormat.new( + 'JSON', 'json', + TranscriptExportFormat.options_json, + TranscriptExportFormat.writer_json + ) +end + +function TranscriptExportFormat.options_json(options) + local rv, value = ImGui.Checkbox(ctx, 'One Object per Transcript Segment', options.object_per_segment) + if rv then + options.object_per_segment = value + end +end + +function TranscriptExportFormat.writer_json(transcript, output_file, options) + if options.object_per_segment then + for _, segment in pairs(transcript:get_segments()) do + output_file:write(segment:to_json()) + output_file:write('\n') + end + else + output_file:write(transcript:to_json()) + end +end + +function TranscriptExportFormat.exporter_srt() + return TranscriptExportFormat.new( + 'SRT', 'srt', + TranscriptExportFormat.options_srt, + TranscriptExportFormat.writer_srt + ) +end + +function TranscriptExportFormat.strip_non_numeric(value) + return value:gsub("[^0-9]", ""):gsub("^0+", "") +end + +function TranscriptExportFormat.options_srt(options) + local rv, value + + rv, value = ImGui.InputText(ctx, 'X1', options.coords_x1, ImGui.InputTextFlags_CharsDecimal()) + if rv then + options.coords_x1 = TranscriptExportFormat.strip_non_numeric(value) + end + + ImGui.SameLine(ctx) + + rv, value = ImGui.InputText(ctx, 'Y1', options.coords_y1, ImGui.InputTextFlags_CharsDecimal()) + if rv then + options.coords_y1 = TranscriptExportFormat.strip_non_numeric(value) + end + + rv, value = ImGui.InputText(ctx, 'X2', options.coords_x2, ImGui.InputTextFlags_CharsDecimal()) + if rv then + options.coords_x2 = TranscriptExportFormat.strip_non_numeric(value) + end + + ImGui.SameLine(ctx) + + rv, value = ImGui.InputText(ctx, 'Y2', options.coords_y2, ImGui.InputTextFlags_CharsDecimal()) + if rv then + options.coords_y2 = TranscriptExportFormat.strip_non_numeric(value) + end +end + +function TranscriptExportFormat.writer_srt(transcript, output_file, options) + local writer = SRTWriter.new { file = output_file, options = options } + writer:write(transcript) +end + +function TranscriptExportFormat.exporter_csv() + return TranscriptExportFormat.new( + 'CSV', 'csv', + TranscriptExportFormat.options_csv, + TranscriptExportFormat.writer_csv + ) +end + +function TranscriptExportFormat.options_csv(options) + local delimiters = CSVWriter.DELIMITERS + + local selected_delimiter = delimiters[1] + + for _, delimiter in ipairs(delimiters) do + if delimiter.char == options.delimiter then + selected_delimiter = delimiter + break + end + end + + if ImGui.BeginCombo(ctx, 'Delimiter', selected_delimiter.name) then + app:trap(function() + for _, delimiter in ipairs(delimiters) do + local is_selected = options.delimiter == delimiter.char + if ImGui.Selectable(ctx, delimiter.name, is_selected) then + options.delimiter = delimiter.char + end + if is_selected then + ImGui.SetItemDefaultFocus(ctx) + end + end + end) + ImGui.EndCombo(ctx) + end + + ImGui.Spacing(ctx) + + local rv, value = ImGui.Checkbox(ctx, 'Include Header Row', options.include_header_row) + if rv then + options.include_header_row = value + end +end + +function TranscriptExportFormat.writer_csv(transcript, output_file, options) + local writer = CSVWriter.new { + file = output_file, + delimiter = options.delimiter, + include_header_row = options.include_header_row + } + writer:write(transcript) +end diff --git a/reascripts/ReaSpeech/source/TranscriptUI.lua b/reascripts/ReaSpeech/source/TranscriptUI.lua new file mode 100644 index 0000000000000000000000000000000000000000..3d3dbb19998fa48257dc4cfb908418780283f4e0 --- /dev/null +++ b/reascripts/ReaSpeech/source/TranscriptUI.lua @@ -0,0 +1,326 @@ +--[[ + + TranscriptUI.lua - @Transcript table & actions UI + +]] + +TranscriptUI = Polo { + TITLE = 'Transcript', + + FLOAT_FORMAT = '%.4f', + + COLUMN_WIDTH = 55, + LARGE_COLUMN_WIDTH = 300, + + ITEM_WIDTH = 125, + + SCORE_COLORS = { + bright_green = 0xa3ff00a6, + dark_green = 0x2cba00a6, + orange = 0xffa700a6, + red = 0xff0000a6 + } +} + +TranscriptUI.table_flags = function (sortable) + local sort_flags = 0 + if sortable then + sort_flags = ImGui.TableFlags_Sortable() | ImGui.TableFlags_SortTristate() + end + return ( + sort_flags + | ImGui.TableFlags_Borders() + | ImGui.TableFlags_Hideable() + | ImGui.TableFlags_Resizable() + | ImGui.TableFlags_Reorderable() + | ImGui.TableFlags_RowBg() + | ImGui.TableFlags_ScrollX() + | ImGui.TableFlags_ScrollY() + | ImGui.TableFlags_SizingFixedFit() + ) +end + +function TranscriptUI:init() + assert(self.transcript, 'missing transcript') + + self.words = false + self.colorize_words = false + self.autoplay = true + + self.transcript_editor = TranscriptEditor.new { transcript = self.transcript } + self.transcript_exporter = TranscriptExporter.new { transcript = self.transcript } +end + +function TranscriptUI:render() + if self.transcript:has_segments() then + ImGui.SeparatorText(ctx, "Transcript") + self:render_result_actions() + self:render_table() + end + + self.transcript_editor:render() + self.transcript_exporter:render() +end + +function TranscriptUI:render_result_actions() + if ImGui.Button(ctx, "Create Regions") then + self:handle_create_markers(true) + end + ImGui.SameLine(ctx) + if ImGui.Button(ctx, "Create Markers") then + self:handle_create_markers(false) + end + ImGui.SameLine(ctx) + if ImGui.Button(ctx, "Create Notes Track") then + self:handle_create_notes_track() + end + + ImGui.SameLine(ctx) + rv, value = ImGui.Checkbox(ctx, "words", self.words) + if rv then + self.words = value + end + + if self.words then + ImGui.SameLine(ctx) + rv, value = ImGui.Checkbox(ctx, "colorize", self.colorize_words) + if rv then + self.colorize_words = value + end + end + + local label_width, _ = ImGui.CalcTextSize(ctx, "search") + ImGui.SameLine(ctx, ImGui.GetWindowWidth(ctx) - self.ITEM_WIDTH - label_width - 10) + local search_changed, search = ImGui.InputText(ctx, 'search', self.transcript.search) + if search_changed then + self:handle_search(search) + end + ImGui.SameLine(ctx, ImGui.GetWindowWidth(ctx) - self.ITEM_WIDTH - label_width - 110) + rv, value = ImGui.Checkbox(ctx, "auto-play", self.autoplay) + if rv then + self.autoplay = value + end + + if self.transcript:has_segments() then + ImGui.Spacing(ctx) + if ImGui.Button(ctx, "Export") then + self:handle_export() + end + + ImGui.SameLine(ctx) + if ImGui.Button(ctx, "Clear") then + self:handle_transcript_clear() + end + end +end + +function TranscriptUI:handle_create_markers(regions) + reaper.PreventUIRefresh(1) + reaper.Undo_BeginBlock() + self.transcript:create_markers(0, regions, self.words) + reaper.Undo_EndBlock( + ("Create %s from speech"):format(regions and 'regions' or 'markers'), -1) + reaper.PreventUIRefresh(-1) +end + +function TranscriptUI:handle_create_notes_track() + reaper.PreventUIRefresh(1) + reaper.Undo_BeginBlock() + self.transcript:create_notes_track(self.words) + reaper.Undo_EndBlock("Create notes track from speech", -1) + reaper.PreventUIRefresh(-1) +end + +function TranscriptUI:handle_export() + self.transcript_exporter:open() +end + +function TranscriptUI:handle_transcript_clear() + self.transcript:clear() +end + +function TranscriptUI:handle_search(search) + self.transcript.search = search + self.transcript:update() +end + +-- formerly ReaSpeechUI:render_table() +function TranscriptUI:render_table() + local columns = self.transcript:get_columns() + local num_columns = #columns + 1 + + local ok = ImGui.BeginTable(ctx, "results", num_columns, self.table_flags(true)) + if ok then + app:trap(function () + ImGui.TableSetupColumn(ctx, "##actions", ImGui.TableColumnFlags_NoSort(), 20) + + for _, column in pairs(columns) do + local column_flags = 0 + if TranscriptSegment.default_hide(column) then + -- reaper.ShowConsoleMsg(string.format('column %s: %s\n', column, TranscriptSegment.default_hide(column))) + column_flags = column_flags | ImGui.TableColumnFlags_DefaultHide() + end + local init_width = self.COLUMN_WIDTH + if column == "text" or column == "file" then + init_width = self.LARGE_COLUMN_WIDTH + end + -- reaper.ShowConsoleMsg(string.format('column %s: %s / flags: %s\n', column, TranscriptSegment.default_hide(column), column_flags)) + ImGui.TableSetupColumn(ctx, column, column_flags, init_width) + end + + ImGui.TableSetupScrollFreeze(ctx, 0, 1) + ImGui.TableHeadersRow(ctx) + + self:sort_table() + + for index, segment in pairs(self.transcript:get_segments()) do + ImGui.TableNextRow(ctx) + ImGui.TableNextColumn(ctx) + self:render_segment_actions(segment, index) + for _, column in pairs(columns) do + ImGui.TableNextColumn(ctx) + self:render_table_cell(segment, column) + end + end + end) + ImGui.EndTable(ctx) + end +end + +function TranscriptUI:render_segment_actions(segment, index) + ImGui.PushFont(ctx, Fonts.icons) + app:trap(function() + ImGui.Text(ctx, Fonts.ICON.pencil) + end) + ImGui.PopFont(ctx) + if ImGui.IsItemHovered(ctx) then + ImGui.SetMouseCursor(ctx, ImGui.MouseCursor_Hand()) + end + if ImGui.IsItemClicked(ctx) then + self.transcript_editor:edit_segment(segment, index) + end + app:tooltip("Edit") +end + +function TranscriptUI:render_table_cell(segment, column) + if column == "text" or column == "word" then + self:render_text(segment, column) + elseif column == "score" then + self:render_score(segment:get(column, 0.0)) + elseif column == "start" or column == "end" then + ImGui.Text(ctx, reaper.format_timestr(segment:get(column, 0.0), '')) + else + local value = segment:get(column) + if type(value) == 'table' then + value = table.concat(value, ', ') + elseif math.type(value) == 'float' then + value = self.FLOAT_FORMAT:format(value) + end + ImGui.Text(ctx, tostring(value)) + end +end + +function TranscriptUI:render_link(text, onclick, text_color, underline_color) + text_color = text_color or 0xffffffff + underline_color = underline_color or 0xffffffa0 + + ImGui.TextColored(ctx, text_color, text) + + if ImGui.IsItemHovered(ctx) then + local rect_min_x, rect_min_y = ImGui.GetItemRectMin(ctx) + local rect_max_x, _ = ImGui.GetItemRectMax(ctx) + local _, rect_size_y = ImGui.GetItemRectSize(ctx) + local line_y = rect_min_y + rect_size_y - 1 + + ImGui.DrawList_AddLine( + ImGui.GetWindowDrawList(ctx), + rect_min_x, line_y, rect_max_x, line_y, + underline_color, 1.0) + ImGui.SetMouseCursor(ctx, ImGui.MouseCursor_Hand()) + end + + if ImGui.IsItemClicked(ctx) then + onclick() + end +end + +function TranscriptUI:render_text(segment, column) + if self.words then + self:render_text_words(segment, column) + else + self:render_text_simple(segment, column) + end +end + +function TranscriptUI:render_text_simple(segment, column) + self:render_link(segment:get(column, ""), function () segment:navigate(nil,self.autoplay) end) +end + +function TranscriptUI:render_text_words(segment, _) + if segment.words then + for i, word in pairs(segment.words) do + if i > 1 then + ImGui.SameLine(ctx, 0, 0) + ImGui.Text(ctx, ' ') + ImGui.SameLine(ctx, 0, 0) + end + local color = nil + if self.colorize_words then + color = self.score_color(word:score()) + end + self:render_link(word.word, function () segment:navigate(i, self.autoplay) end, color) + end + end +end + +function TranscriptUI:render_score(value) + local w, h = 50 * value, 3 + local color = self.score_color(value) + if color then + local draw_list = ImGui.GetWindowDrawList(ctx) + local x, y = ImGui.GetCursorScreenPos(ctx) + y = y + 7 + ImGui.DrawList_AddRectFilled(draw_list, x, y, x + w, y + h, color) + end + ImGui.Dummy(ctx, w, h) +end + +function TranscriptUI.score_color(value) + local colors = TranscriptUI.SCORE_COLORS + + if value > 0.9 then + return colors.bright_green + elseif value > 0.8 then + return colors.dark_green + elseif value > 0.7 then + return colors.orange + elseif value > 0.0 then + return colors.red + else + return nil + end +end + +function TranscriptUI:sort_table() + local specs_dirty, has_specs = ImGui.TableNeedSort(ctx) + if has_specs and specs_dirty then + local columns = self.transcript:get_columns() + local column = nil + local ascending = true + + for next_id = 0, math.huge do + local ok, _, col_idx, _, sort_direction = + ImGui.TableGetColumnSortSpecs(ctx, next_id) + if not ok then break end + + column = columns[col_idx] + ascending = (sort_direction == ImGui.SortDirection_Ascending()) + end + + if column then + self.transcript:sort(column, ascending) + else + self.transcript:update() + end + end +end diff --git a/reascripts/ReaSpeech/source/include/globals.lua b/reascripts/ReaSpeech/source/include/globals.lua new file mode 100644 index 0000000000000000000000000000000000000000..d6d1fdbc954fc9079674b56a9fbfc3a9b2c20773 --- /dev/null +++ b/reascripts/ReaSpeech/source/include/globals.lua @@ -0,0 +1,26 @@ +--[[ + + globals.lua - Global functions + +]]-- + +-- Create ImGui namespace +ImGui = {} +for name, func in pairs(reaper) do + name = name:match('^ImGui_(.+)$') + if name then ImGui[name] = func end +end + +-- For debugging +function dump(o) + if type(o) == 'table' then + local s = '{ ' + for k,v in pairs(o) do + if type(k) ~= 'number' then k = '"'..k..'"' end + s = s .. '['..k..'] = ' .. dump(v) .. ',' + end + return s .. '} ' + else + return tostring(o) + end +end diff --git a/reascripts/ReaSpeech/source/include/header.lua b/reascripts/ReaSpeech/source/include/header.lua new file mode 100644 index 0000000000000000000000000000000000000000..a533fb4c6554323f22de9df5e1063f529fceed3b --- /dev/null +++ b/reascripts/ReaSpeech/source/include/header.lua @@ -0,0 +1,6 @@ +--[[ + + ReaSpeech.lua - Speech recognition tool for Reaper + +]]-- + diff --git a/reascripts/ReaSpeech/source/include/main.lua b/reascripts/ReaSpeech/source/include/main.lua new file mode 100644 index 0000000000000000000000000000000000000000..6eb00754d603175b5a851e163ff6724f52c101d8 --- /dev/null +++ b/reascripts/ReaSpeech/source/include/main.lua @@ -0,0 +1,7 @@ +--[[ + + main.lua - ReaSpeech main routine + +]]-- + +ReaSpeechMain:main() diff --git a/reascripts/ReaSpeech/tests/TestCSVWriter.lua b/reascripts/ReaSpeech/tests/TestCSVWriter.lua new file mode 100644 index 0000000000000000000000000000000000000000..0b98d7daa3c0ef4b759e4ff5e82c7ba112dc5daa --- /dev/null +++ b/reascripts/ReaSpeech/tests/TestCSVWriter.lua @@ -0,0 +1,112 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +app = {} + +local lu = require('luaunit') + +require('mock_reaper') +require('Polo') +require('source/CSVWriter') +require('source/Transcript') + +-- + +reaper.GetMediaItemTake_Source = function () return {fileName = "test_audio.wav"} end +reaper.GetMediaSourceFileName = function (source) return source.fileName end + +TestCSVWriter = {} + +function TestCSVWriter:setUp() + function app:trap(f) return xpcall(f, function(e) print(tostring(e)) end) end + + reaper.__test_setUp() +end + +function TestCSVWriter.make_transcript() + local t = Transcript.new() + t:add_segment(TranscriptSegment.new { + data = {start = 0, ['end'] = 1, text = 'hello'}, + item = {}, + take = {} + }) + t:add_segment(TranscriptSegment.new { + data = {start = 1, ['end'] = 2, text = 'world'}, + item = {}, + take = {} + }) + t:add_segment(TranscriptSegment.new { + data = {start = 2, ['end'] = 3, text = 'something in "quotes"'}, + item = {}, + take = {} + }) + t:update() + + return t +end + +function TestCSVWriter:testFormatTime() + lu.assertEquals(CSVWriter.format_time(0), '00:00:00,000') + lu.assertEquals(CSVWriter.format_time(1), '00:00:01,000') + lu.assertEquals(CSVWriter.format_time(1.5), '00:00:01,500') + lu.assertEquals(CSVWriter.format_time(60), '00:01:00,000') + lu.assertEquals(CSVWriter.format_time(60.5), '00:01:00,500') + lu.assertEquals(CSVWriter.format_time(3600), '01:00:00,000') + lu.assertEquals(CSVWriter.format_time(3600.5), '01:00:00,500') + end + +function TestCSVWriter:testInit() + local f = {} + local writer = CSVWriter.new { file = f } +end + +function TestCSVWriter:testInitNoFile() + lu.assertErrorMsgContains('missing file', CSVWriter.new) +end + +function TestCSVWriter:testWrite() + local t = TestCSVWriter.make_transcript() + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = CSVWriter.new { file = f } + writer:write(t) + local output_str = table.concat(output) + lu.assertEquals(output_str, '1,"00:00:00,000","00:00:01,000","hello","test_audio.wav"\n2,"00:00:01,000","00:00:02,000","world","test_audio.wav"\n3,"00:00:02,000","00:00:03,000","something in ""quotes""","test_audio.wav"\n') +end + +function TestCSVWriter:testCustomDelimiter() + local t = TestCSVWriter.make_transcript() + + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = CSVWriter.new { file = f, delimiter = '\t' } + writer:write(t) + local output_str = table.concat(output) + lu.assertEquals(output_str, '1\t"00:00:00,000"\t"00:00:01,000"\t"hello"\t"test_audio.wav"\n2\t"00:00:01,000"\t"00:00:02,000"\t"world"\t"test_audio.wav"\n3\t"00:00:02,000"\t"00:00:03,000"\t"something in ""quotes"""\t"test_audio.wav"\n') +end + +function TestCSVWriter:testIncludeHeaderRow() + local t = TestCSVWriter.make_transcript() + + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = CSVWriter.new { file = f, include_header_row = true } + writer:write(t) + local output_str = table.concat(output) + lu.assertEquals(output_str, '"Sequence Number","Start Time","End Time","Text","File"\n1,"00:00:00,000","00:00:01,000","hello","test_audio.wav"\n2,"00:00:01,000","00:00:02,000","world","test_audio.wav"\n3,"00:00:02,000","00:00:03,000","something in ""quotes""","test_audio.wav"\n') +end + +-- + +os.exit(lu.LuaUnit.run()) \ No newline at end of file diff --git a/reascripts/ReaSpeech/tests/TestReaSpeechUI.lua b/reascripts/ReaSpeech/tests/TestReaSpeechUI.lua new file mode 100644 index 0000000000000000000000000000000000000000..7b35d6f23192836a4f5fa7618d7c069c6bfc560d --- /dev/null +++ b/reascripts/ReaSpeech/tests/TestReaSpeechUI.lua @@ -0,0 +1,43 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +local lu = require('luaunit') + +require('OptionsConfig') +require('Polo') +require('ReaUtil') +require('mock_reaper') + +require('source/AlertPopup') +require('source/ReaSpeechActionsUI') +require('source/ReaSpeechAPI') +require('source/ReaSpeechProductActivation') +require('source/ReaSpeechProductActivationUI') +require('source/ReaSpeechUI') +require('source/ReaSpeechControlsUI') +require('source/ReaSpeechWorker') +require('source/Transcript') +require('source/TranscriptUI') +require('source/TranscriptEditor') +require('source/TranscriptExporter') + +-- + +TestReaSpeechUI = {} + +function TestReaSpeechUI:setUp() + reaper.__test_setUp() + Script = { + host = "localhost:9000" + } + self.app = ReaSpeechUI.new() +end + +function TestReaSpeechUI:testInit() + lu.assertEquals(#self.app.requests, 0) + lu.assertEquals(#self.app.responses, 0) + lu.assertEquals(#self.app.logs, 0) +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/ReaSpeech/tests/TestSRTWriter.lua b/reascripts/ReaSpeech/tests/TestSRTWriter.lua new file mode 100644 index 0000000000000000000000000000000000000000..caf88407af532359be8cf46ba530fe1117088c91 --- /dev/null +++ b/reascripts/ReaSpeech/tests/TestSRTWriter.lua @@ -0,0 +1,135 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +app = {} + +local lu = require('luaunit') + +require('mock_reaper') +require('Polo') +require('source/SRTWriter') +require('source/Transcript') + +-- + +reaper.GetMediaItemTake_Source = function () return {fileName = "test_audio.wav"} end +reaper.GetMediaSourceFileName = function (source) return source.fileName end + +TestSRTWriter = {} + +function TestSRTWriter:setUp() + function app:trap(f) return xpcall(f, function(e) print(tostring(e)) end) end + + reaper.__test_setUp() +end + +function TestSRTWriter.make_transcript() + local t = Transcript.new() + t:add_segment(TranscriptSegment.new { + data = {start = 0, ['end'] = 1, text = 'hello'}, + item = {}, + take = {} + }) + t:add_segment(TranscriptSegment.new { + data = {start = 1, ['end'] = 2, text = 'world'}, + item = {}, + take = {} + }) + t:update() + return t +end + +function TestSRTWriter:testFormatTime() + lu.assertEquals(SRTWriter.format_time(0), '00:00:00,000') + lu.assertEquals(SRTWriter.format_time(1), '00:00:01,000') + lu.assertEquals(SRTWriter.format_time(1.5), '00:00:01,500') + lu.assertEquals(SRTWriter.format_time(60), '00:01:00,000') + lu.assertEquals(SRTWriter.format_time(60.5), '00:01:00,500') + lu.assertEquals(SRTWriter.format_time(3600), '01:00:00,000') + lu.assertEquals(SRTWriter.format_time(3600.5), '01:00:00,500') +end + +function TestSRTWriter:testInit() + local f = {} + local writer = SRTWriter.new { file = f } +end + +function TestSRTWriter:testInitNoFile() + lu.assertErrorMsgContains('missing file', SRTWriter.new) +end + +function TestSRTWriter:testWrite() + local t = TestSRTWriter.make_transcript() + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = SRTWriter.new { file = f } + writer:write(t) + local output_str = table.concat(output) + lu.assertEquals(output_str, '1\n00:00:00,000 --> 00:00:01,000\nhello\n\n2\n00:00:01,000 --> 00:00:02,000\nworld\n\n') +end + +function TestSRTWriter:testXYCoordinates() + local t = TestSRTWriter.make_transcript() + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = SRTWriter.new { + file = f, + options = { + coords_x1 = '1', + coords_y1 = '2', + coords_x2 = '3', + coords_y2 = '4' + } + } + writer:write(t) + local output_str = table.concat(output) + lu.assertEquals(output_str, '1\n00:00:00,000 --> 00:00:01,000 X1:1 X2:3 Y1:2 Y2:4\nhello\n\n2\n00:00:01,000 --> 00:00:02,000 X1:1 X2:3 Y1:2 Y2:4\nworld\n\n') +end + +function TestSRTWriter:testWriteSegment() + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = SRTWriter.new { file = f } + local segment = { + get = function (self, key) + if key == 'start' then + return 0 + elseif key == 'end' then + return 1 + elseif key == 'text' then + return 'hello' + end + end + } + writer:write_segment(segment, 1) + local output_str = table.concat(output) + lu.assertEquals(output_str, '1\n00:00:00,000 --> 00:00:01,000\nhello\n\n') +end + +function TestSRTWriter:testWriteLine() + local output = {} + local f = { + write = function (self, s) + table.insert(output, s) + end + } + local writer = SRTWriter.new { file = f } + writer:write_line('hello', 1, 0, 1) + local output_str = table.concat(output) + lu.assertEquals(output_str, '1\n00:00:00,000 --> 00:00:01,000\nhello\n\n') +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/ReaSpeech/tests/TestTranscript.lua b/reascripts/ReaSpeech/tests/TestTranscript.lua new file mode 100644 index 0000000000000000000000000000000000000000..c8dc4a9b78572a5c69957a9d2efffa9787a27222 --- /dev/null +++ b/reascripts/ReaSpeech/tests/TestTranscript.lua @@ -0,0 +1,453 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +app = {} + +local lu = require('luaunit') + +require('json') +require('mock_reaper') +require('Polo') +require('source/Transcript') + +-- + +reaper.GetCursorPosition = function () return 0 end +reaper.GetMediaItemInfo_Value = function (_, _) return 0 end +reaper.GetMediaItemTakeInfo_Value = function (_, _) return 0 end +reaper.GetMediaItemTake_Source = function () return {fileName = "test_audio.wav"} end +reaper.GetMediaSourceFileName = function (source) return source.fileName end +reaper.GetSelectedMediaItem = function (_, _) return {} end +reaper.GetSetMediaTrackInfo_String = function (_, _, _, _) end +reaper.GetTrack = function (_, _) return {} end +reaper.InsertTrackAtIndex = function (_, _) end +reaper.Main_OnCommand = function (_, _) end +reaper.SelectAllMediaItems = function (_, _) end +reaper.SetEditCurPos = function (_, _, _) end +reaper.SetMediaItemLength = function (_, _, _) end +reaper.SetMediaItemPosition = function (_, _, _) end +reaper.SetOnlyTrackSelected = function (_) end + +reaper.GetItemStateChunk = function (item, str, isundo) + return true, [[]] +end + +TestTranscript = { + segment = function (data) + local words = data.words + data.words = nil + data['end'] = data.end_ + data.end_ = nil + return TranscriptSegment.new { + data = data, + item = {}, + take = {}, + words = words + } + end, + + word = TranscriptWord.new +} + +function TestTranscript:setUp() + function app:trap(f) return xpcall(f, function(e) print(tostring(e)) end) end + + reaper.__test_setUp() + + self.markers = {} + reaper.AddProjectMarker2 = function (proj, isrgn, pos, rgnend, name, wantidx, color) + table.insert(self.markers, { + proj = proj, + isrgn = isrgn, + pos = pos, + rgnend = rgnend, + name = name, + wantidx = wantidx, + color = color + }) + end + + self.item_state_chunk = "" + reaper.SetItemStateChunk = function (item, str, isundo) + self.item_state_chunk = str + end +end + +function TestTranscript:make_transcript() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test 1", + words = { + self.word { word = "test", start = 1.0, end_ = 1.5, probability = 1.0 }, + self.word { word = "1", start = 1.5, end_ = 2.0, probability = 0.5 } + } + }) + t:add_segment(self.segment { + id = 2, + start = 2.0, + end_ = 3.0, + text = "test 2", + words = { + self.word { word = "test", start = 2.0, end_ = 2.5, probability = 1.0 }, + self.word { word = "2", start = 2.5, end_ = 3.0, probability = 0.5 } + } + }) + t:update() + return t +end + +function TestTranscript:testInit() + local t = Transcript.new() + lu.assertEquals(t.init_data, {}) + lu.assertEquals(t.filtered_data, {}) + lu.assertEquals(t.data, {}) + lu.assertEquals(t.search, '') +end + +function TestTranscript:testClear() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test" + }) + t.search = 'test' + t:update() + lu.assertEquals(t:has_segments(), true) + lu.assertEquals(#t.init_data, 1) + lu.assertEquals(#t.filtered_data, 1) + lu.assertEquals(#t.data, 1) + t:clear() + lu.assertEquals(t:has_segments(), false) + lu.assertEquals(#t.init_data, 0) + lu.assertEquals(#t.filtered_data, 0) + lu.assertEquals(#t.data, 0) + lu.assertEquals(t.search, '') +end + +function TestTranscript:testColumnOrder() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test", + avg_logprob = 0.5 + }) + local columns = t:get_columns() + lu.assertEquals(columns, {"id", "start", "end", "text", "score", "file", "avg_logprob"}) +end + +function TestTranscript:testFileColumn() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test" + }) + t:update() + local segments = t:get_segments() + lu.assertEquals(segments[1]:get_file(), "test_audio") + lu.assertEquals(segments[1]:get('file'), "test_audio") +end + +function TestTranscript:testSearch() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test 1" + }) + t:add_segment(self.segment { + id = 2, + start = 2.0, + ['end'] = 3.0, + text = "test 2" + }) + t.search = 'test 2' + t:update() + lu.assertEquals(#t.init_data, 2) + lu.assertEquals(#t.filtered_data, 1) + lu.assertEquals(#t.data, 1) + local segments = t:get_segments() + lu.assertEquals(segments[1]:get('id'), 2) +end + +function TestTranscript:testSort() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test 1" + }) + t:add_segment(self.segment { + id = 2, + start = 2.0, + end_ = 3.0, + text = "test 2" + }) + t:update() + t:sort('id', true) + local segments = t:get_segments() + lu.assertEquals(segments[1]:get('id'), 1) + lu.assertEquals(segments[2]:get('id'), 2) + t:sort('id', false) + segments = t:get_segments() + lu.assertEquals(segments[1]:get('id'), 2) + lu.assertEquals(segments[2]:get('id'), 1) +end + +function TestTranscript:testDefaultHide() + lu.assertFalse(TranscriptSegment.default_hide('id')) + lu.assertTrue(TranscriptSegment.default_hide('seek')) +end + +function TestTranscript:testSegmentScore() + local s = self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test won", + words = { + self.word { word = "test", start = 1.0, end_ = 1.5, probability = 1.0 }, + self.word { word = "won", start = 1.5, end_ = 2.0, probability = 0.5 } + } + } + lu.assertAlmostEquals(s:score(), 0.75, 0.01) + lu.assertAlmostEquals(s:get('score'), 0.75, 0.01) +end + +function TestTranscript:testSetWords() + local s = self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "the fox", + words = { + self.word { word = "the", start = 1.0, end_ = 1.5, probability = 1.0 }, + self.word { word = "fox", start = 1.5, end_ = 2.0, probability = 0.5 } + } + } + lu.assertEquals(s.words[2].word, "fox") + lu.assertEquals(s:get('text'), "the fox") + lu.assertAlmostEquals(s:get('score'), 0.75, 0.01) + s:set_words({ + self.word { word = "the", start = 1.0, end_ = 1.25, probability = 1.0 }, + self.word { word = "quick", start = 1.25, end_ = 1.5, probability = 1.0 }, + self.word { word = "brown", start = 1.5, end_ = 1.75, probability = 1.0 }, + self.word { word = "fox", start = 1.75, end_ = 2.0, probability = 1.0 } + }) + lu.assertEquals(s.words[2].word, "quick") + lu.assertEquals(s:get('text'), "the quick brown fox") + lu.assertAlmostEquals(s:get('score'), 1.0, 0.01) +end + +function TestTranscript:testMergeWords() + local words = { + self.word { word = "rene", start = 1.0, end_ = 1.5, probability = 1.0 }, + self.word { word = "gade", start = 1.5, end_ = 2.0, probability = 0.5 } + } + TranscriptSegment.merge_words(words, 1, 2) + lu.assertEquals(#words, 1) + lu.assertEquals(words[1].word, "renegade") + lu.assertEquals(words[1].start, 1.0) + lu.assertEquals(words[1].end_, 2.0) + lu.assertAlmostEquals(words[1].probability, 0.75, 0.01) +end + +function TestTranscript:testSplitWords() + local words = { + self.word { word = "renegade", start = 1.0, end_ = 2.0, probability = 1.0 } + } + TranscriptSegment.split_word(words, 1) + lu.assertEquals(#words, 2) + lu.assertEquals(words[1].word, "rene") + lu.assertEquals(words[1].start, 1.0) + lu.assertEquals(words[1].end_, 1.5) + lu.assertAlmostEquals(words[1].probability, 1.0, 0.01) + lu.assertEquals(words[2].word, "gade") + lu.assertEquals(words[2].start, 1.5) + lu.assertEquals(words[2].end_, 2.0) + lu.assertAlmostEquals(words[2].probability, 1.0, 0.01) +end + +function TestTranscript:testCreateMarkers() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test 1" + }) + t:add_segment(self.segment { + id = 2, + start = 2.0, + end_ = 3.0, + text = "test 2" + }) + t:update() + t:create_markers(0, false, false) + lu.assertEquals(#self.markers, 2) + lu.assertEquals(self.markers[1].proj, 0) + lu.assertEquals(self.markers[1].isrgn, false) + lu.assertEquals(self.markers[1].pos, 1.0) + lu.assertEquals(self.markers[1].rgnend, 2.0) + lu.assertEquals(self.markers[1].name, "test 1") + lu.assertEquals(self.markers[1].wantidx, 1) + lu.assertEquals(self.markers[2].proj, 0) + lu.assertEquals(self.markers[2].isrgn, false) + lu.assertEquals(self.markers[2].pos, 2.0) + lu.assertEquals(self.markers[2].rgnend, 3.0) + lu.assertEquals(self.markers[2].name, "test 2") + lu.assertEquals(self.markers[2].wantidx, 2) +end + +function TestTranscript:testCreateRegions() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test 1" + }) + t:add_segment(self.segment { + id = 2, + start = 2.0, + end_ = 3.0, + text = "test 2" + }) + t:update() + t:create_markers(0, true, false) + lu.assertEquals(#self.markers, 2) + lu.assertEquals(self.markers[1].proj, 0) + lu.assertEquals(self.markers[1].isrgn, true) + lu.assertEquals(self.markers[1].pos, 1.0) + lu.assertEquals(self.markers[1].rgnend, 2.0) + lu.assertEquals(self.markers[1].name, "test 1") + lu.assertEquals(self.markers[1].wantidx, 1) + lu.assertEquals(self.markers[2].proj, 0) + lu.assertEquals(self.markers[2].isrgn, true) + lu.assertEquals(self.markers[2].pos, 2.0) + lu.assertEquals(self.markers[2].rgnend, 3.0) + lu.assertEquals(self.markers[2].name, "test 2") + lu.assertEquals(self.markers[2].wantidx, 2) +end + +function TestTranscript:testCreateNotesTrack() + local t = Transcript.new() + t:add_segment(self.segment { + id = 1, + start = 1.0, + end_ = 2.0, + text = "test 1%", + words = { + self.word { word = "test", start = 1.0, end_ = 1.5, probability = 1.0 }, + self.word { word = "1%", start = 1.5, end_ = 2.0, probability = 0.5 } + } + }) + t:update() + t:create_notes_track(false) + lu.assertStrContains(self.item_state_chunk, [[]]) + lu.assertStrContains(self.item_state_chunk, 'IMGRESOURCEFLAGS 11') + t:create_notes_track(true) + lu.assertStrContains(self.item_state_chunk, [[]]) + lu.assertNotStrContains(self.item_state_chunk, 'IMGRESOURCEFLAGS 11') +end + +function TestTranscript:testToJson() + local t = TestTranscript:make_transcript() + local result = t:to_json() + local parsed = json.decode(result) + lu.assertEquals(parsed.segments[1].id, 1) + lu.assertEquals(parsed.segments[1].start, 1.0) + lu.assertEquals(parsed.segments[1]['end'], 2.0) + lu.assertEquals(parsed.segments[1].text, "test 1") + lu.assertEquals(parsed.segments[1].words[1].word, "test") + lu.assertEquals(parsed.segments[1].words[1].start, 1.0) + lu.assertEquals(parsed.segments[1].words[1]['end'], 1.5) + lu.assertEquals(parsed.segments[1].words[1].probability, 1.0) + lu.assertEquals(parsed.segments[1].words[2].word, "1") + lu.assertEquals(parsed.segments[1].words[2].start, 1.5) + lu.assertEquals(parsed.segments[1].words[2]['end'], 2.0) + lu.assertEquals(parsed.segments[1].words[2].probability, 0.5) + lu.assertEquals(parsed.segments[2].id, 2) + lu.assertEquals(parsed.segments[2].start, 2.0) + lu.assertEquals(parsed.segments[2]['end'], 3.0) + lu.assertEquals(parsed.segments[2].text, "test 2") + lu.assertEquals(parsed.segments[2].words[1].word, "test") + lu.assertEquals(parsed.segments[2].words[1].start, 2.0) + lu.assertEquals(parsed.segments[2].words[1]['end'], 2.5) + lu.assertEquals(parsed.segments[2].words[1].probability, 1.0) + lu.assertEquals(parsed.segments[2].words[2].word, "2") + lu.assertEquals(parsed.segments[2].words[2].start, 2.5) + lu.assertEquals(parsed.segments[2].words[2]['end'], 3.0) + lu.assertEquals(parsed.segments[2].words[2].probability, 0.5) + local keys = {} + for k, _ in pairs(parsed.segments[1]) do + table.insert(keys, k) + end + table.sort(keys) + lu.assertEquals(keys, {"end", "file", "id", "start", "text", "words"}) + keys = {} + for k, _ in pairs(parsed.segments[1].words[1]) do + table.insert(keys, k) + end + table.sort(keys) + lu.assertEquals(keys, {"end", "probability", "start", "word"}) +end + +function TestTranscript:testSegmentToJson() + local t = TestTranscript:make_transcript() + local result = t:get_segments()[1]:to_json() + local parsed = json.decode(result) + lu.assertEquals(parsed.id, 1) + lu.assertEquals(parsed.start, 1.0) + lu.assertEquals(parsed['end'], 2.0) + lu.assertEquals(parsed.text, "test 1") + lu.assertEquals(parsed.words[1].word, "test") + lu.assertEquals(parsed.words[1].start, 1.0) + lu.assertEquals(parsed.words[1]['end'], 1.5) + lu.assertEquals(parsed.words[1].probability, 1.0) + lu.assertEquals(parsed.words[2].word, "1") + lu.assertEquals(parsed.words[2].start, 1.5) + lu.assertEquals(parsed.words[2]['end'], 2.0) + lu.assertEquals(parsed.words[2].probability, 0.5) + local keys = {} + for k, _ in pairs(parsed) do + table.insert(keys, k) + end + table.sort(keys) + lu.assertEquals(keys, {"end", "file", "id", "start", "text", "words"}) + keys = {} + for k, _ in pairs(parsed.words[1]) do + table.insert(keys, k) + end + table.sort(keys) + lu.assertEquals(keys, {"end", "probability", "start", "word"}) +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/ReaSpeech/version.lua b/reascripts/ReaSpeech/version.lua new file mode 100644 index 0000000000000000000000000000000000000000..ceb16c1366e7b8afafcbfac5c001fe67e0f7f575 --- /dev/null +++ b/reascripts/ReaSpeech/version.lua @@ -0,0 +1 @@ +ReaSpeechUI.VERSION = "0.1.0" diff --git a/reascripts/common/.luacheckrc b/reascripts/common/.luacheckrc new file mode 100644 index 0000000000000000000000000000000000000000..2e6a75e7030699711be3b18950c1a7568847bb1a --- /dev/null +++ b/reascripts/common/.luacheckrc @@ -0,0 +1,6 @@ +globals = { +} + +allow_defined = true +color = false +ignore = {"212/self", "212/_.*", "432/self", "631"} diff --git a/reascripts/common/Makefile b/reascripts/common/Makefile new file mode 100644 index 0000000000000000000000000000000000000000..61075e36783cef473441104746c97a1f00381271 --- /dev/null +++ b/reascripts/common/Makefile @@ -0,0 +1,38 @@ +LUA53=lua5.3 +LUAC53=luac5.3 +LUA54=lua5.4 +LUAC54=luac5.4 +LUACHECK=luacheck +GIT=git + +LOKASENNAGUI_REPO=https://github.com/TeamAudio/lokasenna-gui.git +LOKASENNAGUI_BRANCH=develop + +source:=$(wildcard libs/*.lua tests/*.lua) +tests:=$(wildcard tests/Test*.lua) + +all: lint test + +lint: $(source) + $(LUACHECK) $? + +test: $(source) $(tests) + true $(foreach test, $(tests), && $(LUA53) $(test) -v) + true $(foreach test, $(tests), && $(LUA54) $(test) -v) + +.PHONY: build/lokasenna-gui +build/lokasenna-gui: + rm -rf build/lokasenna-gui + $(GIT) clone '$(LOKASENNAGUI_REPO)' build/lokasenna-gui + pushd build/lokasenna-gui; $(GIT) checkout '$(LOKASENNAGUI_BRANCH)'; popd + +.PHONY: build/lokasenna-gui.lua +build/lokasenna-gui.lua: build/lokasenna-gui + echo "GUI = {}\r" > build/lokasenna-gui.lua + awk '/-- Error handling --/{p++;if(p==1){next}}p' 'build/lokasenna-gui/Lokasenna_GUI v2/Library/Core.lua' >> build/lokasenna-gui.lua + find 'build/lokasenna-gui/Lokasenna_GUI v2/Library/Classes' -name '*.lua' -exec cat {} + >> build/lokasenna-gui.lua + +.PHONY: vendor/lokasenna-gui.lua +vendor/lokasenna-gui.lua: build/lokasenna-gui.lua + mkdir -p vendor + cp -v build/lokasenna-gui.lua vendor/lokasenna-gui.lua \ No newline at end of file diff --git a/reascripts/common/libs/ImGuiTheme.lua b/reascripts/common/libs/ImGuiTheme.lua new file mode 100644 index 0000000000000000000000000000000000000000..bf64501c569bb785ccacabd3327c88008edc7279 --- /dev/null +++ b/reascripts/common/libs/ImGuiTheme.lua @@ -0,0 +1,83 @@ +--[[ + +ImGuiTheme.lua - Simple theming for the ReaImGui API + +Initialize a new ImGuiTheme via `new` with a table containing (both optional): + +- `colors`->(list table of { `colorKeyValueOrFunction`, `colorValue` }) +- `styles`->(list table of { `styleKeyValueOrFunction`, `styleValue1` [, ...`styleValueN`] }) + +The Returned object contains the following methods to be used where you would have originally +called ImGui functions directly: + +- `push(ctx)`: calls `ImGui_PushStyleColor` and `ImGui_PushStyleVar` for each color and style defined +- `pop(ctx)`: calls `ImGui_PopStyleColor` and `ImGui_PopStyleVar` with the correct `count` argument +]]-- + +ImGuiTheme = { + -- Override the calls to the associated REAPER API methods here + -- ...if you want to + f_color_push = nil, + f_color_pop = nil, + f_style_push = nil, + f_style_pop = nil, +} + +ImGuiTheme.__index = ImGuiTheme +ImGuiTheme.new = function(theme_definition) + local theme = { + colors = ImGuiTheme.get_attribute_values(theme_definition.colors), + styles = ImGuiTheme.get_attribute_values(theme_definition.styles), + } + + theme.color_count = #theme.colors + theme.style_count = #theme.styles + + setmetatable(theme, ImGuiTheme) + + theme:init() + + return theme +end + +function ImGuiTheme:init() + self.f_color_push = ImGuiTheme.get_function('f_color_push', reaper.ImGui_PushStyleColor) + self.f_color_pop = ImGuiTheme.get_function('f_color_pop', reaper.ImGui_PopStyleColor) + self.f_style_push = ImGuiTheme.get_function('f_style_push', reaper.ImGui_PushStyleVar) + self.f_style_pop = ImGuiTheme.get_function('f_style_pop', reaper.ImGui_PopStyleVar) +end + +ImGuiTheme.get_attribute_values = function(raw_attributes) + raw_attributes = raw_attributes or {} + + local attrs = {} + + for _, v in ipairs(raw_attributes) do + if (type(v[1]) == "function") then + v[1] = v[1]() + end + + table.insert(attrs, v) + end + + return attrs +end + +ImGuiTheme.get_function = function(key, default) + return ImGuiTheme[key] or default +end + +function ImGuiTheme:push(ctx) + for i = 1, self.color_count do + self.f_color_push(ctx, self.colors[i][1], table.unpack(self.colors[i], 2)) + end + + for i = 1, self.style_count do + self.f_style_push(ctx, self.styles[i][1], table.unpack(self.styles[i], 2)) + end +end + +function ImGuiTheme:pop(ctx) + self.f_color_pop(ctx, self.color_count) + self.f_style_pop(ctx, self.style_count) +end diff --git a/reascripts/common/libs/OptionsConfig.lua b/reascripts/common/libs/OptionsConfig.lua new file mode 100644 index 0000000000000000000000000000000000000000..52c9197d6e42e226f10acdef84f7554145162d5c --- /dev/null +++ b/reascripts/common/libs/OptionsConfig.lua @@ -0,0 +1,98 @@ +--[[ + + OptionsConfig.lua - General options configuration data model + +]]-- + +-- Constructor +-- +-- Parameters: +-- section: Section to write in reaper-extstate.ini +-- options: Mapping from option names to {type, default} pairs +-- type can be one of: 'string', 'number', 'boolean' +-- +-- Example: +-- local options = OptionsConfig:new { +-- section = "ReaSpeech.Options", +-- options = { +-- patties_per_burger = {'number', 2}, +-- ... +-- } +-- } +-- + +OptionsConfig = {} +OptionsConfig.__index = OptionsConfig + +function OptionsConfig:new(o) + o = o or {} + setmetatable(o, self) + assert(o.section, 'section is required') + o.options = o.options or {} + return o +end + +function OptionsConfig:get(name) + local option = self.options[name] + assert(option, 'undefined option ' .. name) + + local option_type, option_default = table.unpack(option) + + if self:exists(name) then + local str = reaper.GetExtState(self.section, name) + + if option_type == 'number' then + return self:_string_to_number(str) + elseif option_type == 'boolean' then + return self:_string_to_boolean(str) + else + return str + end + else + return option_default + end +end + +function OptionsConfig:set(name, value) + local option = self.options[name] + assert(option, 'undefined option ' .. name) + + local option_type, _ = table.unpack(option) + + local str + if option_type == 'number' then + str = self:_number_to_string(value) + elseif option_type == 'boolean' then + str = self:_boolean_to_string(value) + else + str = tostring(value) + end + + reaper.SetExtState(self.section, name, str, true) +end + +function OptionsConfig:delete(name) + assert(self.options[name], 'undefined option ' .. name) + reaper.DeleteExtState(self.section, name, true) +end + +function OptionsConfig:exists(name) + assert(self.options[name], 'undefined option ' .. name) + return reaper.HasExtState(self.section, name) +end + +function OptionsConfig:_string_to_number(str) + return tonumber(str) or 0 +end + +function OptionsConfig:_string_to_boolean(str) + return str == 'true' +end + +function OptionsConfig:_number_to_string(num) + return tostring(tonumber(num) or 0) +end + +function OptionsConfig:_boolean_to_string(bool) + return bool and 'true' or 'false' +end diff --git a/reascripts/common/libs/Polo.lua b/reascripts/common/libs/Polo.lua new file mode 100644 index 0000000000000000000000000000000000000000..72ef0179f07e1be6188629e34e56a9d09366d2f6 --- /dev/null +++ b/reascripts/common/libs/Polo.lua @@ -0,0 +1,23 @@ +--[[ + + Polo.lua - Plain Old Lua Object (POLO) class generator + +]]-- + +function Polo(definition) + definition.__index = definition + + local new_override = definition.new + + definition.new = function(...) + local o = new_override and new_override(...) or ... or {} + assert(type(o) == 'table') + + setmetatable(o, definition) + if o.init then o:init() end + + return o + end + + return definition +end diff --git a/reascripts/common/libs/ReaIter.lua b/reascripts/common/libs/ReaIter.lua new file mode 100644 index 0000000000000000000000000000000000000000..31337e647203d37d43a802820d923f905f49c5cb --- /dev/null +++ b/reascripts/common/libs/ReaIter.lua @@ -0,0 +1,34 @@ +--[[ + + ReaIter.lua - Reaper iterators + +]]-- + +ReaIter = {} + +ReaIter._make_iterator = function(count_f, item_f) + return function(proj) + proj = proj or 0 + local i = 0 + local n = count_f(proj) + return function () + if i < n then + local item = item_f(proj, i) + i = i + 1 + return item + end + end + end +end + +ReaIter.each_media_item = ReaIter._make_iterator(reaper.CountMediaItems, reaper.GetMediaItem) + +ReaIter.each_selected_media_item = ReaIter._make_iterator(reaper.CountSelectedMediaItems, reaper.GetSelectedMediaItem) + +ReaIter.each_selected_track = ReaIter._make_iterator(reaper.CountSelectedTracks, reaper.GetSelectedTrack) + +ReaIter.each_track = ReaIter._make_iterator(reaper.CountTracks, reaper.GetTrack) + +ReaIter.each_take = ReaIter._make_iterator(reaper.CountTakes, reaper.GetTake) + +ReaIter.each_track_item = ReaIter._make_iterator(reaper.CountTrackMediaItems, reaper.GetTrackMediaItem) diff --git a/reascripts/common/libs/ReaUtil.lua b/reascripts/common/libs/ReaUtil.lua new file mode 100644 index 0000000000000000000000000000000000000000..c12ab2154785251603e44aca335a9086330b94f0 --- /dev/null +++ b/reascripts/common/libs/ReaUtil.lua @@ -0,0 +1,37 @@ +--[[ + + ReaUtil.lua - Utility functions for Reaper Interaction + +]]-- + +ReaUtil = { + ACTIVE_PROJECT = 0 +} + +function ReaUtil.proxy_main_on_command(command_number, flag) + return function (proj) + proj = proj or 0 + reaper.Main_OnCommandEx(command_number, flag, proj) + end +end + +function ReaUtil.disabler(context, error_handler) + error_handler = error_handler or function(msg) + reaper.ShowConsoleMsg(msg .. '\n') + end + + return function(predicate, f) + local safe_f = function() + xpcall(f, error_handler) + end + + if not predicate then + safe_f() + return + end + + reaper.ImGui_BeginDisabled(context, true) + safe_f() + reaper.ImGui_EndDisabled(context) + end +end \ No newline at end of file diff --git a/reascripts/common/libs/Tempfile.lua b/reascripts/common/libs/Tempfile.lua new file mode 100644 index 0000000000000000000000000000000000000000..af652a385da2378855747b85873626a50a6164af --- /dev/null +++ b/reascripts/common/libs/Tempfile.lua @@ -0,0 +1,34 @@ +--[[ + + Tempfile.lua - Temporary filename creator + +]]-- + +Tempfile = { + _names = {} +} + +function Tempfile:name() + if reaper.GetOS():find("Win") then + return self:_add_name(os.getenv("TEMP") .. os.tmpname()) + else + return self:_add_name(os.tmpname()) + end +end + +function Tempfile:remove(name) + if os.remove(name) then + self._names[name] = nil + end +end + +function Tempfile:remove_all() + for name, _ in pairs(self._names) do + os.remove(name) + end +end + +function Tempfile:_add_name(name) + self._names[name] = true + return name +end diff --git a/reascripts/common/libs/mock_reaper.lua b/reascripts/common/libs/mock_reaper.lua new file mode 100644 index 0000000000000000000000000000000000000000..5cb27f94b40215ef3aaafcde3ba4929ed27745a6 --- /dev/null +++ b/reascripts/common/libs/mock_reaper.lua @@ -0,0 +1,109 @@ +reaper = reaper or { + __ext_state__ = {}, + + __test_setUp = function () + reaper.__ext_state__ = {} + end, + + APIExists = function (_) + return false + end, + + ColorToNative = function (r, g, b) + return (math.tointeger(r) or 0) + ((math.tointeger(g) or 0) << 8) + ((math.tointeger(b) or 0) << 16) + end, + + GetExtState = function (section, key) + if reaper.__ext_state__[section] then + return reaper.__ext_state__[section][key] + end + end, + + HasExtState = function (section, key) + return reaper.GetExtState(section, key) ~= nil + end, + + SetExtState = function (section, key, value) + if not reaper.__ext_state__[section] then + reaper.__ext_state__[section] = {} + end + reaper.__ext_state__[section][key] = value + end, + + DeleteExtState = function (section, key) + if reaper.__ext_state__[section] then + reaper.__ext_state__[section][key] = nil + end + end, + + GetOS = function () + return 'Win64' + end, + + GetAppVersion = function () + return '7.07/x64' + end, + + GetResourcePath = function () + return 'tests/resources' + end, + + MB = print, + ShowConsoleMsg = print, + ShowMessageBox = print, + + ImGui_PushStyleColor = function (_context, _key, _value) end, + ImGui_PopStyleColor = function (_context, _count) end, + + ImGui_PushStyleVar = function (_context, _key, _varlength) end, + ImGui_PopStyleVar = function (_context, _count) end, + + defer = function (f) + f() + end, + + get_action_context = function () + local path = debug.getinfo(2, "S").source:sub(2) + return false, path + end, + + ImGui_BeginDisabled = function(_context, _disabled) end, + ImGui_EndDisabled = function(_context) end, +} + +if reaper.__test_setUp then reaper.__test_setUp() end + +gfx = gfx or { + init = function (name, w, h, dock, x, y) + gfx.__name = name + gfx.w = w + gfx.h = h + gfx.__dock = dock + gfx.x = x + gfx.y = y + end, + + clear = function (_) end, + + clienttoscreen = function (_, _) + return 0, 0 + end, + + dock = function (_, _, _, _, _) + return 0, 0, 0, 0, 0 + end, + + measurestr = function (_) + return 0, 0 + end, + + circle = function (_, _, _, _, _) end, + drawstr = function (_) end, + muladdrect = function (_, _, _, _, _, _, _, _, _, _, _, _) end, + rect = function (_, _, _, _) end, + roundrect = function (_, _, _, _, _, _, _) end, + set = function (_, _, _, _) end, + setfont = function (_, _, _, _) end, + setimgdim = function (_, _, _) end, + quit = function () end, +} diff --git a/reascripts/common/scripts/png_to_lua.rb b/reascripts/common/scripts/png_to_lua.rb new file mode 100644 index 0000000000000000000000000000000000000000..1cbc954a4c73c80f66d580805137c7ce2dc9de9c --- /dev/null +++ b/reascripts/common/scripts/png_to_lua.rb @@ -0,0 +1,51 @@ +# png_to_lua.rb - Convert a PNG image to Lua source + +require 'chunky_png' + +if ARGV.length < 2 + puts "Usage: #{$0} " + exit 1 +end + +input_filename = ARGV[0] +output_filename = ARGV[1] + +image = ChunkyPNG::Image.from_file(input_filename) +image_name = File.basename(input_filename, File.extname(input_filename)) + +File.open(output_filename, 'w') do |f| + f.write("IMAGES = IMAGES or {}\n") + f.write("IMAGES['#{image_name}'] = {\n") + f.write(" width = #{image.width},\n") + f.write(" height = #{image.height},\n") + + f.write(" pixels = {\n") + (0...image.height).each do |y| + row = '{' + x = 0 + while x < image.width do + length = 1 + rgba = ChunkyPNG::Color.to_truecolor_alpha_bytes(image[x, y]) + while x + length < image.width do + if rgba == ChunkyPNG::Color.to_truecolor_alpha_bytes(image[x + length, y]) then + length += 1 + else + break + end + end + row << "#{length},#{rgba[0]},#{rgba[1]},#{rgba[2]},#{rgba[3]}," + x += length + end + row << '}' + f.write(" #{row},\n") + end + f.write(" },\n") + + f.write(" bytes = table.concat({\n") + image.to_blob.bytes.each_slice(100) do |slice| + f.write("string.char(#{slice.join(",")}),\n") + end + f.write(" })\n") + + f.write("}\n") +end diff --git a/reascripts/common/tests/TestImGuiTheme.lua b/reascripts/common/tests/TestImGuiTheme.lua new file mode 100644 index 0000000000000000000000000000000000000000..718a50368ff51e266cf76d8c23e6724f08f42a6d --- /dev/null +++ b/reascripts/common/tests/TestImGuiTheme.lua @@ -0,0 +1,125 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +local lu = require('luaunit') + +require('mock_reaper') +require('ImGuiTheme') + +-- + +TestImGuiTheme = { + overrides = { + f_color_push = function() end, + f_color_pop = function() end, + f_style_push = function() end, + f_style_pop = function() end + } +} + +function TestImGuiTheme:setUp() + self.overrides.f_color_push = ImGuiTheme.f_color_push + self.overrides.f_color_pop = ImGuiTheme.f_color_pop + self.overrides.f_style_push = ImGuiTheme.f_style_push + self.overrides.f_style_pop = ImGuiTheme.f_style_pop +end + +function TestImGuiTheme:tearDown() + ImGuiTheme.f_color_push = self.overrides.f_color_push + ImGuiTheme.f_color_pop = self.overrides.f_color_pop + ImGuiTheme.f_style_push = self.overrides.f_style_push + ImGuiTheme.f_style_pop = self.overrides.f_style_pop +end + +function TestImGuiTheme:testColorInit() + local theme = ImGuiTheme.new { + colors = { + { function() return "some key" end, 0xFF0000FF }, + { "just a key", 0x00FF0000 } + } + } + + lu.assertEquals(theme.colors[1][1], "some key") + lu.assertEquals(theme.colors[1][2], 0xFF0000FF) + lu.assertEquals(theme.colors[2][1], "just a key") + lu.assertEquals(theme.colors[2][2], 0x00FF0000) + lu.assertEquals(theme.color_count, 2) +end + +function TestImGuiTheme:testColors() + local theme = ImGuiTheme.new { + colors = { + { function() return "some key" end, 0xFF0000FF }, + { "just a key", 0x00FF0000 } + } + } + + local expectations = { + { "some key", 0xFF0000FF }, + { "just a key", 0x00FF0000 } + } + + local i = 1 + + ImGuiTheme.f_color_push = function(_ctx, key, value) + lu.assertEquals(key, expectations[i][1]) + lu.assertEquals(value, expectations[i][2]) + i = i + 1 + end + + ImGuiTheme.f_color_pop = function(_ctx, count) + lu.assertEquals(count, 2) + end + + theme:push("context") + theme:pop("context") +end + +function TestImGuiTheme:testStyleInit() + local theme = ImGuiTheme.new { + styles = { + { function() return "single argument" end, 1.0 }, + { "multiple arguments", 2.0, 3.0 } + } + } + + lu.assertEquals(theme.styles[1][1], "single argument") + lu.assertEquals(theme.styles[1][2], 1.0) + lu.assertEquals(theme.styles[2][1], "multiple arguments") + lu.assertEquals(theme.styles[2][2], 2.0) + lu.assertEquals(theme.styles[2][3], 3.0) + lu.assertEquals(theme.style_count, 2) +end + +function TestImGuiTheme:testStyles() + local theme = ImGuiTheme.new { + styles = { + { function() return "single argument" end, 1.0 }, + { "multiple arguments", 2.0, 3.0 } + } +} + + local expectations = { + { "single argument", 1.0 }, + { "multiple arguments", 2.0, 3.0 } + } + + local i = 1 + + ImGuiTheme.f_style_push = function(_ctx, key, ...) + local values = ... + lu.assertEquals(key, expectations[i][1]) + lu.assertEquals(values, table.unpack(expectations[i], 2)) + i = i + 1 + end + + ImGuiTheme.f_style_pop = function(_ctx, count) + lu.assertEquals(count, 2) + end + + theme:push("context") + theme:pop("context") +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/common/tests/TestOptionsConfig.lua b/reascripts/common/tests/TestOptionsConfig.lua new file mode 100644 index 0000000000000000000000000000000000000000..2aecb78bd053dab9dccb44d1298039895cd71c71 --- /dev/null +++ b/reascripts/common/tests/TestOptionsConfig.lua @@ -0,0 +1,78 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +local lu = require('luaunit') + +require('mock_reaper') +require('Polo') +require('OptionsConfig') + +-- + +TestOptionsConfig = { + options = OptionsConfig:new { + section = 'ReaSpeech.Options', + options = { + patties_per_burger = {'number', 2}, + num_pickles = {'number', 2}, + } + } +} + +function TestOptionsConfig:setUp() + reaper.__test_setUp() + reaper.__ext_state__[self.options.section] = {} +end + +function TestOptionsConfig:testDefault() + local num_pickles = self.options:get('num_pickles') + lu.assertEquals(num_pickles, 2) +end + +function TestOptionsConfig:testGet() + reaper.__ext_state__[self.options.section].num_pickles = '3' + local num_pickles = self.options:get('num_pickles') + lu.assertEquals(num_pickles, 3) + lu.assertEquals(reaper.__ext_state__[self.options.section].num_pickles, '3') +end + +function TestOptionsConfig:testSet() + self.options:set('num_pickles', 4) + lu.assertEquals(self.options:get('num_pickles'), 4) + lu.assertEquals(reaper.__ext_state__[self.options.section].num_pickles, '4') +end + +function TestOptionsConfig:testDelete() + self.options:set('num_pickles', 4) + self.options:delete('num_pickles') + lu.assertNil(reaper.__ext_state__[self.options.section].num_pickles) +end + +function TestOptionsConfig:testExists() + lu.assertFalse(self.options:exists('num_pickles')) + self.options:set('num_pickles', 4) + lu.assertTrue(self.options:exists('num_pickles')) + self.options:delete('num_pickles') + lu.assertFalse(self.options:exists('num_pickles')) +end + +function TestOptionsConfig:testToString() + lu.assertEquals(self.options:_number_to_string(42), '42') + lu.assertEquals(self.options:_number_to_string('x'), '0') + lu.assertEquals(self.options:_boolean_to_string(true), 'true') + lu.assertEquals(self.options:_boolean_to_string(false), 'false') + lu.assertEquals(self.options:_boolean_to_string('x'), 'true') + lu.assertEquals(self.options:_boolean_to_string(nil), 'false') +end + +function TestOptionsConfig:testFromString() + lu.assertEquals(self.options:_string_to_number('42'), 42) + lu.assertEquals(self.options:_string_to_number('x'), 0) + lu.assertTrue(self.options:_string_to_boolean('true')) + lu.assertFalse(self.options:_string_to_boolean('false')) + lu.assertFalse(self.options:_string_to_boolean('x')) + lu.assertFalse(self.options:_string_to_boolean(nil)) +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/common/tests/TestPolo.lua b/reascripts/common/tests/TestPolo.lua new file mode 100644 index 0000000000000000000000000000000000000000..9eef0a0d5bc2a8e0c1f46a852cd3b23afec172c0 --- /dev/null +++ b/reascripts/common/tests/TestPolo.lua @@ -0,0 +1,93 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +local lu = require('luaunit') + +require('Polo') + +-- + +TestPolo = {} + +function TestPolo:testClassFields() + local TestPoloClass = Polo { + field1 = "this is a field", + field2 = "this is another field", + } + + lu.assertEquals(TestPoloClass.field1, "this is a field") + lu.assertEquals(TestPoloClass.field2, "this is another field") +end + +function TestPolo:testDefaultNewMethod() + local TestPoloClass = Polo {} + + lu.assertNotIsNil(TestPoloClass.new) +end + +function TestPolo:testOverriddenNewMethod() + local TestPoloClass = Polo { + new = function(field1, field2) + return { + field1 = field1, + field2 = field2, + } + end + } + + local testPolo = TestPoloClass.new("this is a field", "this is another field") + + lu.assertEquals(testPolo.field1, "this is a field") + lu.assertIsNil(TestPoloClass.field1) + lu.assertNotEquals(TestPoloClass.field1, "this is a field") + lu.assertEquals(testPolo.field2, "this is another field") + lu.assertIsNil(TestPoloClass.field2) + lu.assertNotEquals(TestPoloClass.field2, "this is another field") +end + +function TestPolo:testOptionalInitMethod() + local TestPoloClass = Polo { + init = function(self) + self.field1 = "this is a field" + self.field2 = "this is another field" + end + } + + local testPolo = TestPoloClass.new() + + lu.assertEquals(testPolo.field1, "this is a field") + lu.assertEquals(testPolo.field2, "this is another field") +end + +function TestPolo:testInitMethodDefinedLater() + local TestPoloClass = Polo {} + + function TestPoloClass:init() + self.field1 = "this is a field" + self.field2 = "this is another field" + end + + local testPolo = TestPoloClass.new() + + lu.assertEquals(testPolo.field1, "this is a field") + lu.assertEquals(testPolo.field2, "this is another field") +end + +function TestPolo:testInstanceMethod() + local TestPoloClass = Polo { + instanceMethodDefinedUpFront = function(self) + return "inside instanceMethodDefinedUpFront" + end + } + + function TestPoloClass:instanceMethodDefinedLater() + return "inside instanceMethodDefinedLater" + end + + local testPolo = TestPoloClass.new() + lu.assertEquals(testPolo:instanceMethodDefinedUpFront(), "inside instanceMethodDefinedUpFront") + lu.assertEquals(testPolo:instanceMethodDefinedLater(), "inside instanceMethodDefinedLater") +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/common/tests/TestReaUtil.lua b/reascripts/common/tests/TestReaUtil.lua new file mode 100644 index 0000000000000000000000000000000000000000..88c0b2d6953a55bfae1301769b4d94ed0a938c1f --- /dev/null +++ b/reascripts/common/tests/TestReaUtil.lua @@ -0,0 +1,128 @@ +package.path = '../common/libs/?.lua;../common/vendor/?.lua;' .. package.path + +local lu = require('luaunit') + +require('ReaUtil') + +require('mock_reaper') + +TestReaUtil = {} + +function TestReaUtil:setUp() + reaper.__test_setUp() +end + +function TestReaUtil:testProxyMainOnCommand() + local proxy = ReaUtil.proxy_main_on_command(1, 0) + lu.assertEquals(type(proxy), "function") +end + +function TestReaUtil:testProxyMainOnCommandProjectArgument() + local proxy = ReaUtil.proxy_main_on_command(1, 0) + + reaper.Main_OnCommandEx = function(_command_number, _flag, proj) + lu.assertEquals(proj, 0) + end + proxy() + + reaper.Main_OnCommandEx = function(_command_number, _flag, proj) + lu.assertEquals(proj, 1) + end + proxy(1) +end + +function TestReaUtil:testProxyMainOnCommandCallsMainOnCommandEx() + local proxy = ReaUtil.proxy_main_on_command(1, 0) + + local main_on_command_ex_called = false + + reaper.Main_OnCommandEx = function(command_number, flag, _proj) + main_on_command_ex_called = true + lu.assertEquals(command_number, 1) + lu.assertEquals(flag, 0) + end + + proxy() + lu.assertEquals(main_on_command_ex_called, true) +end + +function TestReaUtil:testDisablerReturnsFunction() + local disabler = ReaUtil.disabler("imgui context") + lu.assertEquals(type(disabler), "function") +end + +function TestReaUtil:testDisablerDefaultErrorHandler() + local disabler = ReaUtil.disabler("imgui context") + + local default_handler_called = false + + reaper.ShowConsoleMsg = function(_msg) + default_handler_called = true + end + + disabler(true, function() error("error") end) + lu.assertEquals(default_handler_called, true) +end + +function TestReaUtil:testDisablerCustomErrorHandler() + local custom_handler_called = false + local disabler = ReaUtil.disabler("imgui context", function(msg) + custom_handler_called = true + lu.assertEquals(msg, "error") + end) + + disabler(true, function() error("error") end) + lu.assertEquals(custom_handler_called, true) +end + +function TestReaUtil:testDisablerContext() + local disabler = ReaUtil.disabler("imgui context") + + reaper.ImGui_BeginDisabled = function(context, _disabled) + lu.assertEquals(context, "imgui context") + end + + reaper.ImGui_EndDisabled = function(context) + lu.assertEquals(context, "imgui context") + end + + disabler(true, function() end) +end + +function TestReaUtil:testDisablerWrapping() + local disabler = ReaUtil.disabler("imgui context") + + local begin_marker = false + local function_called_marker = false + local end_marker = false + + reaper.ImGui_BeginDisabled = function(_context, _disabled) + begin_marker = true + end + + local f = function() + function_called_marker = true + end + + reaper.ImGui_EndDisabled = function(_context) + end_marker = true + end + + disabler(true, f) + lu.assertEquals(begin_marker, true) + lu.assertEquals(function_called_marker, true) + lu.assertEquals(end_marker, true) + + begin_marker = false + function_called_marker = false + end_marker = false + + disabler(false, f) + lu.assertEquals(begin_marker, false) + lu.assertEquals(function_called_marker, true) + lu.assertEquals(end_marker, false) +end + +-- + +os.exit(lu.LuaUnit.run()) diff --git a/reascripts/common/vendor/json.lua b/reascripts/common/vendor/json.lua new file mode 100644 index 0000000000000000000000000000000000000000..2b037159ad85d8164ff824ea1ae530a9f161abed --- /dev/null +++ b/reascripts/common/vendor/json.lua @@ -0,0 +1,392 @@ +-- +-- json.lua +-- +-- Copyright (c) 2020 rxi +-- +-- Permission is hereby granted, free of charge, to any person obtaining a copy of +-- this software and associated documentation files (the "Software"), to deal in +-- the Software without restriction, including without limitation the rights to +-- use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +-- of the Software, and to permit persons to whom the Software is furnished to do +-- so, subject to the following conditions: +-- +-- The above copyright notice and this permission notice shall be included in all +-- copies or substantial portions of the Software. +-- +-- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +-- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +-- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +-- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +-- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +-- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +-- SOFTWARE. +-- + +json = (function () + +local json = { _version = "0.1.2" } + +------------------------------------------------------------------------------- +-- Encode +------------------------------------------------------------------------------- + +local encode + +local escape_char_map = { + [ "\\" ] = "\\", + [ "\"" ] = "\"", + [ "\b" ] = "b", + [ "\f" ] = "f", + [ "\n" ] = "n", + [ "\r" ] = "r", + [ "\t" ] = "t", +} + +local escape_char_map_inv = { [ "/" ] = "/" } +for k, v in pairs(escape_char_map) do + escape_char_map_inv[v] = k +end + + +local function escape_char(c) + return "\\" .. (escape_char_map[c] or string.format("u%04x", c:byte())) +end + + +local function encode_nil(_) + return "null" +end + + +local function encode_table(val, stack) + local res = {} + stack = stack or {} + + -- Circular reference? + if stack[val] then error("circular reference") end + + stack[val] = true + + if rawget(val, 1) ~= nil or next(val) == nil then + -- Treat as array -- check keys are valid and it is not sparse + local n = 0 + for k in pairs(val) do + if type(k) ~= "number" then + error("invalid table: mixed or invalid key types") + end + n = n + 1 + end + if n ~= #val then + error("invalid table: sparse array") + end + -- Encode + for _, v in ipairs(val) do + table.insert(res, encode(v, stack)) + end + stack[val] = nil + return "[" .. table.concat(res, ",") .. "]" + + else + -- Treat as an object + for k, v in pairs(val) do + if type(k) ~= "string" then + error("invalid table: mixed or invalid key types") + end + table.insert(res, encode(k, stack) .. ":" .. encode(v, stack)) + end + stack[val] = nil + return "{" .. table.concat(res, ",") .. "}" + end +end + + +local function encode_string(val) + return '"' .. val:gsub('[%z\1-\31\\"]', escape_char) .. '"' +end + + +local function encode_number(val) + -- Check for NaN, -inf and inf + if val ~= val or val <= -math.huge or val >= math.huge then + error("unexpected number value '" .. tostring(val) .. "'") + end + return string.format("%.14g", val) +end + + +local type_func_map = { + [ "nil" ] = encode_nil, + [ "table" ] = encode_table, + [ "string" ] = encode_string, + [ "number" ] = encode_number, + [ "boolean" ] = tostring, +} + + +encode = function(val, stack) + local t = type(val) + local f = type_func_map[t] + if f then + return f(val, stack) + end + error("unexpected type '" .. t .. "'") +end + + +function json.encode(val) + return ( encode(val) ) +end + + +------------------------------------------------------------------------------- +-- Decode +------------------------------------------------------------------------------- + +local parse + +local function create_set(...) + local res = {} + for i = 1, select("#", ...) do + res[ select(i, ...) ] = true + end + return res +end + +local space_chars = create_set(" ", "\t", "\r", "\n") +local delim_chars = create_set(" ", "\t", "\r", "\n", "]", "}", ",") +local escape_chars = create_set("\\", "/", '"', "b", "f", "n", "r", "t", "u") +local literals = create_set("true", "false", "null") + +local literal_map = { + [ "true" ] = true, + [ "false" ] = false, + [ "null" ] = nil, +} + + +local function next_char(str, idx, set, negate) + for i = idx, #str do + if set[str:sub(i, i)] ~= negate then + return i + end + end + return #str + 1 +end + + +local function decode_error(str, idx, msg) + local line_count = 1 + local col_count = 1 + for i = 1, idx - 1 do + col_count = col_count + 1 + if str:sub(i, i) == "\n" then + line_count = line_count + 1 + col_count = 1 + end + end + error( string.format("%s at line %d col %d", msg, line_count, col_count) ) +end + + +local function codepoint_to_utf8(n) + -- http://scripts.sil.org/cms/scripts/page.php?site_id=nrsi&id=iws-appendixa + local f = math.floor + if n <= 0x7f then + return string.char(n) + elseif n <= 0x7ff then + return string.char(f(n / 64) + 192, n % 64 + 128) + elseif n <= 0xffff then + return string.char(f(n / 4096) + 224, f(n % 4096 / 64) + 128, n % 64 + 128) + elseif n <= 0x10ffff then + return string.char(f(n / 262144) + 240, f(n % 262144 / 4096) + 128, + f(n % 4096 / 64) + 128, n % 64 + 128) + end + error( string.format("invalid unicode codepoint '%x'", n) ) +end + + +local function parse_unicode_escape(s) + local n1 = tonumber( s:sub(1, 4), 16 ) + local n2 = tonumber( s:sub(7, 10), 16 ) + -- Surrogate pair? + if n2 then + return codepoint_to_utf8((n1 - 0xd800) * 0x400 + (n2 - 0xdc00) + 0x10000) + else + return codepoint_to_utf8(n1) + end +end + + +local function parse_string(str, i) + local res = "" + local j = i + 1 + local k = j + + while j <= #str do + local x = str:byte(j) + + if x < 32 then + decode_error(str, j, "control character in string") + + elseif x == 92 then -- `\`: Escape + res = res .. str:sub(k, j - 1) + j = j + 1 + local c = str:sub(j, j) + if c == "u" then + local hex = str:match("^[dD][89aAbB]%x%x\\u%x%x%x%x", j + 1) + or str:match("^%x%x%x%x", j + 1) + or decode_error(str, j - 1, "invalid unicode escape in string") + res = res .. parse_unicode_escape(hex) + j = j + #hex + else + if not escape_chars[c] then + decode_error(str, j - 1, "invalid escape char '" .. c .. "' in string") + end + res = res .. escape_char_map_inv[c] + end + k = j + 1 + + elseif x == 34 then -- `"`: End of string + res = res .. str:sub(k, j - 1) + return res, j + 1 + end + + j = j + 1 + end + + decode_error(str, i, "expected closing quote for string") +end + + +local function parse_number(str, i) + local x = next_char(str, i, delim_chars) + local s = str:sub(i, x - 1) + local n = tonumber(s) + if not n then + decode_error(str, i, "invalid number '" .. s .. "'") + end + return n, x +end + + +local function parse_literal(str, i) + local x = next_char(str, i, delim_chars) + local word = str:sub(i, x - 1) + if not literals[word] then + decode_error(str, i, "invalid literal '" .. word .. "'") + end + return literal_map[word], x +end + + +local function parse_array(str, i) + local res = {} + local n = 1 + i = i + 1 + while 1 do + local x + i = next_char(str, i, space_chars, true) + -- Empty / end of array? + if str:sub(i, i) == "]" then + i = i + 1 + break + end + -- Read token + x, i = parse(str, i) + res[n] = x + n = n + 1 + -- Next token + i = next_char(str, i, space_chars, true) + local chr = str:sub(i, i) + i = i + 1 + if chr == "]" then break end + if chr ~= "," then decode_error(str, i, "expected ']' or ','") end + end + return res, i +end + + +local function parse_object(str, i) + local res = {} + i = i + 1 + while 1 do + local key, val + i = next_char(str, i, space_chars, true) + -- Empty / end of object? + if str:sub(i, i) == "}" then + i = i + 1 + break + end + -- Read key + if str:sub(i, i) ~= '"' then + decode_error(str, i, "expected string for key") + end + key, i = parse(str, i) + -- Read ':' delimiter + i = next_char(str, i, space_chars, true) + if str:sub(i, i) ~= ":" then + decode_error(str, i, "expected ':' after key") + end + i = next_char(str, i + 1, space_chars, true) + -- Read value + val, i = parse(str, i) + -- Set + res[key] = val + -- Next token + i = next_char(str, i, space_chars, true) + local chr = str:sub(i, i) + i = i + 1 + if chr == "}" then break end + if chr ~= "," then decode_error(str, i, "expected '}' or ','") end + end + return res, i +end + + +local char_func_map = { + [ '"' ] = parse_string, + [ "0" ] = parse_number, + [ "1" ] = parse_number, + [ "2" ] = parse_number, + [ "3" ] = parse_number, + [ "4" ] = parse_number, + [ "5" ] = parse_number, + [ "6" ] = parse_number, + [ "7" ] = parse_number, + [ "8" ] = parse_number, + [ "9" ] = parse_number, + [ "-" ] = parse_number, + [ "t" ] = parse_literal, + [ "f" ] = parse_literal, + [ "n" ] = parse_literal, + [ "[" ] = parse_array, + [ "{" ] = parse_object, +} + + +parse = function(str, idx) + local chr = str:sub(idx, idx) + local f = char_func_map[chr] + if f then + return f(str, idx) + end + decode_error(str, idx, "unexpected character '" .. chr .. "'") +end + + +function json.decode(str) + if type(str) ~= "string" then + error("expected argument of type string, got " .. type(str)) + end + local res, idx = parse(str, next_char(str, 1, space_chars, true)) + idx = next_char(str, idx, space_chars, true) + if idx <= #str then + decode_error(str, idx, "trailing garbage") + end + return res +end + + +return json + +end)() diff --git a/reascripts/common/vendor/luaunit.lua b/reascripts/common/vendor/luaunit.lua new file mode 100644 index 0000000000000000000000000000000000000000..4741478bc7332d589a41aedc96a9da45be07e937 --- /dev/null +++ b/reascripts/common/vendor/luaunit.lua @@ -0,0 +1,3453 @@ +--[[ + luaunit.lua + +Description: A unit testing framework +Homepage: https://github.com/bluebird75/luaunit +Development by Philippe Fremy +Based on initial work of Ryu, Gwang (http://www.gpgstudy.com/gpgiki/LuaUnit) +License: BSD License, see LICENSE.txt +]]-- + +require("math") +local M={} + +-- private exported functions (for testing) +M.private = {} + +M.VERSION='3.4' +M._VERSION=M.VERSION -- For LuaUnit v2 compatibility + +-- a version which distinguish between regular Lua and LuaJit +M._LUAVERSION = (jit and jit.version) or _VERSION + +--[[ Some people like assertEquals( actual, expected ) and some people prefer +assertEquals( expected, actual ). +]]-- +M.ORDER_ACTUAL_EXPECTED = true +M.PRINT_TABLE_REF_IN_ERROR_MSG = false +M.LINE_LENGTH = 80 +M.TABLE_DIFF_ANALYSIS_THRESHOLD = 10 -- display deep analysis for more than 10 items +M.LIST_DIFF_ANALYSIS_THRESHOLD = 10 -- display deep analysis for more than 10 items + +-- this setting allow to remove entries from the stack-trace, for +-- example to hide a call to a framework which would be calling luaunit +M.STRIP_EXTRA_ENTRIES_IN_STACK_TRACE = 0 + +--[[ EPS is meant to help with Lua's floating point math in simple corner +cases like almostEquals(1.1-0.1, 1), which may not work as-is (e.g. on numbers +with rational binary representation) if the user doesn't provide some explicit +error margin. + +The default margin used by almostEquals() in such cases is EPS; and since +Lua may be compiled with different numeric precisions (single vs. double), we +try to select a useful default for it dynamically. Note: If the initial value +is not acceptable, it can be changed by the user to better suit specific needs. + +See also: https://en.wikipedia.org/wiki/Machine_epsilon +]] +M.EPS = 2^-52 -- = machine epsilon for "double", ~2.22E-16 +if math.abs(1.1 - 1 - 0.1) > M.EPS then + -- rounding error is above EPS, assume single precision + M.EPS = 2^-23 -- = machine epsilon for "float", ~1.19E-07 +end + +-- set this to false to debug luaunit +local STRIP_LUAUNIT_FROM_STACKTRACE = true + +M.VERBOSITY_DEFAULT = 10 +M.VERBOSITY_LOW = 1 +M.VERBOSITY_QUIET = 0 +M.VERBOSITY_VERBOSE = 20 +M.DEFAULT_DEEP_ANALYSIS = nil +M.FORCE_DEEP_ANALYSIS = true +M.DISABLE_DEEP_ANALYSIS = false + +-- set EXPORT_ASSERT_TO_GLOBALS to have all asserts visible as global values +-- EXPORT_ASSERT_TO_GLOBALS = true + +-- we need to keep a copy of the script args before it is overriden +local cmdline_argv = rawget(_G, "arg") + +M.FAILURE_PREFIX = 'LuaUnit test FAILURE: ' -- prefix string for failed tests +M.SUCCESS_PREFIX = 'LuaUnit test SUCCESS: ' -- prefix string for successful tests finished early +M.SKIP_PREFIX = 'LuaUnit test SKIP: ' -- prefix string for skipped tests + + + +M.USAGE=[[Usage: lua [options] [testname1 [testname2] ... ] +Options: + -h, --help: Print this help + --version: Print version information + -v, --verbose: Increase verbosity + -q, --quiet: Set verbosity to minimum + -e, --error: Stop on first error + -f, --failure: Stop on first failure or error + -s, --shuffle: Shuffle tests before running them + -o, --output OUTPUT: Set output type to OUTPUT + Possible values: text, tap, junit, nil + -n, --name NAME: For junit only, mandatory name of xml file + -r, --repeat NUM: Execute all tests NUM times, e.g. to trig the JIT + -p, --pattern PATTERN: Execute all test names matching the Lua PATTERN + May be repeated to include several patterns + Make sure you escape magic chars like +? with % + -x, --exclude PATTERN: Exclude all test names matching the Lua PATTERN + May be repeated to exclude several patterns + Make sure you escape magic chars like +? with % + testname1, testname2, ... : tests to run in the form of testFunction, + TestClass or TestClass.testMethod + +You may also control LuaUnit options with the following environment variables: +* LUAUNIT_OUTPUT: same as --output +* LUAUNIT_JUNIT_FNAME: same as --name ]] + +---------------------------------------------------------------- +-- +-- general utility functions +-- +---------------------------------------------------------------- + +--[[ Note on catching exit + +I have seen the case where running a big suite of test cases and one of them would +perform a os.exit(0), making the outside world think that the full test suite was executed +successfully. + +This is an attempt to mitigate this problem: we override os.exit() to now let a test +exit the framework while we are running. When we are not running, it behaves normally. +]] + +M.oldOsExit = os.exit +os.exit = function(...) + if M.LuaUnit and #M.LuaUnit.instances ~= 0 then + local msg = [[You are trying to exit but there is still a running instance of LuaUnit. +LuaUnit expects to run until the end before exiting with a complete status of successful/failed tests. + +To force exit LuaUnit while running, please call before os.exit (assuming lu is the luaunit module loaded): + + lu.unregisterCurrentSuite() + +]] + M.private.error_fmt(2, msg) + end + M.oldOsExit(...) +end + +local function pcall_or_abort(func, ...) + -- unpack is a global function for Lua 5.1, otherwise use table.unpack + local unpack = rawget(_G, "unpack") or table.unpack + local result = {pcall(func, ...)} + if not result[1] then + -- an error occurred + print(result[2]) -- error message + print() + print(M.USAGE) + os.exit(-1) + end + return unpack(result, 2) +end + +local crossTypeOrdering = { + number = 1, boolean = 2, string = 3, table = 4, other = 5 +} +local crossTypeComparison = { + number = function(a, b) return a < b end, + string = function(a, b) return a < b end, + other = function(a, b) return tostring(a) < tostring(b) end, +} + +local function crossTypeSort(a, b) + local type_a, type_b = type(a), type(b) + if type_a == type_b then + local func = crossTypeComparison[type_a] or crossTypeComparison.other + return func(a, b) + end + type_a = crossTypeOrdering[type_a] or crossTypeOrdering.other + type_b = crossTypeOrdering[type_b] or crossTypeOrdering.other + return type_a < type_b +end + +local function __genSortedIndex( t ) + -- Returns a sequence consisting of t's keys, sorted. + local sortedIndex = {} + + for key,_ in pairs(t) do + table.insert(sortedIndex, key) + end + + table.sort(sortedIndex, crossTypeSort) + return sortedIndex +end +M.private.__genSortedIndex = __genSortedIndex + +local function sortedNext(state, control) + -- Equivalent of the next() function of table iteration, but returns the + -- keys in sorted order (see __genSortedIndex and crossTypeSort). + -- The state is a temporary variable during iteration and contains the + -- sorted key table (state.sortedIdx). It also stores the last index (into + -- the keys) used by the iteration, to find the next one quickly. + local key + + --print("sortedNext: control = "..tostring(control) ) + if control == nil then + -- start of iteration + state.count = #state.sortedIdx + state.lastIdx = 1 + key = state.sortedIdx[1] + return key, state.t[key] + end + + -- normally, we expect the control variable to match the last key used + if control ~= state.sortedIdx[state.lastIdx] then + -- strange, we have to find the next value by ourselves + -- the key table is sorted in crossTypeSort() order! -> use bisection + local lower, upper = 1, state.count + repeat + state.lastIdx = math.modf((lower + upper) / 2) + key = state.sortedIdx[state.lastIdx] + if key == control then + break -- key found (and thus prev index) + end + if crossTypeSort(key, control) then + -- key < control, continue search "right" (towards upper bound) + lower = state.lastIdx + 1 + else + -- key > control, continue search "left" (towards lower bound) + upper = state.lastIdx - 1 + end + until lower > upper + if lower > upper then -- only true if the key wasn't found, ... + state.lastIdx = state.count -- ... so ensure no match in code below + end + end + + -- proceed by retrieving the next value (or nil) from the sorted keys + state.lastIdx = state.lastIdx + 1 + key = state.sortedIdx[state.lastIdx] + if key then + return key, state.t[key] + end + + -- getting here means returning `nil`, which will end the iteration +end + +local function sortedPairs(tbl) + -- Equivalent of the pairs() function on tables. Allows to iterate in + -- sorted order. As required by "generic for" loops, this will return the + -- iterator (function), an "invariant state", and the initial control value. + -- (see http://www.lua.org/pil/7.2.html) + return sortedNext, {t = tbl, sortedIdx = __genSortedIndex(tbl)}, nil +end +M.private.sortedPairs = sortedPairs + +-- seed the random with a strongly varying seed +math.randomseed(math.floor(os.clock()*1E11)) + +local function randomizeTable( t ) + -- randomize the item orders of the table t + for i = #t, 2, -1 do + local j = math.random(i) + if i ~= j then + t[i], t[j] = t[j], t[i] + end + end +end +M.private.randomizeTable = randomizeTable + +local function strsplit(delimiter, text) +-- Split text into a list consisting of the strings in text, separated +-- by strings matching delimiter (which may _NOT_ be a pattern). +-- Example: strsplit(", ", "Anna, Bob, Charlie, Dolores") + if delimiter == "" or delimiter == nil then -- this would result in endless loops + error("delimiter is nil or empty string!") + end + if text == nil then + return nil + end + + local list, pos, first, last = {}, 1 + while true do + first, last = text:find(delimiter, pos, true) + if first then -- found? + table.insert(list, text:sub(pos, first - 1)) + pos = last + 1 + else + table.insert(list, text:sub(pos)) + break + end + end + return list +end +M.private.strsplit = strsplit + +local function hasNewLine( s ) + -- return true if s has a newline + return (string.find(s, '\n', 1, true) ~= nil) +end +M.private.hasNewLine = hasNewLine + +local function prefixString( prefix, s ) + -- Prefix all the lines of s with prefix + return prefix .. string.gsub(s, '\n', '\n' .. prefix) +end +M.private.prefixString = prefixString + +local function strMatch(s, pattern, start, final ) + -- return true if s matches completely the pattern from index start to index end + -- return false in every other cases + -- if start is nil, matches from the beginning of the string + -- if final is nil, matches to the end of the string + start = start or 1 + final = final or string.len(s) + + local foundStart, foundEnd = string.find(s, pattern, start, false) + return foundStart == start and foundEnd == final +end +M.private.strMatch = strMatch + +local function patternFilter(patterns, expr) + -- Run `expr` through the inclusion and exclusion rules defined in patterns + -- and return true if expr shall be included, false for excluded. + -- Inclusion pattern are defined as normal patterns, exclusions + -- patterns start with `!` and are followed by a normal pattern + + -- result: nil = UNKNOWN (not matched yet), true = ACCEPT, false = REJECT + -- default: true if no explicit "include" is found, set to false otherwise + local default, result = true, nil + + if patterns ~= nil then + for _, pattern in ipairs(patterns) do + local exclude = pattern:sub(1,1) == '!' + if exclude then + pattern = pattern:sub(2) + else + -- at least one include pattern specified, a match is required + default = false + end + -- print('pattern: ',pattern) + -- print('exclude: ',exclude) + -- print('default: ',default) + + if string.find(expr, pattern) then + -- set result to false when excluding, true otherwise + result = not exclude + end + end + end + + if result ~= nil then + return result + end + return default +end +M.private.patternFilter = patternFilter + +local function xmlEscape( s ) + -- Return s escaped for XML attributes + -- escapes table: + -- " " + -- ' ' + -- < < + -- > > + -- & & + + return string.gsub( s, '.', { + ['&'] = "&", + ['"'] = """, + ["'"] = "'", + ['<'] = "<", + ['>'] = ">", + } ) +end +M.private.xmlEscape = xmlEscape + +local function xmlCDataEscape( s ) + -- Return s escaped for CData section, escapes: "]]>" + return string.gsub( s, ']]>', ']]>' ) +end +M.private.xmlCDataEscape = xmlCDataEscape + + +local function lstrip( s ) + --[[Return s with all leading white spaces and tabs removed]] + local idx = 0 + while idx < s:len() do + idx = idx + 1 + local c = s:sub(idx,idx) + if c ~= ' ' and c ~= '\t' then + break + end + end + return s:sub(idx) +end +M.private.lstrip = lstrip + +local function extractFileLineInfo( s ) + --[[ From a string in the form "(leading spaces) dir1/dir2\dir3\file.lua:linenb: msg" + + Return the "file.lua:linenb" information + ]] + local s2 = lstrip(s) + local firstColon = s2:find(':', 1, true) + if firstColon == nil then + -- string is not in the format file:line: + return s + end + local secondColon = s2:find(':', firstColon+1, true) + if secondColon == nil then + -- string is not in the format file:line: + return s + end + + return s2:sub(1, secondColon-1) +end +M.private.extractFileLineInfo = extractFileLineInfo + + +local function stripLuaunitTrace2( stackTrace, errMsg ) + --[[ + -- Example of a traceback: + < + [C]: in function 'xpcall' + ./luaunit.lua:1449: in function 'protectedCall' + ./luaunit.lua:1508: in function 'execOneFunction' + ./luaunit.lua:1596: in function 'runSuiteByInstances' + ./luaunit.lua:1660: in function 'runSuiteByNames' + ./luaunit.lua:1736: in function 'runSuite' + example_with_luaunit.lua:140: in main chunk + [C]: in ?>> + error message: <> + + Other example: + < + [C]: in function 'xpcall' + ./luaunit.lua:1517: in function 'protectedCall' + ./luaunit.lua:1578: in function 'execOneFunction' + ./luaunit.lua:1677: in function 'runSuiteByInstances' + ./luaunit.lua:1730: in function 'runSuiteByNames' + ./luaunit.lua:1806: in function 'runSuite' + example_with_luaunit.lua:140: in main chunk + [C]: in ?>> + error message: <> + + < + [C]: in function 'xpcall' + luaunit2/luaunit.lua:1532: in function 'protectedCall' + luaunit2/luaunit.lua:1591: in function 'execOneFunction' + luaunit2/luaunit.lua:1679: in function 'runSuiteByInstances' + luaunit2/luaunit.lua:1743: in function 'runSuiteByNames' + luaunit2/luaunit.lua:1819: in function 'runSuite' + luaunit2/example_with_luaunit.lua:140: in main chunk + [C]: in ?>> + error message: <> + + + -- first line is "stack traceback": KEEP + -- next line may be luaunit line: REMOVE + -- next lines are call in the program under testOk: REMOVE + -- next lines are calls from luaunit to call the program under test: KEEP + + -- Strategy: + -- keep first line + -- remove lines that are part of luaunit + -- kepp lines until we hit a luaunit line + + The strategy for stripping is: + * keep first line "stack traceback:" + * part1: + * analyse all lines of the stack from bottom to top of the stack (first line to last line) + * extract the "file:line:" part of the line + * compare it with the "file:line" part of the error message + * if it does not match strip the line + * if it matches, keep the line and move to part 2 + * part2: + * anything NOT starting with luaunit.lua is the interesting part of the stack trace + * anything starting again with luaunit.lua is part of the test launcher and should be stripped out + ]] + + local function isLuaunitInternalLine( s ) + -- return true if line of stack trace comes from inside luaunit + return s:find('[/\\]luaunit%.lua:%d+: ') ~= nil + end + + -- print( '<<'..stackTrace..'>>' ) + + local t = strsplit( '\n', stackTrace ) + -- print( prettystr(t) ) + + local idx = 2 + + local errMsgFileLine = extractFileLineInfo(errMsg) + -- print('emfi="'..errMsgFileLine..'"') + + -- remove lines that are still part of luaunit + while t[idx] and extractFileLineInfo(t[idx]) ~= errMsgFileLine do + -- print('Removing : '..t[idx] ) + table.remove(t, idx) + end + + -- keep lines until we hit luaunit again + while t[idx] and (not isLuaunitInternalLine(t[idx])) do + -- print('Keeping : '..t[idx] ) + idx = idx + 1 + end + + -- remove remaining luaunit lines + while t[idx] do + -- print('Removing2 : '..t[idx] ) + table.remove(t, idx) + end + + -- print( prettystr(t) ) + return table.concat( t, '\n') + +end +M.private.stripLuaunitTrace2 = stripLuaunitTrace2 + + +local function prettystr_sub(v, indentLevel, printTableRefs, cycleDetectTable ) + local type_v = type(v) + if "string" == type_v then + -- use clever delimiters according to content: + -- enclose with single quotes if string contains ", but no ' + if v:find('"', 1, true) and not v:find("'", 1, true) then + return "'" .. v .. "'" + end + -- use double quotes otherwise, escape embedded " + return '"' .. v:gsub('"', '\\"') .. '"' + + elseif "table" == type_v then + --if v.__class__ then + -- return string.gsub( tostring(v), 'table', v.__class__ ) + --end + return M.private._table_tostring(v, indentLevel, printTableRefs, cycleDetectTable) + + elseif "number" == type_v then + -- eliminate differences in formatting between various Lua versions + if v ~= v then + return "#NaN" -- "not a number" + end + if v == math.huge then + return "#Inf" -- "infinite" + end + if v == -math.huge then + return "-#Inf" + end + if _VERSION == "Lua 5.3" then + local i = math.tointeger(v) + if i then + return tostring(i) + end + end + end + + return tostring(v) +end + +local function prettystr( v ) + --[[ Pretty string conversion, to display the full content of a variable of any type. + + * string are enclosed with " by default, or with ' if string contains a " + * tables are expanded to show their full content, with indentation in case of nested tables + ]]-- + local cycleDetectTable = {} + local s = prettystr_sub(v, 1, M.PRINT_TABLE_REF_IN_ERROR_MSG, cycleDetectTable) + if cycleDetectTable.detected and not M.PRINT_TABLE_REF_IN_ERROR_MSG then + -- some table contain recursive references, + -- so we must recompute the value by including all table references + -- else the result looks like crap + cycleDetectTable = {} + s = prettystr_sub(v, 1, true, cycleDetectTable) + end + return s +end +M.prettystr = prettystr + +function M.adjust_err_msg_with_iter( err_msg, iter_msg ) + --[[ Adjust the error message err_msg: trim the FAILURE_PREFIX or SUCCESS_PREFIX information if needed, + add the iteration message if any and return the result. + + err_msg: string, error message captured with pcall + iter_msg: a string describing the current iteration ("iteration N") or nil + if there is no iteration in this test. + + Returns: (new_err_msg, test_status) + new_err_msg: string, adjusted error message, or nil in case of success + test_status: M.NodeStatus.FAIL, SUCCESS or ERROR according to the information + contained in the error message. + ]] + if iter_msg then + iter_msg = iter_msg..', ' + else + iter_msg = '' + end + + local RE_FILE_LINE = '.*:%d+: ' + + -- error message is not necessarily a string, + -- so convert the value to string with prettystr() + if type( err_msg ) ~= 'string' then + err_msg = prettystr( err_msg ) + end + + if (err_msg:find( M.SUCCESS_PREFIX ) == 1) or err_msg:match( '('..RE_FILE_LINE..')' .. M.SUCCESS_PREFIX .. ".*" ) then + -- test finished early with success() + return nil, M.NodeStatus.SUCCESS + end + + if (err_msg:find( M.SKIP_PREFIX ) == 1) or (err_msg:match( '('..RE_FILE_LINE..')' .. M.SKIP_PREFIX .. ".*" ) ~= nil) then + -- substitute prefix by iteration message + err_msg = err_msg:gsub('.*'..M.SKIP_PREFIX, iter_msg, 1) + -- print("failure detected") + return err_msg, M.NodeStatus.SKIP + end + + if (err_msg:find( M.FAILURE_PREFIX ) == 1) or (err_msg:match( '('..RE_FILE_LINE..')' .. M.FAILURE_PREFIX .. ".*" ) ~= nil) then + -- substitute prefix by iteration message + err_msg = err_msg:gsub(M.FAILURE_PREFIX, iter_msg, 1) + -- print("failure detected") + return err_msg, M.NodeStatus.FAIL + end + + + + -- print("error detected") + -- regular error, not a failure + if iter_msg then + local match + -- "./test\\test_luaunit.lua:2241: some error msg + match = err_msg:match( '(.*:%d+: ).*' ) + if match then + err_msg = err_msg:gsub( match, match .. iter_msg ) + else + -- no file:line: infromation, just add the iteration info at the beginning of the line + err_msg = iter_msg .. err_msg + end + end + return err_msg, M.NodeStatus.ERROR +end + +local function tryMismatchFormatting( table_a, table_b, doDeepAnalysis, margin ) + --[[ + Prepares a nice error message when comparing tables, performing a deeper + analysis. + + Arguments: + * table_a, table_b: tables to be compared + * doDeepAnalysis: + M.DEFAULT_DEEP_ANALYSIS: (the default if not specified) perform deep analysis only for big lists and big dictionnaries + M.FORCE_DEEP_ANALYSIS : always perform deep analysis + M.DISABLE_DEEP_ANALYSIS: never perform deep analysis + * margin: supplied only for almost equality + + Returns: {success, result} + * success: false if deep analysis could not be performed + in this case, just use standard assertion message + * result: if success is true, a multi-line string with deep analysis of the two lists + ]] + + -- check if table_a & table_b are suitable for deep analysis + if type(table_a) ~= 'table' or type(table_b) ~= 'table' then + return false + end + + if doDeepAnalysis == M.DISABLE_DEEP_ANALYSIS then + return false + end + + local len_a, len_b, isPureList = #table_a, #table_b, true + + for k1, v1 in pairs(table_a) do + if type(k1) ~= 'number' or k1 > len_a then + -- this table a mapping + isPureList = false + break + end + end + + if isPureList then + for k2, v2 in pairs(table_b) do + if type(k2) ~= 'number' or k2 > len_b then + -- this table a mapping + isPureList = false + break + end + end + end + + if isPureList and math.min(len_a, len_b) < M.LIST_DIFF_ANALYSIS_THRESHOLD then + if not (doDeepAnalysis == M.FORCE_DEEP_ANALYSIS) then + return false + end + end + + if isPureList then + return M.private.mismatchFormattingPureList( table_a, table_b, margin ) + else + -- only work on mapping for the moment + -- return M.private.mismatchFormattingMapping( table_a, table_b, doDeepAnalysis ) + return false + end +end +M.private.tryMismatchFormatting = tryMismatchFormatting + +local function getTaTbDescr() + if not M.ORDER_ACTUAL_EXPECTED then + return 'expected', 'actual' + end + return 'actual', 'expected' +end + +local function extendWithStrFmt( res, ... ) + table.insert( res, string.format( ... ) ) +end + +local function mismatchFormattingMapping( table_a, table_b, doDeepAnalysis ) + --[[ + Prepares a nice error message when comparing tables which are not pure lists, performing a deeper + analysis. + + Returns: {success, result} + * success: false if deep analysis could not be performed + in this case, just use standard assertion message + * result: if success is true, a multi-line string with deep analysis of the two lists + ]] + + -- disable for the moment + --[[ + local result = {} + local descrTa, descrTb = getTaTbDescr() + + local keysCommon = {} + local keysOnlyTa = {} + local keysOnlyTb = {} + local keysDiffTaTb = {} + + local k, v + + for k,v in pairs( table_a ) do + if is_equal( v, table_b[k] ) then + table.insert( keysCommon, k ) + else + if table_b[k] == nil then + table.insert( keysOnlyTa, k ) + else + table.insert( keysDiffTaTb, k ) + end + end + end + + for k,v in pairs( table_b ) do + if not is_equal( v, table_a[k] ) and table_a[k] == nil then + table.insert( keysOnlyTb, k ) + end + end + + local len_a = #keysCommon + #keysDiffTaTb + #keysOnlyTa + local len_b = #keysCommon + #keysDiffTaTb + #keysOnlyTb + local limited_display = (len_a < 5 or len_b < 5) + + if math.min(len_a, len_b) < M.TABLE_DIFF_ANALYSIS_THRESHOLD then + return false + end + + if not limited_display then + if len_a == len_b then + extendWithStrFmt( result, 'Table A (%s) and B (%s) both have %d items', descrTa, descrTb, len_a ) + else + extendWithStrFmt( result, 'Table A (%s) has %d items and table B (%s) has %d items', descrTa, len_a, descrTb, len_b ) + end + + if #keysCommon == 0 and #keysDiffTaTb == 0 then + table.insert( result, 'Table A and B have no keys in common, they are totally different') + else + local s_other = 'other ' + if #keysCommon then + extendWithStrFmt( result, 'Table A and B have %d identical items', #keysCommon ) + else + table.insert( result, 'Table A and B have no identical items' ) + s_other = '' + end + + if #keysDiffTaTb ~= 0 then + result[#result] = string.format( '%s and %d items differing present in both tables', result[#result], #keysDiffTaTb) + else + result[#result] = string.format( '%s and no %sitems differing present in both tables', result[#result], s_other, #keysDiffTaTb) + end + end + + extendWithStrFmt( result, 'Table A has %d keys not present in table B and table B has %d keys not present in table A', #keysOnlyTa, #keysOnlyTb ) + end + + local function keytostring(k) + if "string" == type(k) and k:match("^[_%a][_%w]*$") then + return k + end + return prettystr(k) + end + + if #keysDiffTaTb ~= 0 then + table.insert( result, 'Items differing in A and B:') + for k,v in sortedPairs( keysDiffTaTb ) do + extendWithStrFmt( result, ' - A[%s]: %s', keytostring(v), prettystr(table_a[v]) ) + extendWithStrFmt( result, ' + B[%s]: %s', keytostring(v), prettystr(table_b[v]) ) + end + end + + if #keysOnlyTa ~= 0 then + table.insert( result, 'Items only in table A:' ) + for k,v in sortedPairs( keysOnlyTa ) do + extendWithStrFmt( result, ' - A[%s]: %s', keytostring(v), prettystr(table_a[v]) ) + end + end + + if #keysOnlyTb ~= 0 then + table.insert( result, 'Items only in table B:' ) + for k,v in sortedPairs( keysOnlyTb ) do + extendWithStrFmt( result, ' + B[%s]: %s', keytostring(v), prettystr(table_b[v]) ) + end + end + + if #keysCommon ~= 0 then + table.insert( result, 'Items common to A and B:') + for k,v in sortedPairs( keysCommon ) do + extendWithStrFmt( result, ' = A and B [%s]: %s', keytostring(v), prettystr(table_a[v]) ) + end + end + + return true, table.concat( result, '\n') + ]] +end +M.private.mismatchFormattingMapping = mismatchFormattingMapping + +local function mismatchFormattingPureList( table_a, table_b, margin ) + --[[ + Prepares a nice error message when comparing tables which are lists, performing a deeper + analysis. + + margin is supplied only for almost equality + + Returns: {success, result} + * success: false if deep analysis could not be performed + in this case, just use standard assertion message + * result: if success is true, a multi-line string with deep analysis of the two lists + ]] + local result, descrTa, descrTb = {}, getTaTbDescr() + + local len_a, len_b, refa, refb = #table_a, #table_b, '', '' + if M.PRINT_TABLE_REF_IN_ERROR_MSG then + refa, refb = string.format( '<%s> ', M.private.table_ref(table_a)), string.format('<%s> ', M.private.table_ref(table_b) ) + end + local longest, shortest = math.max(len_a, len_b), math.min(len_a, len_b) + local deltalv = longest - shortest + + local commonUntil = shortest + for i = 1, shortest do + if not M.private.is_table_equals(table_a[i], table_b[i], margin) then + commonUntil = i - 1 + break + end + end + + local commonBackTo = shortest - 1 + for i = 0, shortest - 1 do + if not M.private.is_table_equals(table_a[len_a-i], table_b[len_b-i], margin) then + commonBackTo = i - 1 + break + end + end + + + table.insert( result, 'List difference analysis:' ) + if len_a == len_b then + -- TODO: handle expected/actual naming + extendWithStrFmt( result, '* lists %sA (%s) and %sB (%s) have the same size', refa, descrTa, refb, descrTb ) + else + extendWithStrFmt( result, '* list sizes differ: list %sA (%s) has %d items, list %sB (%s) has %d items', refa, descrTa, len_a, refb, descrTb, len_b ) + end + + extendWithStrFmt( result, '* lists A and B start differing at index %d', commonUntil+1 ) + if commonBackTo >= 0 then + if deltalv > 0 then + extendWithStrFmt( result, '* lists A and B are equal again from index %d for A, %d for B', len_a-commonBackTo, len_b-commonBackTo ) + else + extendWithStrFmt( result, '* lists A and B are equal again from index %d', len_a-commonBackTo ) + end + end + + local function insertABValue(ai, bi) + bi = bi or ai + if M.private.is_table_equals( table_a[ai], table_b[bi], margin) then + return extendWithStrFmt( result, ' = A[%d], B[%d]: %s', ai, bi, prettystr(table_a[ai]) ) + else + extendWithStrFmt( result, ' - A[%d]: %s', ai, prettystr(table_a[ai])) + extendWithStrFmt( result, ' + B[%d]: %s', bi, prettystr(table_b[bi])) + end + end + + -- common parts to list A & B, at the beginning + if commonUntil > 0 then + table.insert( result, '* Common parts:' ) + for i = 1, commonUntil do + insertABValue( i ) + end + end + + -- diffing parts to list A & B + if commonUntil < shortest - commonBackTo - 1 then + table.insert( result, '* Differing parts:' ) + for i = commonUntil + 1, shortest - commonBackTo - 1 do + insertABValue( i ) + end + end + + -- display indexes of one list, with no match on other list + if shortest - commonBackTo <= longest - commonBackTo - 1 then + table.insert( result, '* Present only in one list:' ) + for i = shortest - commonBackTo, longest - commonBackTo - 1 do + if len_a > len_b then + extendWithStrFmt( result, ' - A[%d]: %s', i, prettystr(table_a[i]) ) + -- table.insert( result, '+ (no matching B index)') + else + -- table.insert( result, '- no matching A index') + extendWithStrFmt( result, ' + B[%d]: %s', i, prettystr(table_b[i]) ) + end + end + end + + -- common parts to list A & B, at the end + if commonBackTo >= 0 then + table.insert( result, '* Common parts at the end of the lists' ) + for i = longest - commonBackTo, longest do + if len_a > len_b then + insertABValue( i, i-deltalv ) + else + insertABValue( i-deltalv, i ) + end + end + end + + return true, table.concat( result, '\n') +end +M.private.mismatchFormattingPureList = mismatchFormattingPureList + +local function prettystrPairs(value1, value2, suffix_a, suffix_b) + --[[ + This function helps with the recurring task of constructing the "expected + vs. actual" error messages. It takes two arbitrary values and formats + corresponding strings with prettystr(). + + To keep the (possibly complex) output more readable in case the resulting + strings contain line breaks, they get automatically prefixed with additional + newlines. Both suffixes are optional (default to empty strings), and get + appended to the "value1" string. "suffix_a" is used if line breaks were + encountered, "suffix_b" otherwise. + + Returns the two formatted strings (including padding/newlines). + ]] + local str1, str2 = prettystr(value1), prettystr(value2) + if hasNewLine(str1) or hasNewLine(str2) then + -- line break(s) detected, add padding + return "\n" .. str1 .. (suffix_a or ""), "\n" .. str2 + end + return str1 .. (suffix_b or ""), str2 +end +M.private.prettystrPairs = prettystrPairs + +local UNKNOWN_REF = 'table 00-unknown ref' +local ref_generator = { value=1, [UNKNOWN_REF]=0 } + +local function table_ref( t ) + -- return the default tostring() for tables, with the table ID, even if the table has a metatable + -- with the __tostring converter + local ref = '' + local mt = getmetatable( t ) + if mt == nil then + ref = tostring(t) + else + local success, result + success, result = pcall(setmetatable, t, nil) + if not success then + -- protected table, if __tostring is defined, we can + -- not get the reference. And we can not know in advance. + ref = tostring(t) + if not ref:match( 'table: 0?x?[%x]+' ) then + return UNKNOWN_REF + end + else + ref = tostring(t) + setmetatable( t, mt ) + end + end + -- strip the "table: " part + ref = ref:sub(8) + if ref ~= UNKNOWN_REF and ref_generator[ref] == nil then + -- Create a new reference number + ref_generator[ref] = ref_generator.value + ref_generator.value = ref_generator.value+1 + end + if M.PRINT_TABLE_REF_IN_ERROR_MSG then + return string.format('table %02d-%s', ref_generator[ref], ref) + else + return string.format('table %02d', ref_generator[ref]) + end +end +M.private.table_ref = table_ref + +local TABLE_TOSTRING_SEP = ", " +local TABLE_TOSTRING_SEP_LEN = string.len(TABLE_TOSTRING_SEP) + +local function _table_tostring( tbl, indentLevel, printTableRefs, cycleDetectTable ) + printTableRefs = printTableRefs or M.PRINT_TABLE_REF_IN_ERROR_MSG + cycleDetectTable = cycleDetectTable or {} + cycleDetectTable[tbl] = true + + local result, dispOnMultLines = {}, false + + -- like prettystr but do not enclose with "" if the string is just alphanumerical + -- this is better for displaying table keys who are often simple strings + local function keytostring(k) + if "string" == type(k) and k:match("^[_%a][_%w]*$") then + return k + end + return prettystr_sub(k, indentLevel+1, printTableRefs, cycleDetectTable) + end + + local mt = getmetatable( tbl ) + + if mt and mt.__tostring then + -- if table has a __tostring() function in its metatable, use it to display the table + -- else, compute a regular table + result = tostring(tbl) + if type(result) ~= 'string' then + return string.format( '', prettystr(result) ) + end + result = strsplit( '\n', result ) + return M.private._table_tostring_format_multiline_string( result, indentLevel ) + + else + -- no metatable, compute the table representation + + local entry, count, seq_index = nil, 0, 1 + for k, v in sortedPairs( tbl ) do + + -- key part + if k == seq_index then + -- for the sequential part of tables, we'll skip the "=" output + entry = '' + seq_index = seq_index + 1 + elseif cycleDetectTable[k] then + -- recursion in the key detected + cycleDetectTable.detected = true + entry = "<"..table_ref(k)..">=" + else + entry = keytostring(k) .. "=" + end + + -- value part + if cycleDetectTable[v] then + -- recursion in the value detected! + cycleDetectTable.detected = true + entry = entry .. "<"..table_ref(v)..">" + else + entry = entry .. + prettystr_sub( v, indentLevel+1, printTableRefs, cycleDetectTable ) + end + count = count + 1 + result[count] = entry + end + return M.private._table_tostring_format_result( tbl, result, indentLevel, printTableRefs ) + end + +end +M.private._table_tostring = _table_tostring -- prettystr_sub() needs it + +local function _table_tostring_format_multiline_string( tbl_str, indentLevel ) + local indentString = '\n'..string.rep(" ", indentLevel - 1) + return table.concat( tbl_str, indentString ) + +end +M.private._table_tostring_format_multiline_string = _table_tostring_format_multiline_string + + +local function _table_tostring_format_result( tbl, result, indentLevel, printTableRefs ) + -- final function called in _table_to_string() to format the resulting list of + -- string describing the table. + + local dispOnMultLines = false + + -- set dispOnMultLines to true if the maximum LINE_LENGTH would be exceeded with the values + local totalLength = 0 + for k, v in ipairs( result ) do + totalLength = totalLength + string.len( v ) + if totalLength >= M.LINE_LENGTH then + dispOnMultLines = true + break + end + end + + -- set dispOnMultLines to true if the max LINE_LENGTH would be exceeded + -- with the values and the separators. + if not dispOnMultLines then + -- adjust with length of separator(s): + -- two items need 1 sep, three items two seps, ... plus len of '{}' + if #result > 0 then + totalLength = totalLength + TABLE_TOSTRING_SEP_LEN * (#result - 1) + end + dispOnMultLines = (totalLength + 2 >= M.LINE_LENGTH) + end + + -- now reformat the result table (currently holding element strings) + if dispOnMultLines then + local indentString = string.rep(" ", indentLevel - 1) + result = { + "{\n ", + indentString, + table.concat(result, ",\n " .. indentString), + "\n", + indentString, + "}" + } + else + result = {"{", table.concat(result, TABLE_TOSTRING_SEP), "}"} + end + if printTableRefs then + table.insert(result, 1, "<"..table_ref(tbl).."> ") -- prepend table ref + end + return table.concat(result) +end +M.private._table_tostring_format_result = _table_tostring_format_result -- prettystr_sub() needs it + +local function table_findkeyof(t, element) + -- Return the key k of the given element in table t, so that t[k] == element + -- (or `nil` if element is not present within t). Note that we use our + -- 'general' is_equal comparison for matching, so this function should + -- handle table-type elements gracefully and consistently. + if type(t) == "table" then + for k, v in pairs(t) do + if M.private.is_table_equals(v, element) then + return k + end + end + end + return nil +end + +local function _is_table_items_equals(actual, expected ) + local type_a, type_e = type(actual), type(expected) + + if type_a ~= type_e then + return false + + elseif (type_a == 'table') --[[and (type_e == 'table')]] then + for k, v in pairs(actual) do + if table_findkeyof(expected, v) == nil then + return false -- v not contained in expected + end + end + for k, v in pairs(expected) do + if table_findkeyof(actual, v) == nil then + return false -- v not contained in actual + end + end + return true + + elseif actual ~= expected then + return false + end + + return true +end + +--[[ +This is a specialized metatable to help with the bookkeeping of recursions +in _is_table_equals(). It provides an __index table that implements utility +functions for easier management of the table. The "cached" method queries +the state of a specific (actual,expected) pair; and the "store" method sets +this state to the given value. The state of pairs not "seen" / visited is +assumed to be `nil`. +]] +local _recursion_cache_MT = { + __index = { + -- Return the cached value for an (actual,expected) pair (or `nil`) + cached = function(t, actual, expected) + local subtable = t[actual] or {} + return subtable[expected] + end, + + -- Store cached value for a specific (actual,expected) pair. + -- Returns the value, so it's easy to use for a "tailcall" (return ...). + store = function(t, actual, expected, value, asymmetric) + local subtable = t[actual] + if not subtable then + subtable = {} + t[actual] = subtable + end + subtable[expected] = value + + -- Unless explicitly marked "asymmetric": Consider the recursion + -- on (expected,actual) to be equivalent to (actual,expected) by + -- default, and thus cache the value for both. + if not asymmetric then + t:store(expected, actual, value, true) + end + + return value + end + } +} + +local function _is_table_equals(actual, expected, cycleDetectTable, marginForAlmostEqual) + --[[Returns true if both table are equal. + + If argument marginForAlmostEqual is suppied, number comparison is done using alomstEqual instead + of strict equality. + + cycleDetectTable is an internal argument used during recursion on tables. + ]] + --print('_is_table_equals( \n '..prettystr(actual)..'\n , '..prettystr(expected).. + -- '\n , '..prettystr(cycleDetectTable)..'\n , '..prettystr(marginForAlmostEqual)..' )') + + local type_a, type_e = type(actual), type(expected) + + if type_a ~= type_e then + return false -- different types won't match + end + + if type_a == 'number' then + if marginForAlmostEqual ~= nil then + return M.almostEquals(actual, expected, marginForAlmostEqual) + else + return actual == expected + end + elseif type_a ~= 'table' then + -- other types compare directly + return actual == expected + end + + cycleDetectTable = cycleDetectTable or { actual={}, expected={} } + if cycleDetectTable.actual[ actual ] then + -- oh, we hit a cycle in actual + if cycleDetectTable.expected[ expected ] then + -- uh, we hit a cycle at the same time in expected + -- so the two tables have similar structure + return true + end + + -- cycle was hit only in actual, the structure differs from expected + return false + end + + if cycleDetectTable.expected[ expected ] then + -- no cycle in actual, but cycle in expected + -- the structure differ + return false + end + + -- at this point, no table cycle detected, we are + -- seeing this table for the first time + + -- mark the cycle detection + cycleDetectTable.actual[ actual ] = true + cycleDetectTable.expected[ expected ] = true + + + local actualKeysMatched = {} + for k, v in pairs(actual) do + actualKeysMatched[k] = true -- Keep track of matched keys + if not _is_table_equals(v, expected[k], cycleDetectTable, marginForAlmostEqual) then + -- table differs on this key + -- clear the cycle detection before returning + cycleDetectTable.actual[ actual ] = nil + cycleDetectTable.expected[ expected ] = nil + return false + end + end + + for k, v in pairs(expected) do + if not actualKeysMatched[k] then + -- Found a key that we did not see in "actual" -> mismatch + -- clear the cycle detection before returning + cycleDetectTable.actual[ actual ] = nil + cycleDetectTable.expected[ expected ] = nil + return false + end + -- Otherwise actual[k] was already matched against v = expected[k]. + end + + -- all key match, we have a match ! + cycleDetectTable.actual[ actual ] = nil + cycleDetectTable.expected[ expected ] = nil + return true +end +M.private._is_table_equals = _is_table_equals + +local function failure(main_msg, extra_msg_or_nil, level) + -- raise an error indicating a test failure + -- for error() compatibility we adjust "level" here (by +1), to report the + -- calling context + local msg + if type(extra_msg_or_nil) == 'string' and extra_msg_or_nil:len() > 0 then + msg = extra_msg_or_nil .. '\n' .. main_msg + else + msg = main_msg + end + error(M.FAILURE_PREFIX .. msg, (level or 1) + 1 + M.STRIP_EXTRA_ENTRIES_IN_STACK_TRACE) +end + +local function is_table_equals(actual, expected, marginForAlmostEqual) + return _is_table_equals(actual, expected, nil, marginForAlmostEqual) +end +M.private.is_table_equals = is_table_equals + +local function fail_fmt(level, extra_msg_or_nil, ...) + -- failure with printf-style formatted message and given error level + failure(string.format(...), extra_msg_or_nil, (level or 1) + 1) +end +M.private.fail_fmt = fail_fmt + +local function error_fmt(level, ...) + -- printf-style error() + error(string.format(...), (level or 1) + 1 + M.STRIP_EXTRA_ENTRIES_IN_STACK_TRACE) +end +M.private.error_fmt = error_fmt + +---------------------------------------------------------------- +-- +-- assertions +-- +---------------------------------------------------------------- + +local function errorMsgEquality(actual, expected, doDeepAnalysis, margin) + -- margin is supplied only for almost equal verification + + if not M.ORDER_ACTUAL_EXPECTED then + expected, actual = actual, expected + end + if type(expected) == 'string' or type(expected) == 'table' then + local strExpected, strActual = prettystrPairs(expected, actual) + local result = string.format("expected: %s\nactual: %s", strExpected, strActual) + if margin then + result = result .. '\nwere not equal by the margin of: '..prettystr(margin) + end + + -- extend with mismatch analysis if possible: + local success, mismatchResult + success, mismatchResult = tryMismatchFormatting( actual, expected, doDeepAnalysis, margin ) + if success then + result = table.concat( { result, mismatchResult }, '\n' ) + end + return result + end + return string.format("expected: %s, actual: %s", + prettystr(expected), prettystr(actual)) +end + +function M.assertError(f, ...) + -- assert that calling f with the arguments will raise an error + -- example: assertError( f, 1, 2 ) => f(1,2) should generate an error + if pcall( f, ... ) then + failure( "Expected an error when calling function but no error generated", nil, 2 ) + end +end + +function M.fail( msg ) + -- stops a test due to a failure + failure( msg, nil, 2 ) +end + +function M.failIf( cond, msg ) + -- Fails a test with "msg" if condition is true + if cond then + failure( msg, nil, 2 ) + end +end + +function M.skip(msg) + -- skip a running test + error_fmt(2, M.SKIP_PREFIX .. msg) +end + +function M.skipIf( cond, msg ) + -- skip a running test if condition is met + if cond then + error_fmt(2, M.SKIP_PREFIX .. msg) + end +end + +function M.runOnlyIf( cond, msg ) + -- continue a running test if condition is met, else skip it + if not cond then + error_fmt(2, M.SKIP_PREFIX .. prettystr(msg)) + end +end + +function M.success() + -- stops a test with a success + error_fmt(2, M.SUCCESS_PREFIX) +end + +function M.successIf( cond ) + -- stops a test with a success if condition is met + if cond then + error_fmt(2, M.SUCCESS_PREFIX) + end +end + + +------------------------------------------------------------------ +-- Equality assertions +------------------------------------------------------------------ + +function M.assertEquals(actual, expected, extra_msg_or_nil, doDeepAnalysis) + if type(actual) == 'table' and type(expected) == 'table' then + if not is_table_equals(actual, expected) then + failure( errorMsgEquality(actual, expected, doDeepAnalysis), extra_msg_or_nil, 2 ) + end + elseif type(actual) ~= type(expected) then + failure( errorMsgEquality(actual, expected), extra_msg_or_nil, 2 ) + elseif actual ~= expected then + failure( errorMsgEquality(actual, expected), extra_msg_or_nil, 2 ) + end +end + +function M.almostEquals( actual, expected, margin ) + if type(actual) ~= 'number' or type(expected) ~= 'number' or type(margin) ~= 'number' then + error_fmt(3, 'almostEquals: must supply only number arguments.\nArguments supplied: %s, %s, %s', + prettystr(actual), prettystr(expected), prettystr(margin)) + end + if margin < 0 then + error_fmt(3, 'almostEquals: margin must not be negative, current value is ' .. margin) + end + return math.abs(expected - actual) <= margin +end + +function M.assertAlmostEquals( actual, expected, margin, extra_msg_or_nil ) + -- check that two floats are close by margin + margin = margin or M.EPS + if type(margin) ~= 'number' then + error_fmt(2, 'almostEquals: margin must be a number, not %s', prettystr(margin)) + end + + if type(actual) == 'table' and type(expected) == 'table' then + -- handle almost equals for table + if not is_table_equals(actual, expected, margin) then + failure( errorMsgEquality(actual, expected, nil, margin), extra_msg_or_nil, 2 ) + end + elseif type(actual) == 'number' and type(expected) == 'number' and type(margin) == 'number' then + if not M.almostEquals(actual, expected, margin) then + if not M.ORDER_ACTUAL_EXPECTED then + expected, actual = actual, expected + end + local delta = math.abs(actual - expected) + fail_fmt(2, extra_msg_or_nil, 'Values are not almost equal\n' .. + 'Actual: %s, expected: %s, delta %s above margin of %s', + actual, expected, delta, margin) + end + else + error_fmt(3, 'almostEquals: must supply only number or table arguments.\nArguments supplied: %s, %s, %s', + prettystr(actual), prettystr(expected), prettystr(margin)) + end +end + +function M.assertNotEquals(actual, expected, extra_msg_or_nil) + if type(actual) ~= type(expected) then + return + end + + if type(actual) == 'table' and type(expected) == 'table' then + if not is_table_equals(actual, expected) then + return + end + elseif actual ~= expected then + return + end + fail_fmt(2, extra_msg_or_nil, 'Received the not expected value: %s', prettystr(actual)) +end + +function M.assertNotAlmostEquals( actual, expected, margin, extra_msg_or_nil ) + -- check that two floats are not close by margin + margin = margin or M.EPS + if M.almostEquals(actual, expected, margin) then + if not M.ORDER_ACTUAL_EXPECTED then + expected, actual = actual, expected + end + local delta = math.abs(actual - expected) + fail_fmt(2, extra_msg_or_nil, 'Values are almost equal\nActual: %s, expected: %s' .. + ', delta %s below margin of %s', + actual, expected, delta, margin) + end +end + +function M.assertItemsEquals(actual, expected, extra_msg_or_nil) + -- checks that the items of table expected + -- are contained in table actual. Warning, this function + -- is at least O(n^2) + if not _is_table_items_equals(actual, expected ) then + expected, actual = prettystrPairs(expected, actual) + fail_fmt(2, extra_msg_or_nil, 'Content of the tables are not identical:\nExpected: %s\nActual: %s', + expected, actual) + end +end + +------------------------------------------------------------------ +-- String assertion +------------------------------------------------------------------ + +function M.assertStrContains( str, sub, isPattern, extra_msg_or_nil ) + -- this relies on lua string.find function + -- a string always contains the empty string + -- assert( type(str) == 'string', 'Argument 1 of assertStrContains() should be a string.' ) ) + -- assert( type(sub) == 'string', 'Argument 2 of assertStrContains() should be a string.' ) ) + if not string.find(str, sub, 1, not isPattern) then + sub, str = prettystrPairs(sub, str, '\n') + fail_fmt(2, extra_msg_or_nil, 'Could not find %s %s in string %s', + isPattern and 'pattern' or 'substring', sub, str) + end +end + +function M.assertStrIContains( str, sub, extra_msg_or_nil ) + -- this relies on lua string.find function + -- a string always contains the empty string + if not string.find(str:lower(), sub:lower(), 1, true) then + sub, str = prettystrPairs(sub, str, '\n') + fail_fmt(2, extra_msg_or_nil, 'Could not find (case insensitively) substring %s in string %s', + sub, str) + end +end + +function M.assertNotStrContains( str, sub, isPattern, extra_msg_or_nil ) + -- this relies on lua string.find function + -- a string always contains the empty string + if string.find(str, sub, 1, not isPattern) then + sub, str = prettystrPairs(sub, str, '\n') + fail_fmt(2, extra_msg_or_nil, 'Found the not expected %s %s in string %s', + isPattern and 'pattern' or 'substring', sub, str) + end +end + +function M.assertNotStrIContains( str, sub, extra_msg_or_nil ) + -- this relies on lua string.find function + -- a string always contains the empty string + if string.find(str:lower(), sub:lower(), 1, true) then + sub, str = prettystrPairs(sub, str, '\n') + fail_fmt(2, extra_msg_or_nil, 'Found (case insensitively) the not expected substring %s in string %s', + sub, str) + end +end + +function M.assertStrMatches( str, pattern, start, final, extra_msg_or_nil ) + -- Verify a full match for the string + if not strMatch( str, pattern, start, final ) then + pattern, str = prettystrPairs(pattern, str, '\n') + fail_fmt(2, extra_msg_or_nil, 'Could not match pattern %s with string %s', + pattern, str) + end +end + +local function _assertErrorMsgEquals( stripFileAndLine, expectedMsg, func, ... ) + local no_error, error_msg = pcall( func, ... ) + if no_error then + failure( 'No error generated when calling function but expected error: '..M.prettystr(expectedMsg), nil, 3 ) + end + if type(expectedMsg) == "string" and type(error_msg) ~= "string" then + -- table are converted to string automatically + error_msg = tostring(error_msg) + end + local differ = false + if stripFileAndLine then + if error_msg:gsub("^.+:%d+: ", "") ~= expectedMsg then + differ = true + end + else + if error_msg ~= expectedMsg then + local tr = type(error_msg) + local te = type(expectedMsg) + if te == 'table' then + if tr ~= 'table' then + differ = true + else + local ok = pcall(M.assertItemsEquals, error_msg, expectedMsg) + if not ok then + differ = true + end + end + else + differ = true + end + end + end + + if differ then + error_msg, expectedMsg = prettystrPairs(error_msg, expectedMsg) + fail_fmt(3, nil, 'Error message expected: %s\nError message received: %s\n', + expectedMsg, error_msg) + end +end + +function M.assertErrorMsgEquals( expectedMsg, func, ... ) + -- assert that calling f with the arguments will raise an error + -- example: assertError( f, 1, 2 ) => f(1,2) should generate an error + _assertErrorMsgEquals(false, expectedMsg, func, ...) +end + +function M.assertErrorMsgContentEquals(expectedMsg, func, ...) + _assertErrorMsgEquals(true, expectedMsg, func, ...) +end + +function M.assertErrorMsgContains( partialMsg, func, ... ) + -- assert that calling f with the arguments will raise an error + -- example: assertError( f, 1, 2 ) => f(1,2) should generate an error + local no_error, error_msg = pcall( func, ... ) + if no_error then + failure( 'No error generated when calling function but expected error containing: '..prettystr(partialMsg), nil, 2 ) + end + if type(error_msg) ~= "string" then + error_msg = tostring(error_msg) + end + if not string.find( error_msg, partialMsg, nil, true ) then + error_msg, partialMsg = prettystrPairs(error_msg, partialMsg) + fail_fmt(2, nil, 'Error message does not contain: %s\nError message received: %s\n', + partialMsg, error_msg) + end +end + +function M.assertErrorMsgMatches( expectedMsg, func, ... ) + -- assert that calling f with the arguments will raise an error + -- example: assertError( f, 1, 2 ) => f(1,2) should generate an error + local no_error, error_msg = pcall( func, ... ) + if no_error then + failure( 'No error generated when calling function but expected error matching: "'..expectedMsg..'"', nil, 2 ) + end + if type(error_msg) ~= "string" then + error_msg = tostring(error_msg) + end + if not strMatch( error_msg, expectedMsg ) then + expectedMsg, error_msg = prettystrPairs(expectedMsg, error_msg) + fail_fmt(2, nil, 'Error message does not match pattern: %s\nError message received: %s\n', + expectedMsg, error_msg) + end +end + +------------------------------------------------------------------ +-- Type assertions +------------------------------------------------------------------ + +function M.assertEvalToTrue(value, extra_msg_or_nil) + if not value then + failure("expected: a value evaluating to true, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertEvalToFalse(value, extra_msg_or_nil) + if value then + failure("expected: false or nil, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertIsTrue(value, extra_msg_or_nil) + if value ~= true then + failure("expected: true, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertNotIsTrue(value, extra_msg_or_nil) + if value == true then + failure("expected: not true, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertIsFalse(value, extra_msg_or_nil) + if value ~= false then + failure("expected: false, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertNotIsFalse(value, extra_msg_or_nil) + if value == false then + failure("expected: not false, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertIsNil(value, extra_msg_or_nil) + if value ~= nil then + failure("expected: nil, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertNotIsNil(value, extra_msg_or_nil) + if value == nil then + failure("expected: not nil, actual: nil", extra_msg_or_nil, 2) + end +end + +--[[ +Add type assertion functions to the module table M. Each of these functions +takes a single parameter "value", and checks that its Lua type matches the +expected string (derived from the function name): + +M.assertIsXxx(value) -> ensure that type(value) conforms to "xxx" +]] +for _, funcName in ipairs( + {'assertIsNumber', 'assertIsString', 'assertIsTable', 'assertIsBoolean', + 'assertIsFunction', 'assertIsUserdata', 'assertIsThread'} +) do + local typeExpected = funcName:match("^assertIs([A-Z]%a*)$") + -- Lua type() always returns lowercase, also make sure the match() succeeded + typeExpected = typeExpected and typeExpected:lower() + or error("bad function name '"..funcName.."' for type assertion") + + M[funcName] = function(value, extra_msg_or_nil) + if type(value) ~= typeExpected then + if type(value) == 'nil' then + fail_fmt(2, extra_msg_or_nil, 'expected: a %s value, actual: nil', + typeExpected, type(value), prettystrPairs(value)) + else + fail_fmt(2, extra_msg_or_nil, 'expected: a %s value, actual: type %s, value %s', + typeExpected, type(value), prettystrPairs(value)) + end + end + end +end + +--[[ +Add shortcuts for verifying type of a variable, without failure (luaunit v2 compatibility) +M.isXxx(value) -> returns true if type(value) conforms to "xxx" +]] +for _, typeExpected in ipairs( + {'Number', 'String', 'Table', 'Boolean', + 'Function', 'Userdata', 'Thread', 'Nil' } +) do + local typeExpectedLower = typeExpected:lower() + local isType = function(value) + return (type(value) == typeExpectedLower) + end + M['is'..typeExpected] = isType + M['is_'..typeExpectedLower] = isType +end + +--[[ +Add non-type assertion functions to the module table M. Each of these functions +takes a single parameter "value", and checks that its Lua type differs from the +expected string (derived from the function name): + +M.assertNotIsXxx(value) -> ensure that type(value) is not "xxx" +]] +for _, funcName in ipairs( + {'assertNotIsNumber', 'assertNotIsString', 'assertNotIsTable', 'assertNotIsBoolean', + 'assertNotIsFunction', 'assertNotIsUserdata', 'assertNotIsThread'} +) do + local typeUnexpected = funcName:match("^assertNotIs([A-Z]%a*)$") + -- Lua type() always returns lowercase, also make sure the match() succeeded + typeUnexpected = typeUnexpected and typeUnexpected:lower() + or error("bad function name '"..funcName.."' for type assertion") + + M[funcName] = function(value, extra_msg_or_nil) + if type(value) == typeUnexpected then + fail_fmt(2, extra_msg_or_nil, 'expected: not a %s type, actual: value %s', + typeUnexpected, prettystrPairs(value)) + end + end +end + +function M.assertIs(actual, expected, extra_msg_or_nil) + if actual ~= expected then + if not M.ORDER_ACTUAL_EXPECTED then + actual, expected = expected, actual + end + local old_print_table_ref_in_error_msg = M.PRINT_TABLE_REF_IN_ERROR_MSG + M.PRINT_TABLE_REF_IN_ERROR_MSG = true + expected, actual = prettystrPairs(expected, actual, '\n', '') + M.PRINT_TABLE_REF_IN_ERROR_MSG = old_print_table_ref_in_error_msg + fail_fmt(2, extra_msg_or_nil, 'expected and actual object should not be different\nExpected: %s\nReceived: %s', + expected, actual) + end +end + +function M.assertNotIs(actual, expected, extra_msg_or_nil) + if actual == expected then + local old_print_table_ref_in_error_msg = M.PRINT_TABLE_REF_IN_ERROR_MSG + M.PRINT_TABLE_REF_IN_ERROR_MSG = true + local s_expected + if not M.ORDER_ACTUAL_EXPECTED then + s_expected = prettystrPairs(actual) + else + s_expected = prettystrPairs(expected) + end + M.PRINT_TABLE_REF_IN_ERROR_MSG = old_print_table_ref_in_error_msg + fail_fmt(2, extra_msg_or_nil, 'expected and actual object should be different: %s', s_expected ) + end +end + + +------------------------------------------------------------------ +-- Scientific assertions +------------------------------------------------------------------ + + +function M.assertIsNaN(value, extra_msg_or_nil) + if type(value) ~= "number" or value == value then + failure("expected: NaN, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertNotIsNaN(value, extra_msg_or_nil) + if type(value) == "number" and value ~= value then + failure("expected: not NaN, actual: NaN", extra_msg_or_nil, 2) + end +end + +function M.assertIsInf(value, extra_msg_or_nil) + if type(value) ~= "number" or math.abs(value) ~= math.huge then + failure("expected: #Inf, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertIsPlusInf(value, extra_msg_or_nil) + if type(value) ~= "number" or value ~= math.huge then + failure("expected: #Inf, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertIsMinusInf(value, extra_msg_or_nil) + if type(value) ~= "number" or value ~= -math.huge then + failure("expected: -#Inf, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertNotIsPlusInf(value, extra_msg_or_nil) + if type(value) == "number" and value == math.huge then + failure("expected: not #Inf, actual: #Inf", extra_msg_or_nil, 2) + end +end + +function M.assertNotIsMinusInf(value, extra_msg_or_nil) + if type(value) == "number" and value == -math.huge then + failure("expected: not -#Inf, actual: -#Inf", extra_msg_or_nil, 2) + end +end + +function M.assertNotIsInf(value, extra_msg_or_nil) + if type(value) == "number" and math.abs(value) == math.huge then + failure("expected: not infinity, actual: " .. prettystr(value), extra_msg_or_nil, 2) + end +end + +function M.assertIsPlusZero(value, extra_msg_or_nil) + if type(value) ~= 'number' or value ~= 0 then + failure("expected: +0.0, actual: " ..prettystr(value), extra_msg_or_nil, 2) + else if (1/value == -math.huge) then + -- more precise error diagnosis + failure("expected: +0.0, actual: -0.0", extra_msg_or_nil, 2) + else if (1/value ~= math.huge) then + -- strange, case should have already been covered + failure("expected: +0.0, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end + end + end +end + +function M.assertIsMinusZero(value, extra_msg_or_nil) + if type(value) ~= 'number' or value ~= 0 then + failure("expected: -0.0, actual: " ..prettystr(value), extra_msg_or_nil, 2) + else if (1/value == math.huge) then + -- more precise error diagnosis + failure("expected: -0.0, actual: +0.0", extra_msg_or_nil, 2) + else if (1/value ~= -math.huge) then + -- strange, case should have already been covered + failure("expected: -0.0, actual: " ..prettystr(value), extra_msg_or_nil, 2) + end + end + end +end + +function M.assertNotIsPlusZero(value, extra_msg_or_nil) + if type(value) == 'number' and (1/value == math.huge) then + failure("expected: not +0.0, actual: +0.0", extra_msg_or_nil, 2) + end +end + +function M.assertNotIsMinusZero(value, extra_msg_or_nil) + if type(value) == 'number' and (1/value == -math.huge) then + failure("expected: not -0.0, actual: -0.0", extra_msg_or_nil, 2) + end +end + +function M.assertTableContains(t, expected, extra_msg_or_nil) + -- checks that table t contains the expected element + if table_findkeyof(t, expected) == nil then + t, expected = prettystrPairs(t, expected) + fail_fmt(2, extra_msg_or_nil, 'Table %s does NOT contain the expected element %s', + t, expected) + end +end + +function M.assertNotTableContains(t, expected, extra_msg_or_nil) + -- checks that table t doesn't contain the expected element + local k = table_findkeyof(t, expected) + if k ~= nil then + t, expected = prettystrPairs(t, expected) + fail_fmt(2, extra_msg_or_nil, 'Table %s DOES contain the unwanted element %s (at key %s)', + t, expected, prettystr(k)) + end +end + +---------------------------------------------------------------- +-- Compatibility layer +---------------------------------------------------------------- + +-- for compatibility with LuaUnit v2.x +function M.wrapFunctions() + -- In LuaUnit version <= 2.1 , this function was necessary to include + -- a test function inside the global test suite. Nowadays, the functions + -- are simply run directly as part of the test discovery process. + -- so just do nothing ! + io.stderr:write[[Use of WrapFunctions() is no longer needed. +Just prefix your test function names with "test" or "Test" and they +will be picked up and run by LuaUnit. +]] +end + +local list_of_funcs = { + -- { official function name , alias } + + -- general assertions + { 'assertEquals' , 'assert_equals' }, + { 'assertItemsEquals' , 'assert_items_equals' }, + { 'assertNotEquals' , 'assert_not_equals' }, + { 'assertAlmostEquals' , 'assert_almost_equals' }, + { 'assertNotAlmostEquals' , 'assert_not_almost_equals' }, + { 'assertEvalToTrue' , 'assert_eval_to_true' }, + { 'assertEvalToFalse' , 'assert_eval_to_false' }, + { 'assertStrContains' , 'assert_str_contains' }, + { 'assertStrIContains' , 'assert_str_icontains' }, + { 'assertNotStrContains' , 'assert_not_str_contains' }, + { 'assertNotStrIContains' , 'assert_not_str_icontains' }, + { 'assertStrMatches' , 'assert_str_matches' }, + { 'assertError' , 'assert_error' }, + { 'assertErrorMsgEquals' , 'assert_error_msg_equals' }, + { 'assertErrorMsgContains' , 'assert_error_msg_contains' }, + { 'assertErrorMsgMatches' , 'assert_error_msg_matches' }, + { 'assertErrorMsgContentEquals', 'assert_error_msg_content_equals' }, + { 'assertIs' , 'assert_is' }, + { 'assertNotIs' , 'assert_not_is' }, + { 'assertTableContains' , 'assert_table_contains' }, + { 'assertNotTableContains' , 'assert_not_table_contains' }, + { 'wrapFunctions' , 'WrapFunctions' }, + { 'wrapFunctions' , 'wrap_functions' }, + + -- type assertions: assertIsXXX -> assert_is_xxx + { 'assertIsNumber' , 'assert_is_number' }, + { 'assertIsString' , 'assert_is_string' }, + { 'assertIsTable' , 'assert_is_table' }, + { 'assertIsBoolean' , 'assert_is_boolean' }, + { 'assertIsNil' , 'assert_is_nil' }, + { 'assertIsTrue' , 'assert_is_true' }, + { 'assertIsFalse' , 'assert_is_false' }, + { 'assertIsNaN' , 'assert_is_nan' }, + { 'assertIsInf' , 'assert_is_inf' }, + { 'assertIsPlusInf' , 'assert_is_plus_inf' }, + { 'assertIsMinusInf' , 'assert_is_minus_inf' }, + { 'assertIsPlusZero' , 'assert_is_plus_zero' }, + { 'assertIsMinusZero' , 'assert_is_minus_zero' }, + { 'assertIsFunction' , 'assert_is_function' }, + { 'assertIsThread' , 'assert_is_thread' }, + { 'assertIsUserdata' , 'assert_is_userdata' }, + + -- type assertions: assertIsXXX -> assertXxx + { 'assertIsNumber' , 'assertNumber' }, + { 'assertIsString' , 'assertString' }, + { 'assertIsTable' , 'assertTable' }, + { 'assertIsBoolean' , 'assertBoolean' }, + { 'assertIsNil' , 'assertNil' }, + { 'assertIsTrue' , 'assertTrue' }, + { 'assertIsFalse' , 'assertFalse' }, + { 'assertIsNaN' , 'assertNaN' }, + { 'assertIsInf' , 'assertInf' }, + { 'assertIsPlusInf' , 'assertPlusInf' }, + { 'assertIsMinusInf' , 'assertMinusInf' }, + { 'assertIsPlusZero' , 'assertPlusZero' }, + { 'assertIsMinusZero' , 'assertMinusZero'}, + { 'assertIsFunction' , 'assertFunction' }, + { 'assertIsThread' , 'assertThread' }, + { 'assertIsUserdata' , 'assertUserdata' }, + + -- type assertions: assertIsXXX -> assert_xxx (luaunit v2 compat) + { 'assertIsNumber' , 'assert_number' }, + { 'assertIsString' , 'assert_string' }, + { 'assertIsTable' , 'assert_table' }, + { 'assertIsBoolean' , 'assert_boolean' }, + { 'assertIsNil' , 'assert_nil' }, + { 'assertIsTrue' , 'assert_true' }, + { 'assertIsFalse' , 'assert_false' }, + { 'assertIsNaN' , 'assert_nan' }, + { 'assertIsInf' , 'assert_inf' }, + { 'assertIsPlusInf' , 'assert_plus_inf' }, + { 'assertIsMinusInf' , 'assert_minus_inf' }, + { 'assertIsPlusZero' , 'assert_plus_zero' }, + { 'assertIsMinusZero' , 'assert_minus_zero' }, + { 'assertIsFunction' , 'assert_function' }, + { 'assertIsThread' , 'assert_thread' }, + { 'assertIsUserdata' , 'assert_userdata' }, + + -- type assertions: assertNotIsXXX -> assert_not_is_xxx + { 'assertNotIsNumber' , 'assert_not_is_number' }, + { 'assertNotIsString' , 'assert_not_is_string' }, + { 'assertNotIsTable' , 'assert_not_is_table' }, + { 'assertNotIsBoolean' , 'assert_not_is_boolean' }, + { 'assertNotIsNil' , 'assert_not_is_nil' }, + { 'assertNotIsTrue' , 'assert_not_is_true' }, + { 'assertNotIsFalse' , 'assert_not_is_false' }, + { 'assertNotIsNaN' , 'assert_not_is_nan' }, + { 'assertNotIsInf' , 'assert_not_is_inf' }, + { 'assertNotIsPlusInf' , 'assert_not_plus_inf' }, + { 'assertNotIsMinusInf' , 'assert_not_minus_inf' }, + { 'assertNotIsPlusZero' , 'assert_not_plus_zero' }, + { 'assertNotIsMinusZero' , 'assert_not_minus_zero' }, + { 'assertNotIsFunction' , 'assert_not_is_function' }, + { 'assertNotIsThread' , 'assert_not_is_thread' }, + { 'assertNotIsUserdata' , 'assert_not_is_userdata' }, + + -- type assertions: assertNotIsXXX -> assertNotXxx (luaunit v2 compat) + { 'assertNotIsNumber' , 'assertNotNumber' }, + { 'assertNotIsString' , 'assertNotString' }, + { 'assertNotIsTable' , 'assertNotTable' }, + { 'assertNotIsBoolean' , 'assertNotBoolean' }, + { 'assertNotIsNil' , 'assertNotNil' }, + { 'assertNotIsTrue' , 'assertNotTrue' }, + { 'assertNotIsFalse' , 'assertNotFalse' }, + { 'assertNotIsNaN' , 'assertNotNaN' }, + { 'assertNotIsInf' , 'assertNotInf' }, + { 'assertNotIsPlusInf' , 'assertNotPlusInf' }, + { 'assertNotIsMinusInf' , 'assertNotMinusInf' }, + { 'assertNotIsPlusZero' , 'assertNotPlusZero' }, + { 'assertNotIsMinusZero' , 'assertNotMinusZero' }, + { 'assertNotIsFunction' , 'assertNotFunction' }, + { 'assertNotIsThread' , 'assertNotThread' }, + { 'assertNotIsUserdata' , 'assertNotUserdata' }, + + -- type assertions: assertNotIsXXX -> assert_not_xxx + { 'assertNotIsNumber' , 'assert_not_number' }, + { 'assertNotIsString' , 'assert_not_string' }, + { 'assertNotIsTable' , 'assert_not_table' }, + { 'assertNotIsBoolean' , 'assert_not_boolean' }, + { 'assertNotIsNil' , 'assert_not_nil' }, + { 'assertNotIsTrue' , 'assert_not_true' }, + { 'assertNotIsFalse' , 'assert_not_false' }, + { 'assertNotIsNaN' , 'assert_not_nan' }, + { 'assertNotIsInf' , 'assert_not_inf' }, + { 'assertNotIsPlusInf' , 'assert_not_plus_inf' }, + { 'assertNotIsMinusInf' , 'assert_not_minus_inf' }, + { 'assertNotIsPlusZero' , 'assert_not_plus_zero' }, + { 'assertNotIsMinusZero' , 'assert_not_minus_zero' }, + { 'assertNotIsFunction' , 'assert_not_function' }, + { 'assertNotIsThread' , 'assert_not_thread' }, + { 'assertNotIsUserdata' , 'assert_not_userdata' }, + + -- all assertions with Coroutine duplicate Thread assertions + { 'assertIsThread' , 'assertIsCoroutine' }, + { 'assertIsThread' , 'assertCoroutine' }, + { 'assertIsThread' , 'assert_is_coroutine' }, + { 'assertIsThread' , 'assert_coroutine' }, + { 'assertNotIsThread' , 'assertNotIsCoroutine' }, + { 'assertNotIsThread' , 'assertNotCoroutine' }, + { 'assertNotIsThread' , 'assert_not_is_coroutine' }, + { 'assertNotIsThread' , 'assert_not_coroutine' }, +} + +-- Create all aliases in M +for _,v in ipairs( list_of_funcs ) do + local funcname, alias = v[1], v[2] + M[alias] = M[funcname] + + if EXPORT_ASSERT_TO_GLOBALS then + _G[funcname] = M[funcname] + _G[alias] = M[funcname] + end +end + +---------------------------------------------------------------- +-- +-- Outputters +-- +---------------------------------------------------------------- + +-- A common "base" class for outputters +-- For concepts involved (class inheritance) see http://www.lua.org/pil/16.2.html + +local genericOutput = { __class__ = 'genericOutput' } -- class +local genericOutput_MT = { __index = genericOutput } -- metatable +M.genericOutput = genericOutput -- publish, so that custom classes may derive from it + +function genericOutput.new(runner, default_verbosity) + -- runner is the "parent" object controlling the output, usually a LuaUnit instance + local t = { runner = runner } + if runner then + t.result = runner.result + t.verbosity = runner.verbosity or default_verbosity + t.fname = runner.fname + else + t.verbosity = default_verbosity + end + return setmetatable( t, genericOutput_MT) +end + +-- abstract ("empty") methods +function genericOutput:startSuite() + -- Called once, when the suite is started +end + +function genericOutput:startClass(className) + -- Called each time a new test class is started +end + +function genericOutput:startTest(testName) + -- called each time a new test is started, right before the setUp() + -- the current test status node is already created and available in: self.result.currentNode +end + +function genericOutput:updateStatus(node) + -- called with status failed or error as soon as the error/failure is encountered + -- this method is NOT called for a successful test because a test is marked as successful by default + -- and does not need to be updated +end + +function genericOutput:endTest(node) + -- called when the test is finished, after the tearDown() method +end + +function genericOutput:endClass() + -- called when executing the class is finished, before moving on to the next class of at the end of the test execution +end + +function genericOutput:endSuite() + -- called at the end of the test suite execution +end + + +---------------------------------------------------------------- +-- class TapOutput +---------------------------------------------------------------- + +local TapOutput = genericOutput.new() -- derived class +local TapOutput_MT = { __index = TapOutput } -- metatable +TapOutput.__class__ = 'TapOutput' + + -- For a good reference for TAP format, check: http://testanything.org/tap-specification.html + + function TapOutput.new(runner) + local t = genericOutput.new(runner, M.VERBOSITY_LOW) + return setmetatable( t, TapOutput_MT) + end + function TapOutput:startSuite() + print("1.."..self.result.selectedCount) + print('# Started on '..self.result.startDate) + end + function TapOutput:startClass(className) + if className ~= '[TestFunctions]' then + print('# Starting class: '..className) + end + end + + function TapOutput:updateStatus( node ) + if node:isSkipped() then + io.stdout:write("ok ", self.result.currentTestNumber, "\t# SKIP ", node.msg, "\n" ) + return + end + + io.stdout:write("not ok ", self.result.currentTestNumber, "\t", node.testName, "\n") + if self.verbosity > M.VERBOSITY_LOW then + print( prefixString( '# ', node.msg ) ) + end + if (node:isFailure() or node:isError()) and self.verbosity > M.VERBOSITY_DEFAULT then + print( prefixString( '# ', node.stackTrace ) ) + end + end + + function TapOutput:endTest( node ) + if node:isSuccess() then + io.stdout:write("ok ", self.result.currentTestNumber, "\t", node.testName, "\n") + end + end + + function TapOutput:endSuite() + print( '# '..M.LuaUnit.statusLine( self.result ) ) + return self.result.notSuccessCount + end + + +-- class TapOutput end + +---------------------------------------------------------------- +-- class JUnitOutput +---------------------------------------------------------------- + +-- See directory junitxml for more information about the junit format +local JUnitOutput = genericOutput.new() -- derived class +local JUnitOutput_MT = { __index = JUnitOutput } -- metatable +JUnitOutput.__class__ = 'JUnitOutput' + + function JUnitOutput.new(runner) + local t = genericOutput.new(runner, M.VERBOSITY_LOW) + t.testList = {} + return setmetatable( t, JUnitOutput_MT ) + end + + function JUnitOutput:startSuite() + -- open xml file early to deal with errors + if self.fname == nil then + error('With Junit, an output filename must be supplied with --name!') + end + if string.sub(self.fname,-4) ~= '.xml' then + self.fname = self.fname..'.xml' + end + self.fd = io.open(self.fname, "w") + if self.fd == nil then + error("Could not open file for writing: "..self.fname) + end + + print('# XML output to '..self.fname) + print('# Started on '..self.result.startDate) + end + function JUnitOutput:startClass(className) + if className ~= '[TestFunctions]' then + print('# Starting class: '..className) + end + end + function JUnitOutput:startTest(testName) + print('# Starting test: '..testName) + end + + function JUnitOutput:updateStatus( node ) + if node:isFailure() then + print( '# Failure: ' .. prefixString( '# ', node.msg ):sub(4, nil) ) + -- print('# ' .. node.stackTrace) + elseif node:isError() then + print( '# Error: ' .. prefixString( '# ' , node.msg ):sub(4, nil) ) + -- print('# ' .. node.stackTrace) + end + end + + function JUnitOutput:endSuite() + print( '# '..M.LuaUnit.statusLine(self.result)) + + -- XML file writing + self.fd:write('\n') + self.fd:write('\n') + self.fd:write(string.format( + ' \n', + self.result.runCount, self.result.startIsodate, self.result.duration, self.result.errorCount, self.result.failureCount, self.result.skippedCount )) + self.fd:write(" \n") + self.fd:write(string.format(' \n', _VERSION ) ) + self.fd:write(string.format(' \n', M.VERSION) ) + -- XXX please include system name and version if possible + self.fd:write(" \n") + + for i,node in ipairs(self.result.allTests) do + self.fd:write(string.format(' \n', + node.className, node.testName, node.duration ) ) + if node:isNotSuccess() then + self.fd:write(node:statusXML()) + end + self.fd:write(' \n') + end + + -- Next two lines are needed to validate junit ANT xsd, but really not useful in general: + self.fd:write(' \n') + self.fd:write(' \n') + + self.fd:write(' \n') + self.fd:write('\n') + self.fd:close() + return self.result.notSuccessCount + end + + +-- class TapOutput end + +---------------------------------------------------------------- +-- class TextOutput +---------------------------------------------------------------- + +--[[ Example of other unit-tests suite text output + +-- Python Non verbose: + +For each test: . or F or E + +If some failed tests: + ============== + ERROR / FAILURE: TestName (testfile.testclass) + --------- + Stack trace + + +then -------------- +then "Ran x tests in 0.000s" +then OK or FAILED (failures=1, error=1) + +-- Python Verbose: +testname (filename.classname) ... ok +testname (filename.classname) ... FAIL +testname (filename.classname) ... ERROR + +then -------------- +then "Ran x tests in 0.000s" +then OK or FAILED (failures=1, error=1) + +-- Ruby: +Started + . + Finished in 0.002695 seconds. + + 1 tests, 2 assertions, 0 failures, 0 errors + +-- Ruby: +>> ruby tc_simple_number2.rb +Loaded suite tc_simple_number2 +Started +F.. +Finished in 0.038617 seconds. + + 1) Failure: +test_failure(TestSimpleNumber) [tc_simple_number2.rb:16]: +Adding doesn't work. +<3> expected but was +<4>. + +3 tests, 4 assertions, 1 failures, 0 errors + +-- Java Junit +.......F. +Time: 0,003 +There was 1 failure: +1) testCapacity(junit.samples.VectorTest)junit.framework.AssertionFailedError + at junit.samples.VectorTest.testCapacity(VectorTest.java:87) + at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) + at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) + at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) + +FAILURES!!! +Tests run: 8, Failures: 1, Errors: 0 + + +-- Maven + +# mvn test +------------------------------------------------------- + T E S T S +------------------------------------------------------- +Running math.AdditionTest +Tests run: 2, Failures: 1, Errors: 0, Skipped: 0, Time elapsed: +0.03 sec <<< FAILURE! + +Results : + +Failed tests: + testLireSymbole(math.AdditionTest) + +Tests run: 2, Failures: 1, Errors: 0, Skipped: 0 + + +-- LuaUnit +---- non verbose +* display . or F or E when running tests +---- verbose +* display test name + ok/fail +---- +* blank line +* number) ERROR or FAILURE: TestName + Stack trace +* blank line +* number) ERROR or FAILURE: TestName + Stack trace + +then -------------- +then "Ran x tests in 0.000s (%d not selected, %d skipped)" +then OK or FAILED (failures=1, error=1) + + +]] + +local TextOutput = genericOutput.new() -- derived class +local TextOutput_MT = { __index = TextOutput } -- metatable +TextOutput.__class__ = 'TextOutput' + + function TextOutput.new(runner) + local t = genericOutput.new(runner, M.VERBOSITY_DEFAULT) + t.errorList = {} + return setmetatable( t, TextOutput_MT ) + end + + function TextOutput:startSuite() + if self.verbosity > M.VERBOSITY_DEFAULT then + print( 'Started on '.. self.result.startDate ) + end + end + + function TextOutput:startTest(testName) + if self.verbosity > M.VERBOSITY_DEFAULT then + io.stdout:write( " ", self.result.currentNode.testName, " ... " ) + end + end + + function TextOutput:endTest( node ) + if node:isSuccess() then + if self.verbosity > M.VERBOSITY_DEFAULT then + io.stdout:write("Ok\n") + else + io.stdout:write(".") + io.stdout:flush() + end + else + if self.verbosity > M.VERBOSITY_DEFAULT then + print( node.status ) + print( node.msg ) + --[[ + -- find out when to do this: + if self.verbosity > M.VERBOSITY_DEFAULT then + print( node.stackTrace ) + end + ]] + else + -- write only the first character of status E, F or S + io.stdout:write(string.sub(node.status, 1, 1)) + io.stdout:flush() + end + end + end + + function TextOutput:displayOneFailedTest( index, fail ) + print(index..") "..fail.testName ) + print( fail.msg ) + print( fail.stackTrace ) + print() + end + + function TextOutput:displayErroredTests() + if #self.result.errorTests ~= 0 then + print("Tests with errors:") + print("------------------") + for i, v in ipairs(self.result.errorTests) do + self:displayOneFailedTest(i, v) + end + end + end + + function TextOutput:displayFailedTests() + if #self.result.failedTests ~= 0 then + print("Failed tests:") + print("-------------") + for i, v in ipairs(self.result.failedTests) do + self:displayOneFailedTest(i, v) + end + end + end + + function TextOutput:endSuite() + if self.verbosity > M.VERBOSITY_DEFAULT then + print("=========================================================") + else + print() + end + self:displayErroredTests() + self:displayFailedTests() + print( M.LuaUnit.statusLine( self.result ) ) + if self.result.notSuccessCount == 0 then + print('OK') + end + end + +-- class TextOutput end + + +---------------------------------------------------------------- +-- class NilOutput +---------------------------------------------------------------- + +local function nopCallable() + --print(42) + return nopCallable +end + +local NilOutput = { __class__ = 'NilOuptut' } -- class +local NilOutput_MT = { __index = nopCallable } -- metatable + +function NilOutput.new(runner) + return setmetatable( { __class__ = 'NilOutput' }, NilOutput_MT ) +end + +---------------------------------------------------------------- +-- +-- class LuaUnit +-- +---------------------------------------------------------------- + +M.LuaUnit = { + outputType = TextOutput, + verbosity = M.VERBOSITY_DEFAULT, + __class__ = 'LuaUnit', + instances = {} +} +local LuaUnit_MT = { __index = M.LuaUnit } + +if EXPORT_ASSERT_TO_GLOBALS then + LuaUnit = M.LuaUnit +end + + function M.LuaUnit.new() + local newInstance = setmetatable( {}, LuaUnit_MT ) + return newInstance + end + + -----------------[[ Utility methods ]]--------------------- + + function M.LuaUnit.asFunction(aObject) + -- return "aObject" if it is a function, and nil otherwise + if 'function' == type(aObject) then + return aObject + end + end + + function M.LuaUnit.splitClassMethod(someName) + --[[ + Return a pair of className, methodName strings for a name in the form + "class.method". If no class part (or separator) is found, will return + nil, someName instead (the latter being unchanged). + + This convention thus also replaces the older isClassMethod() test: + You just have to check for a non-nil className (return) value. + ]] + local separator = string.find(someName, '.', 1, true) + if separator then + return someName:sub(1, separator - 1), someName:sub(separator + 1) + end + return nil, someName + end + + function M.LuaUnit.isMethodTestName( s ) + -- return true is the name matches the name of a test method + -- default rule is that is starts with 'Test' or with 'test' + return string.sub(s, 1, 4):lower() == 'test' + end + + function M.LuaUnit.isTestName( s ) + -- return true is the name matches the name of a test + -- default rule is that is starts with 'Test' or with 'test' + return string.sub(s, 1, 4):lower() == 'test' + end + + function M.LuaUnit.collectTests() + -- return a list of all test names in the global namespace + -- that match LuaUnit.isTestName + + local testNames = {} + for k, _ in pairs(_G) do + if type(k) == "string" and M.LuaUnit.isTestName( k ) then + table.insert( testNames , k ) + end + end + table.sort( testNames ) + return testNames + end + + function M.LuaUnit.parseCmdLine( cmdLine ) + -- parse the command line + -- Supported command line parameters: + -- --verbose, -v: increase verbosity + -- --quiet, -q: silence output + -- --error, -e: treat errors as fatal (quit program) + -- --output, -o, + name: select output type + -- --pattern, -p, + pattern: run test matching pattern, may be repeated + -- --exclude, -x, + pattern: run test not matching pattern, may be repeated + -- --shuffle, -s, : shuffle tests before reunning them + -- --name, -n, + fname: name of output file for junit, default to stdout + -- --repeat, -r, + num: number of times to execute each test + -- [testnames, ...]: run selected test names + -- + -- Returns a table with the following fields: + -- verbosity: nil, M.VERBOSITY_DEFAULT, M.VERBOSITY_QUIET, M.VERBOSITY_VERBOSE + -- output: nil, 'tap', 'junit', 'text', 'nil' + -- testNames: nil or a list of test names to run + -- exeRepeat: num or 1 + -- pattern: nil or a list of patterns + -- exclude: nil or a list of patterns + + local result, state = {}, nil + local SET_OUTPUT = 1 + local SET_PATTERN = 2 + local SET_EXCLUDE = 3 + local SET_FNAME = 4 + local SET_REPEAT = 5 + + if cmdLine == nil then + return result + end + + local function parseOption( option ) + if option == '--help' or option == '-h' then + result['help'] = true + return + elseif option == '--version' then + result['version'] = true + return + elseif option == '--verbose' or option == '-v' then + result['verbosity'] = M.VERBOSITY_VERBOSE + return + elseif option == '--quiet' or option == '-q' then + result['verbosity'] = M.VERBOSITY_QUIET + return + elseif option == '--error' or option == '-e' then + result['quitOnError'] = true + return + elseif option == '--failure' or option == '-f' then + result['quitOnFailure'] = true + return + elseif option == '--shuffle' or option == '-s' then + result['shuffle'] = true + return + elseif option == '--output' or option == '-o' then + state = SET_OUTPUT + return state + elseif option == '--name' or option == '-n' then + state = SET_FNAME + return state + elseif option == '--repeat' or option == '-r' then + state = SET_REPEAT + return state + elseif option == '--pattern' or option == '-p' then + state = SET_PATTERN + return state + elseif option == '--exclude' or option == '-x' then + state = SET_EXCLUDE + return state + end + error('Unknown option: '..option,3) + end + + local function setArg( cmdArg, state ) + if state == SET_OUTPUT then + result['output'] = cmdArg + return + elseif state == SET_FNAME then + result['fname'] = cmdArg + return + elseif state == SET_REPEAT then + result['exeRepeat'] = tonumber(cmdArg) + or error('Malformed -r argument: '..cmdArg) + return + elseif state == SET_PATTERN then + if result['pattern'] then + table.insert( result['pattern'], cmdArg ) + else + result['pattern'] = { cmdArg } + end + return + elseif state == SET_EXCLUDE then + local notArg = '!'..cmdArg + if result['pattern'] then + table.insert( result['pattern'], notArg ) + else + result['pattern'] = { notArg } + end + return + end + error('Unknown parse state: '.. state) + end + + + for i, cmdArg in ipairs(cmdLine) do + if state ~= nil then + setArg( cmdArg, state, result ) + state = nil + else + if cmdArg:sub(1,1) == '-' then + state = parseOption( cmdArg ) + else + if result['testNames'] then + table.insert( result['testNames'], cmdArg ) + else + result['testNames'] = { cmdArg } + end + end + end + end + + if result['help'] then + M.LuaUnit.help() + end + + if result['version'] then + M.LuaUnit.version() + end + + if state ~= nil then + error('Missing argument after '..cmdLine[ #cmdLine ],2 ) + end + + return result + end + + function M.LuaUnit.help() + print(M.USAGE) + os.exit(0) + end + + function M.LuaUnit.version() + print('LuaUnit v'..M.VERSION..' by Philippe Fremy ') + os.exit(0) + end + +---------------------------------------------------------------- +-- class NodeStatus +---------------------------------------------------------------- + + local NodeStatus = { __class__ = 'NodeStatus' } -- class + local NodeStatus_MT = { __index = NodeStatus } -- metatable + M.NodeStatus = NodeStatus + + -- values of status + NodeStatus.SUCCESS = 'SUCCESS' + NodeStatus.SKIP = 'SKIP' + NodeStatus.FAIL = 'FAIL' + NodeStatus.ERROR = 'ERROR' + + function NodeStatus.new( number, testName, className ) + -- default constructor, test are PASS by default + local t = { number = number, testName = testName, className = className } + setmetatable( t, NodeStatus_MT ) + t:success() + return t + end + + function NodeStatus:success() + self.status = self.SUCCESS + -- useless because lua does this for us, but it helps me remembering the relevant field names + self.msg = nil + self.stackTrace = nil + end + + function NodeStatus:skip(msg) + self.status = self.SKIP + self.msg = msg + self.stackTrace = nil + end + + function NodeStatus:fail(msg, stackTrace) + self.status = self.FAIL + self.msg = msg + self.stackTrace = stackTrace + end + + function NodeStatus:error(msg, stackTrace) + self.status = self.ERROR + self.msg = msg + self.stackTrace = stackTrace + end + + function NodeStatus:isSuccess() + return self.status == NodeStatus.SUCCESS + end + + function NodeStatus:isNotSuccess() + -- Return true if node is either failure or error or skip + return (self.status == NodeStatus.FAIL or self.status == NodeStatus.ERROR or self.status == NodeStatus.SKIP) + end + + function NodeStatus:isSkipped() + return self.status == NodeStatus.SKIP + end + + function NodeStatus:isFailure() + return self.status == NodeStatus.FAIL + end + + function NodeStatus:isError() + return self.status == NodeStatus.ERROR + end + + function NodeStatus:statusXML() + if self:isError() then + return table.concat( + {' \n', + ' \n'}) + elseif self:isFailure() then + return table.concat( + {' \n', + ' \n'}) + elseif self:isSkipped() then + return table.concat({' ', xmlEscape(self.msg),'\n' } ) + end + return ' \n' -- (not XSD-compliant! normally shouldn't get here) + end + + --------------[[ Output methods ]]------------------------- + + local function conditional_plural(number, singular) + -- returns a grammatically well-formed string "%d " + local suffix = '' + if number ~= 1 then -- use plural + suffix = (singular:sub(-2) == 'ss') and 'es' or 's' + end + return string.format('%d %s%s', number, singular, suffix) + end + + function M.LuaUnit.statusLine(result) + -- return status line string according to results + local s = { + string.format('Ran %d tests in %0.3f seconds', + result.runCount, result.duration), + conditional_plural(result.successCount, 'success'), + } + if result.notSuccessCount > 0 then + if result.failureCount > 0 then + table.insert(s, conditional_plural(result.failureCount, 'failure')) + end + if result.errorCount > 0 then + table.insert(s, conditional_plural(result.errorCount, 'error')) + end + else + table.insert(s, '0 failures') + end + if result.skippedCount > 0 then + table.insert(s, string.format("%d skipped", result.skippedCount)) + end + if result.nonSelectedCount > 0 then + table.insert(s, string.format("%d non-selected", result.nonSelectedCount)) + end + return table.concat(s, ', ') + end + + function M.LuaUnit:startSuite(selectedCount, nonSelectedCount) + self.result = { + selectedCount = selectedCount, + nonSelectedCount = nonSelectedCount, + successCount = 0, + runCount = 0, + currentTestNumber = 0, + currentClassName = "", + currentNode = nil, + suiteStarted = true, + startTime = os.clock(), + startDate = os.date(os.getenv('LUAUNIT_DATEFMT')), + startIsodate = os.date('%Y-%m-%dT%H:%M:%S'), + patternIncludeFilter = self.patternIncludeFilter, + + -- list of test node status + allTests = {}, + failedTests = {}, + errorTests = {}, + skippedTests = {}, + + failureCount = 0, + errorCount = 0, + notSuccessCount = 0, + skippedCount = 0, + } + + self.outputType = self.outputType or TextOutput + self.output = self.outputType.new(self) + self.output:startSuite() + end + + function M.LuaUnit:startClass( className, classInstance ) + self.result.currentClassName = className + self.output:startClass( className ) + self:setupClass( className, classInstance ) + end + + function M.LuaUnit:startTest( testName ) + self.result.currentTestNumber = self.result.currentTestNumber + 1 + self.result.runCount = self.result.runCount + 1 + self.result.currentNode = NodeStatus.new( + self.result.currentTestNumber, + testName, + self.result.currentClassName + ) + self.result.currentNode.startTime = os.clock() + table.insert( self.result.allTests, self.result.currentNode ) + self.output:startTest( testName ) + end + + function M.LuaUnit:updateStatus( err ) + -- "err" is expected to be a table / result from protectedCall() + if err.status == NodeStatus.SUCCESS then + return + end + + local node = self.result.currentNode + + --[[ As a first approach, we will report only one error or one failure for one test. + + However, we can have the case where the test is in failure, and the teardown is in error. + In such case, it's a good idea to report both a failure and an error in the test suite. This is + what Python unittest does for example. However, it mixes up counts so need to be handled carefully: for + example, there could be more (failures + errors) count that tests. What happens to the current node ? + + We will do this more intelligent version later. + ]] + + -- if the node is already in failure/error, just don't report the new error (see above) + if node.status ~= NodeStatus.SUCCESS then + return + end + + if err.status == NodeStatus.FAIL then + node:fail( err.msg, err.trace ) + table.insert( self.result.failedTests, node ) + elseif err.status == NodeStatus.ERROR then + node:error( err.msg, err.trace ) + table.insert( self.result.errorTests, node ) + elseif err.status == NodeStatus.SKIP then + node:skip( err.msg ) + table.insert( self.result.skippedTests, node ) + else + error('No such status: ' .. prettystr(err.status)) + end + + self.output:updateStatus( node ) + end + + function M.LuaUnit:endTest() + local node = self.result.currentNode + -- print( 'endTest() '..prettystr(node)) + -- print( 'endTest() '..prettystr(node:isNotSuccess())) + node.duration = os.clock() - node.startTime + node.startTime = nil + self.output:endTest( node ) + + if node:isSuccess() then + self.result.successCount = self.result.successCount + 1 + elseif node:isError() then + if self.quitOnError or self.quitOnFailure then + -- Runtime error - abort test execution as requested by + -- "--error" option. This is done by setting a special + -- flag that gets handled in internalRunSuiteByInstances(). + print("\nERROR during LuaUnit test execution:\n" .. node.msg) + self.result.aborted = true + end + elseif node:isFailure() then + if self.quitOnFailure then + -- Failure - abort test execution as requested by + -- "--failure" option. This is done by setting a special + -- flag that gets handled in internalRunSuiteByInstances(). + print("\nFailure during LuaUnit test execution:\n" .. node.msg) + self.result.aborted = true + end + elseif node:isSkipped() then + self.result.runCount = self.result.runCount - 1 + else + error('No such node status: ' .. prettystr(node.status)) + end + self.result.currentNode = nil + end + + function M.LuaUnit:endClass() + self:teardownClass( self.lastClassName, self.lastClassInstance ) + self.output:endClass() + end + + function M.LuaUnit:endSuite() + if self.result.suiteStarted == false then + error('LuaUnit:endSuite() -- suite was already ended' ) + end + self.result.duration = os.clock()-self.result.startTime + self.result.suiteStarted = false + + -- Expose test counts for outputter's endSuite(). This could be managed + -- internally instead by using the length of the lists of failed tests + -- but unit tests rely on these fields being present. + self.result.failureCount = #self.result.failedTests + self.result.errorCount = #self.result.errorTests + self.result.notSuccessCount = self.result.failureCount + self.result.errorCount + self.result.skippedCount = #self.result.skippedTests + + self.output:endSuite() + end + + function M.LuaUnit:setOutputType(outputType, fname) + -- Configures LuaUnit runner output + -- outputType is one of: NIL, TAP, JUNIT, TEXT + -- when outputType is junit, the additional argument fname is used to set the name of junit output file + -- for other formats, fname is ignored + if outputType:upper() == "NIL" then + self.outputType = NilOutput + return + end + if outputType:upper() == "TAP" then + self.outputType = TapOutput + return + end + if outputType:upper() == "JUNIT" then + self.outputType = JUnitOutput + if fname then + self.fname = fname + end + return + end + if outputType:upper() == "TEXT" then + self.outputType = TextOutput + return + end + error( 'No such format: '..outputType,2) + end + + --------------[[ Runner ]]----------------- + + function M.LuaUnit:protectedCall(classInstance, methodInstance, prettyFuncName) + -- if classInstance is nil, this is just a function call + -- else, it's method of a class being called. + + local function err_handler(e) + -- transform error into a table, adding the traceback information + return { + status = NodeStatus.ERROR, + msg = e, + trace = string.sub(debug.traceback("", 1), 2) + } + end + + local ok, err + if classInstance then + -- stupid Lua < 5.2 does not allow xpcall with arguments so let's use a workaround + ok, err = xpcall( function () methodInstance(classInstance) end, err_handler ) + else + ok, err = xpcall( function () methodInstance() end, err_handler ) + end + if ok then + return {status = NodeStatus.SUCCESS} + end + -- print('ok="'..prettystr(ok)..'" err="'..prettystr(err)..'"') + + local iter_msg + iter_msg = self.exeRepeat and 'iteration '..self.currentCount + + err.msg, err.status = M.adjust_err_msg_with_iter( err.msg, iter_msg ) + + if err.status == NodeStatus.SUCCESS or err.status == NodeStatus.SKIP then + err.trace = nil + return err + end + + -- reformat / improve the stack trace + if prettyFuncName then -- we do have the real method name + err.trace = err.trace:gsub("in (%a+) 'methodInstance'", "in %1 '"..prettyFuncName.."'") + end + if STRIP_LUAUNIT_FROM_STACKTRACE then + err.trace = stripLuaunitTrace2(err.trace, err.msg) + end + + return err -- return the error "object" (table) + end + + + function M.LuaUnit:execOneFunction(className, methodName, classInstance, methodInstance) + -- When executing a test function, className and classInstance must be nil + -- When executing a class method, all parameters must be set + + if type(methodInstance) ~= 'function' then + self:unregisterSuite() + error( tostring(methodName)..' must be a function, not '..type(methodInstance)) + end + + local prettyFuncName + if className == nil then + className = '[TestFunctions]' + prettyFuncName = methodName + else + prettyFuncName = className..'.'..methodName + end + + if self.lastClassName ~= className then + if self.lastClassName ~= nil then + self:endClass() + end + self:startClass( className, classInstance ) + self.lastClassName = className + self.lastClassInstance = classInstance + end + + self:startTest(prettyFuncName) + + local node = self.result.currentNode + for iter_n = 1, self.exeRepeat or 1 do + if node:isNotSuccess() then + break + end + self.currentCount = iter_n + + -- run setUp first (if any) + if classInstance then + local func = self.asFunction( classInstance.setUp ) or + self.asFunction( classInstance.Setup ) or + self.asFunction( classInstance.setup ) or + self.asFunction( classInstance.SetUp ) + if func then + self:updateStatus(self:protectedCall(classInstance, func, className..'.setUp')) + end + end + + -- run testMethod() + if node:isSuccess() then + self:updateStatus(self:protectedCall(classInstance, methodInstance, prettyFuncName)) + end + + -- lastly, run tearDown (if any) + if classInstance then + local func = self.asFunction( classInstance.tearDown ) or + self.asFunction( classInstance.TearDown ) or + self.asFunction( classInstance.teardown ) or + self.asFunction( classInstance.Teardown ) + if func then + self:updateStatus(self:protectedCall(classInstance, func, className..'.tearDown')) + end + end + end + + self:endTest() + end + + function M.LuaUnit.expandOneClass( result, className, classInstance ) + --[[ + Input: a list of { name, instance }, a class name, a class instance + Ouptut: modify result to add all test method instance in the form: + { className.methodName, classInstance } + ]] + for methodName, methodInstance in sortedPairs(classInstance) do + if M.LuaUnit.asFunction(methodInstance) and M.LuaUnit.isMethodTestName( methodName ) then + table.insert( result, { className..'.'..methodName, classInstance } ) + end + end + end + + function M.LuaUnit.expandClasses( listOfNameAndInst ) + --[[ + -- expand all classes (provided as {className, classInstance}) to a list of {className.methodName, classInstance} + -- functions and methods remain untouched + + Input: a list of { name, instance } + + Output: + * { function name, function instance } : do nothing + * { class.method name, class instance }: do nothing + * { class name, class instance } : add all method names in the form of (className.methodName, classInstance) + ]] + local result = {} + + for i,v in ipairs( listOfNameAndInst ) do + local name, instance = v[1], v[2] + if M.LuaUnit.asFunction(instance) then + table.insert( result, { name, instance } ) + else + if type(instance) ~= 'table' then + error( 'Instance must be a table or a function, not a '..type(instance)..' with value '..prettystr(instance)) + end + local className, methodName = M.LuaUnit.splitClassMethod( name ) + if className then + local methodInstance = instance[methodName] + if methodInstance == nil then + error( "Could not find method in class "..tostring(className).." for method "..tostring(methodName) ) + end + table.insert( result, { name, instance } ) + else + M.LuaUnit.expandOneClass( result, name, instance ) + end + end + end + + return result + end + + function M.LuaUnit.applyPatternFilter( patternIncFilter, listOfNameAndInst ) + local included, excluded = {}, {} + for i, v in ipairs( listOfNameAndInst ) do + -- local name, instance = v[1], v[2] + if patternFilter( patternIncFilter, v[1] ) then + table.insert( included, v ) + else + table.insert( excluded, v ) + end + end + return included, excluded + end + + local function getKeyInListWithGlobalFallback( key, listOfNameAndInst ) + local result = nil + for i,v in ipairs( listOfNameAndInst ) do + if(listOfNameAndInst[i][1] == key) then + result = listOfNameAndInst[i][2] + break + end + end + if(not M.LuaUnit.asFunction( result ) ) then + result = _G[key] + end + return result + end + + function M.LuaUnit:setupSuite( listOfNameAndInst ) + local setupSuite = getKeyInListWithGlobalFallback("setupSuite", listOfNameAndInst) + if self.asFunction( setupSuite ) then + self:updateStatus( self:protectedCall( nil, setupSuite, 'setupSuite' ) ) + end + end + + function M.LuaUnit:teardownSuite(listOfNameAndInst) + local teardownSuite = getKeyInListWithGlobalFallback("teardownSuite", listOfNameAndInst) + if self.asFunction( teardownSuite ) then + self:updateStatus( self:protectedCall( nil, teardownSuite, 'teardownSuite') ) + end + end + + function M.LuaUnit:setupClass( className, instance ) + if type( instance ) == 'table' and self.asFunction( instance.setupClass ) then + self:updateStatus( self:protectedCall( instance, instance.setupClass, className..'.setupClass' ) ) + end + end + + function M.LuaUnit:teardownClass( className, instance ) + if type( instance ) == 'table' and self.asFunction( instance.teardownClass ) then + self:updateStatus( self:protectedCall( instance, instance.teardownClass, className..'.teardownClass' ) ) + end + end + + function M.LuaUnit:internalRunSuiteByInstances( listOfNameAndInst ) + --[[ Run an explicit list of tests. Each item of the list must be one of: + * { function name, function instance } + * { class name, class instance } + * { class.method name, class instance } + + This function is internal to LuaUnit. The official API to perform this action is runSuiteByInstances() + ]] + + local expandedList = self.expandClasses( listOfNameAndInst ) + if self.shuffle then + randomizeTable( expandedList ) + end + local filteredList, filteredOutList = self.applyPatternFilter( + self.patternIncludeFilter, expandedList ) + + self:startSuite( #filteredList, #filteredOutList ) + self:setupSuite( listOfNameAndInst ) + + for i,v in ipairs( filteredList ) do + local name, instance = v[1], v[2] + if M.LuaUnit.asFunction(instance) then + self:execOneFunction( nil, name, nil, instance ) + else + -- expandClasses() should have already taken care of sanitizing the input + assert( type(instance) == 'table' ) + local className, methodName = M.LuaUnit.splitClassMethod( name ) + assert( className ~= nil ) + local methodInstance = instance[methodName] + assert(methodInstance ~= nil) + self:execOneFunction( className, methodName, instance, methodInstance ) + end + if self.result.aborted then + break -- "--error" or "--failure" option triggered + end + end + + if self.lastClassName ~= nil then + self:endClass() + end + + self:teardownSuite( listOfNameAndInst ) + self:endSuite() + + if self.result.aborted then + print("LuaUnit ABORTED (as requested by --error or --failure option)") + self:unregisterSuite() + os.exit(-2) + end + end + + function M.LuaUnit:internalRunSuiteByNames( listOfName ) + --[[ Run LuaUnit with a list of generic names, coming either from command-line or from global + namespace analysis. Convert the list into a list of (name, valid instances (table or function)) + and calls internalRunSuiteByInstances. + ]] + + local instanceName, instance + local listOfNameAndInst = {} + + for i,name in ipairs( listOfName ) do + local className, methodName = M.LuaUnit.splitClassMethod( name ) + if className then + instanceName = className + instance = _G[instanceName] + + if instance == nil then + self:unregisterSuite() + error( "No such name in global space: "..instanceName ) + end + + if type(instance) ~= 'table' then + self:unregisterSuite() + error( 'Instance of '..instanceName..' must be a table, not '..type(instance)) + end + + local methodInstance = instance[methodName] + if methodInstance == nil then + self:unregisterSuite() + error( "Could not find method in class "..tostring(className).." for method "..tostring(methodName) ) + end + + else + -- for functions and classes + instanceName = name + instance = _G[instanceName] + end + + if instance == nil then + self:unregisterSuite() + error( "No such name in global space: "..instanceName ) + end + + if (type(instance) ~= 'table' and type(instance) ~= 'function') then + self:unregisterSuite() + error( 'Name must match a function or a table: '..instanceName ) + end + + table.insert( listOfNameAndInst, { name, instance } ) + end + + self:internalRunSuiteByInstances( listOfNameAndInst ) + end + + function M.LuaUnit.run(...) + -- Run some specific test classes. + -- If no arguments are passed, run the class names specified on the + -- command line. If no class name is specified on the command line + -- run all classes whose name starts with 'Test' + -- + -- If arguments are passed, they must be strings of the class names + -- that you want to run or generic command line arguments (-o, -p, -v, ...) + local runner = M.LuaUnit.new() + return runner:runSuite(...) + end + + function M.LuaUnit:registerSuite() + -- register the current instance into our global array of instances + -- print('-> Register suite') + M.LuaUnit.instances[ #M.LuaUnit.instances+1 ] = self + end + + function M.unregisterCurrentSuite() + -- force unregister the last registered suite + table.remove(M.LuaUnit.instances, #M.LuaUnit.instances) + end + + function M.LuaUnit:unregisterSuite() + -- print('<- Unregister suite') + -- remove our current instqances from the global array of instances + local instanceIdx = nil + for i, instance in ipairs(M.LuaUnit.instances) do + if instance == self then + instanceIdx = i + break + end + end + + if instanceIdx ~= nil then + table.remove(M.LuaUnit.instances, instanceIdx) + -- print('Unregister done') + end + + end + + function M.LuaUnit:initFromArguments( ... ) + --[[Parses all arguments from either command-line or direct call and set internal + flags of LuaUnit runner according to it. + + Return the list of names which were possibly passed on the command-line or as arguments + ]] + local args = {...} + if type(args[1]) == 'table' and args[1].__class__ == 'LuaUnit' then + -- run was called with the syntax M.LuaUnit:runSuite() + -- we support both M.LuaUnit.run() and M.LuaUnit:run() + -- strip out the first argument self to make it a command-line argument list + table.remove(args,1) + end + + if #args == 0 then + args = cmdline_argv + end + + local options = pcall_or_abort( M.LuaUnit.parseCmdLine, args ) + + -- We expect these option fields to be either `nil` or contain + -- valid values, so it's safe to always copy them directly. + self.verbosity = options.verbosity + self.quitOnError = options.quitOnError + self.quitOnFailure = options.quitOnFailure + + self.exeRepeat = options.exeRepeat + self.patternIncludeFilter = options.pattern + self.shuffle = options.shuffle + + options.output = options.output or os.getenv('LUAUNIT_OUTPUT') + options.fname = options.fname or os.getenv('LUAUNIT_JUNIT_FNAME') + + if options.output then + if options.output:lower() == 'junit' and options.fname == nil then + print('With junit output, a filename must be supplied with -n or --name') + os.exit(-1) + end + pcall_or_abort(self.setOutputType, self, options.output, options.fname) + end + + return options.testNames + end + + function M.LuaUnit:runSuite( ... ) + testNames = self:initFromArguments(...) + self:registerSuite() + self:internalRunSuiteByNames( testNames or M.LuaUnit.collectTests() ) + self:unregisterSuite() + return self.result.notSuccessCount + end + + function M.LuaUnit:runSuiteByInstances( listOfNameAndInst, commandLineArguments ) + --[[ + Run all test functions or tables provided as input. + + Input: a list of { name, instance } + instance can either be a function or a table containing test functions starting with the prefix "test" + + return the number of failures and errors, 0 meaning success + ]] + -- parse the command-line arguments + testNames = self:initFromArguments( commandLineArguments ) + self:registerSuite() + self:internalRunSuiteByInstances( listOfNameAndInst ) + self:unregisterSuite() + return self.result.notSuccessCount + end + + + +-- class LuaUnit + +-- For compatbility with LuaUnit v2 +M.run = M.LuaUnit.run +M.Run = M.LuaUnit.run + +function M:setVerbosity( verbosity ) + -- set the verbosity value (as integer) + M.LuaUnit.verbosity = verbosity +end +M.set_verbosity = M.setVerbosity +M.SetVerbosity = M.setVerbosity + + +return M + diff --git a/reascripts/common/vendor/url.lua b/reascripts/common/vendor/url.lua new file mode 100644 index 0000000000000000000000000000000000000000..ba4ec8ca1aafb918372a5d3690bb93b98aefb84f --- /dev/null +++ b/reascripts/common/vendor/url.lua @@ -0,0 +1,55 @@ +--- Python-style URL quoting library. +-- +-- @module pl.url + +url = (function () + +local url = {} + +local function quote_char(c) + return string.format("%%%02X", string.byte(c)) +end + +--- Quote the url, replacing special characters using the '%xx' escape. +-- @string s the string +-- @bool quote_plus Also escape slashes and replace spaces by plus signs. +-- @return The quoted string, or if `s` wasn't a string, just plain unaltered `s`. +function url.quote(s, quote_plus) + if type(s) ~= "string" then + return s + end + + s = s:gsub("\n", "\r\n") + s = s:gsub("([^A-Za-z0-9 %-_%./])", quote_char) + if quote_plus then + s = s:gsub(" ", "+") + s = s:gsub("/", quote_char) + else + s = s:gsub(" ", "%%20") + end + + return s +end + +local function unquote_char(h) + return string.char(tonumber(h, 16)) +end + +--- Unquote the url, replacing '%xx' escapes and plus signs. +-- @string s the string +-- @return The unquoted string, or if `s` wasn't a string, just plain unaltered `s`. +function url.unquote(s) + if type(s) ~= "string" then + return s + end + + s = s:gsub("+", " ") + s = s:gsub("%%(%x%x)", unquote_char) + s = s:gsub("\r\n", "\n") + + return s +end + +return url + +end)() diff --git a/scripts/.gitattributes b/scripts/.gitattributes new file mode 100644 index 0000000000000000000000000000000000000000..4fbd8ef99ccbff92e517dc20f9d94e92c2698a65 --- /dev/null +++ b/scripts/.gitattributes @@ -0,0 +1,2 @@ +* text eol=lf +*.bat text eol=crlf diff --git a/scripts/reawatch b/scripts/reawatch new file mode 100755 index 0000000000000000000000000000000000000000..01d9d3720bd1f03f8a48675ab11c5c5dd4162dce --- /dev/null +++ b/scripts/reawatch @@ -0,0 +1,2 @@ +#!/bin/sh +docker exec -w /app/reascripts/$1 reaspeech-reaspeech-1 sh -c "fswatch -m poll_monitor source/*.lua tests/*.lua | xargs -I{} make" diff --git a/scripts/reawatch-gpu b/scripts/reawatch-gpu new file mode 100755 index 0000000000000000000000000000000000000000..b9d08584e374a852ae0e2d30675480bf981522c5 --- /dev/null +++ b/scripts/reawatch-gpu @@ -0,0 +1,2 @@ +#!/bin/sh +docker exec -w /app/reascripts/$1 reaspeech-reaspeech-gpu-1 sh -c "fswatch -m poll_monitor source/*.lua tests/*.lua | xargs -I{} make" diff --git a/scripts/reawatch-gpu.bat b/scripts/reawatch-gpu.bat new file mode 100644 index 0000000000000000000000000000000000000000..3824da1fb7a13c69ca0921fe0a36336922b45292 --- /dev/null +++ b/scripts/reawatch-gpu.bat @@ -0,0 +1 @@ +docker exec -w /app/reascripts/%1 reaspeech-reaspeech-gpu-1 sh -c "fswatch -m poll_monitor source/*.lua tests/*.lua | xargs -I{} make" diff --git a/scripts/reawatch.bat b/scripts/reawatch.bat new file mode 100644 index 0000000000000000000000000000000000000000..4f0b9796b0134f0ea47545304da6241355e85534 --- /dev/null +++ b/scripts/reawatch.bat @@ -0,0 +1 @@ +docker exec -w /app/reascripts/%1 reaspeech-reaspeech-1 sh -c "fswatch -m poll_monitor source/*.lua tests/*.lua | xargs -I{} make" diff --git a/tempfile b/tempfile new file mode 100644 index 0000000000000000000000000000000000000000..d114d220f0fd4a4c61df156affe56f2f1367bd59 --- /dev/null +++ b/tempfile @@ -0,0 +1,6 @@ + +301 Moved Permanently + +

301 Moved Permanently

+ +