diff --git a/.env-dev.template b/.env-dev.template index 37d355f..f34bb54 100644 --- a/.env-dev.template +++ b/.env-dev.template @@ -1,4 +1,4 @@ -DB_HOST=postgres +DB_HOST=${POSTGRES_DATABASE_HOST} DB_USER=postgres DB_PASSWORD=postgres DB_NAME=postgres diff --git a/alembic.ini b/alembic.ini new file mode 100644 index 0000000..c10d4ca --- /dev/null +++ b/alembic.ini @@ -0,0 +1,116 @@ +# A generic, single database configuration. + +[alembic] +# path to migration scripts +script_location = alembic + +# template used to generate migration file names; The default value is %%(rev)s_%%(slug)s +# Uncomment the line below if you want the files to be prepended with date and time +# see https://alembic.sqlalchemy.org/en/latest/tutorial.html#editing-the-ini-file +# for all available tokens +# file_template = %%(year)d_%%(month).2d_%%(day).2d_%%(hour).2d%%(minute).2d-%%(rev)s_%%(slug)s + +# sys.path path, will be prepended to sys.path if present. +# defaults to the current working directory. +prepend_sys_path = . + +# timezone to use when rendering the date within the migration file +# as well as the filename. +# If specified, requires the python>=3.9 or backports.zoneinfo library. +# Any required deps can installed by adding `alembic[tz]` to the pip requirements +# string value is passed to ZoneInfo() +# leave blank for localtime +# timezone = + +# max length of characters to apply to the +# "slug" field +# truncate_slug_length = 40 + +# set to 'true' to run the environment during +# the 'revision' command, regardless of autogenerate +# revision_environment = false + +# set to 'true' to allow .pyc and .pyo files without +# a source .py file to be detected as revisions in the +# versions/ directory +# sourceless = false + +# version location specification; This defaults +# to alembic/versions. When using multiple version +# directories, initial revisions must be specified with --version-path. +# The path separator used here should be the separator specified by "version_path_separator" below. +# version_locations = %(here)s/bar:%(here)s/bat:alembic/versions + +# version path separator; As mentioned above, this is the character used to split +# version_locations. The default within new alembic.ini files is "os", which uses os.pathsep. +# If this key is omitted entirely, it falls back to the legacy behavior of splitting on spaces and/or commas. +# Valid values for version_path_separator are: +# +# version_path_separator = : +# version_path_separator = ; +# version_path_separator = space +version_path_separator = os # Use os.pathsep. Default configuration used for new projects. + +# set to 'true' to search source files recursively +# in each "version_locations" directory +# new in Alembic version 1.10 +# recursive_version_locations = false + +# the output encoding used when revision files +# are written from script.py.mako +# output_encoding = utf-8 + +sqlalchemy.url = driver://user:pass@localhost/dbname + + +[post_write_hooks] +# post_write_hooks defines scripts or Python functions that are run +# on newly generated revision scripts. See the documentation for further +# detail and examples + +# format using "black" - use the console_scripts runner, against the "black" entrypoint +# hooks = black +# black.type = console_scripts +# black.entrypoint = black +# black.options = -l 79 REVISION_SCRIPT_FILENAME + +# lint with attempts to fix using "ruff" - use the exec runner, execute a binary +# hooks = ruff +# ruff.type = exec +# ruff.executable = %(here)s/.venv/bin/ruff +# ruff.options = --fix REVISION_SCRIPT_FILENAME + +# Logging configuration +[loggers] +keys = root,sqlalchemy,alembic + +[handlers] +keys = console + +[formatters] +keys = generic + +[logger_root] +level = WARN +handlers = console +qualname = + +[logger_sqlalchemy] +level = WARN +handlers = +qualname = sqlalchemy.engine + +[logger_alembic] +level = INFO +handlers = +qualname = alembic + +[handler_console] +class = StreamHandler +args = (sys.stderr,) +level = NOTSET +formatter = generic + +[formatter_generic] +format = %(levelname)-5.5s [%(name)s] %(message)s +datefmt = %H:%M:%S diff --git a/alembic/env.py b/alembic/env.py new file mode 100644 index 0000000..e3f3954 --- /dev/null +++ b/alembic/env.py @@ -0,0 +1,90 @@ +from logging.config import fileConfig + +from sqlalchemy import engine_from_config +from sqlalchemy import pool + +from alembic import context + +import os +from server.app.models import Base + +# this is the Alembic Config object, which provides +# access to the values within the .ini file in use. +config = context.config + +# Interpret the config file for Python logging. +# This line sets up loggers basically. +if config.config_file_name is not None: + fileConfig(config.config_file_name) + +# add your model's MetaData object here +# for 'autogenerate' support +# from myapp import mymodel +# target_metadata = mymodel.Base.metadata +target_metadata = Base.metadata + +# other values from the config, defined by the needs of env.py, +# can be acquired: +# my_important_option = config.get_main_option("my_important_option") +# ... etc. + +db_name = os.getenv("DB_NAME") +db_user = os.getenv("DB_USER") +db_password = os.getenv("DB_PASSWORD") +db_host = os.getenv("DB_HOST") +db_port = os.getenv("DB_PORT") + +# Construct the SQLAlchemy connection URL using URL class +db_url = f"postgresql://{db_user}:{db_password}@{db_host}:{db_port}/{db_name}" +config.set_main_option('sqlalchemy.url', db_url) + +def run_migrations_offline() -> None: + """Run migrations in 'offline' mode. + + This configures the context with just a URL + and not an Engine, though an Engine is acceptable + here as well. By skipping the Engine creation + we don't even need a DBAPI to be available. + + Calls to context.execute() here emit the given string to the + script output. + + """ + url = config.get_main_option("sqlalchemy.url") + context.configure( + url=url, + target_metadata=target_metadata, + literal_binds=True, + dialect_opts={"paramstyle": "named"}, + ) + + with context.begin_transaction(): + context.run_migrations() + + +def run_migrations_online() -> None: + """Run migrations in 'online' mode. + + In this scenario we need to create an Engine + and associate a connection with the context. + + """ + connectable = engine_from_config( + config.get_section(config.config_ini_section, {}), + prefix="sqlalchemy.", + poolclass=pool.NullPool, + ) + + with connectable.connect() as connection: + context.configure( + connection=connection, target_metadata=target_metadata + ) + + with context.begin_transaction(): + context.run_migrations() + + +if context.is_offline_mode(): + run_migrations_offline() +else: + run_migrations_online() diff --git a/alembic/script.py.mako b/alembic/script.py.mako new file mode 100644 index 0000000..fbc4b07 --- /dev/null +++ b/alembic/script.py.mako @@ -0,0 +1,26 @@ +"""${message} + +Revision ID: ${up_revision} +Revises: ${down_revision | comma,n} +Create Date: ${create_date} + +""" +from typing import Sequence, Union + +from alembic import op +import sqlalchemy as sa +${imports if imports else ""} + +# revision identifiers, used by Alembic. +revision: str = ${repr(up_revision)} +down_revision: Union[str, None] = ${repr(down_revision)} +branch_labels: Union[str, Sequence[str], None] = ${repr(branch_labels)} +depends_on: Union[str, Sequence[str], None] = ${repr(depends_on)} + + +def upgrade() -> None: + ${upgrades if upgrades else "pass"} + + +def downgrade() -> None: + ${downgrades if downgrades else "pass"} diff --git a/alembic/versions/3e403863733c_initial_revision.py b/alembic/versions/3e403863733c_initial_revision.py new file mode 100644 index 0000000..f25da7f --- /dev/null +++ b/alembic/versions/3e403863733c_initial_revision.py @@ -0,0 +1,276 @@ +"""Initial Revision + +Revision ID: 3e403863733c +Revises: +Create Date: 2024-06-13 17:52:24.176257 + +""" +from typing import Sequence, Union + +from alembic import op +import sqlalchemy as sa + + +# revision identifiers, used by Alembic. +revision: str = '3e403863733c' +down_revision: Union[str, None] = None +branch_labels: Union[str, Sequence[str], None] = None +depends_on: Union[str, Sequence[str], None] = None + + +def upgrade() -> None: + # ### commands auto generated by Alembic - please adjust! ### + op.create_table('plugins', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.Column('description', sa.String(), nullable=True), + sa.Column('pypi_url', sa.String(), nullable=True), + sa.Column('dsl', sa.String(), nullable=True), + sa.Column('doc', sa.String(), nullable=True), + sa.Column('icon', sa.String(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.Column('updated_at', sa.DateTime(), nullable=True), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_plugins_id'), 'plugins', ['id'], unique=False) + op.create_index(op.f('ix_plugins_name'), 'plugins', ['name'], unique=False) + op.create_table('projects', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.Column('description', sa.String(), nullable=True), + sa.Column('project_class', sa.String(), nullable=True), + sa.Column('parent_id', sa.Integer(), nullable=True), + sa.Column('output_url', sa.String(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.Column('updated_at', sa.DateTime(), nullable=True), + sa.Column('is_template', sa.Boolean(), nullable=True), + sa.Column('is_deleted', sa.Boolean(), nullable=True), + sa.Column('engine_url', sa.Boolean(), nullable=True), + sa.Column('file_upload', sa.Boolean(), nullable=True), + sa.Column('audio_upload', sa.Boolean(), nullable=True), + sa.Column('has_credentials', sa.Boolean(), nullable=True), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_projects_id'), 'projects', ['id'], unique=False) + op.create_index(op.f('ix_projects_name'), 'projects', ['name'], unique=False) + op.create_table('tags', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_tags_id'), 'tags', ['id'], unique=False) + op.create_index(op.f('ix_tags_name'), 'tags', ['name'], unique=False) + op.create_table('templates', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.Column('description', sa.String(), nullable=True), + sa.Column('project_class', sa.String(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.Column('updated_at', sa.DateTime(), nullable=True), + sa.Column('engine_url', sa.Boolean(), nullable=True), + sa.Column('file_upload', sa.Boolean(), nullable=True), + sa.Column('audio_upload', sa.Boolean(), nullable=True), + sa.Column('has_credentials', sa.Boolean(), nullable=True), + sa.Column('verified', sa.Boolean(), nullable=True), + sa.Column('sort', sa.Integer(), nullable=True), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_templates_id'), 'templates', ['id'], unique=False) + op.create_index(op.f('ix_templates_name'), 'templates', ['name'], unique=False) + op.create_table('users', + sa.Column('oid', sa.String(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.Column('email', sa.String(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.PrimaryKeyConstraint('oid') + ) + op.create_index(op.f('ix_users_oid'), 'users', ['oid'], unique=False) + op.create_table('analytics_logs', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('user_id', sa.String(), nullable=True), + sa.Column('representation_name', sa.String(), nullable=True), + sa.Column('data', sa.JSON(), nullable=True), + sa.Column('timestamp', sa.DateTime(), nullable=True), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.ForeignKeyConstraint(['user_id'], ['users.oid'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_analytics_logs_id'), 'analytics_logs', ['id'], unique=False) + op.create_index(op.f('ix_analytics_logs_name'), 'analytics_logs', ['name'], unique=False) + op.create_index(op.f('ix_analytics_logs_project_id'), 'analytics_logs', ['project_id'], unique=False) + op.create_index(op.f('ix_analytics_logs_user_id'), 'analytics_logs', ['user_id'], unique=False) + op.create_table('dev_chat_messages', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('session_id', sa.String(), nullable=True), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('sender_id', sa.String(), nullable=True), + sa.Column('sender', sa.Enum('user', 'bot', name='chat_user_type'), nullable=True), + sa.Column('type', sa.Enum('instruction', 'feedback', 'thought', 'output', 'start', 'dsl_state', 'end', 'error', 'representation_edit', 'files', name='chat_message_type'), nullable=True), + sa.Column('message', sa.String(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.Column('special_link', sa.String(), nullable=True), + sa.Column('pbyc_comments', sa.String(), nullable=True), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_dev_chat_messages_id'), 'dev_chat_messages', ['id'], unique=False) + op.create_index(op.f('ix_dev_chat_messages_project_id'), 'dev_chat_messages', ['project_id'], unique=False) + op.create_table('exports', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('sandbox_responsecode', sa.Integer(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_exports_id'), 'exports', ['id'], unique=False) + op.create_index(op.f('ix_exports_project_id'), 'exports', ['project_id'], unique=False) + op.create_table('output_chat_messages', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('session_id', sa.String(), nullable=True), + sa.Column('sender_id', sa.String(), nullable=True), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('sender', sa.Enum('user', 'bot', name='output_user_type'), nullable=True), + sa.Column('type', sa.Enum('instruction', 'feedback', 'thought', 'output', 'callback', 'debug', 'start', 'end', 'error', name='output_message_type'), nullable=True), + sa.Column('message', sa.String(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.Column('special_link', sa.String(), nullable=True), + sa.Column('pbyc_comments', sa.String(), nullable=True), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_output_chat_messages_id'), 'output_chat_messages', ['id'], unique=False) + op.create_index(op.f('ix_output_chat_messages_project_id'), 'output_chat_messages', ['project_id'], unique=False) + op.create_table('project_has_credentials', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('name', sa.String(), nullable=True), + sa.Column('description', sa.String(), nullable=True), + sa.Column('mandatory', sa.Boolean(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_project_has_credentials_id'), 'project_has_credentials', ['id'], unique=False) + op.create_index(op.f('ix_project_has_credentials_project_id'), 'project_has_credentials', ['project_id'], unique=False) + op.create_table('representation', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('name', sa.String(), nullable=True), + sa.Column('type', sa.String(), nullable=True), + sa.Column('text', sa.String(), nullable=True), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('is_editable', sa.Boolean(), nullable=True), + sa.Column('sort_order', sa.Integer(), nullable=True), + sa.Column('is_pbyc_viewable', sa.Boolean(), nullable=True), + sa.Column('is_user_viewable', sa.Boolean(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_representation_id'), 'representation', ['id'], unique=False) + op.create_index(op.f('ix_representation_name'), 'representation', ['name'], unique=False) + op.create_index(op.f('ix_representation_project_id'), 'representation', ['project_id'], unique=False) + op.create_table('template_has_credentials', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('template_id', sa.Integer(), nullable=True), + sa.Column('name', sa.String(), nullable=True), + sa.Column('description', sa.String(), nullable=True), + sa.Column('mandatory', sa.Boolean(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.ForeignKeyConstraint(['template_id'], ['templates.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_template_has_credentials_id'), 'template_has_credentials', ['id'], unique=False) + op.create_index(op.f('ix_template_has_credentials_template_id'), 'template_has_credentials', ['template_id'], unique=False) + op.create_table('template_has_plugins', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('template_id', sa.Integer(), nullable=True), + sa.Column('plugin_id', sa.Integer(), nullable=True), + sa.ForeignKeyConstraint(['plugin_id'], ['plugins.id'], ), + sa.ForeignKeyConstraint(['template_id'], ['templates.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_template_has_plugins_id'), 'template_has_plugins', ['id'], unique=False) + op.create_index(op.f('ix_template_has_plugins_plugin_id'), 'template_has_plugins', ['plugin_id'], unique=False) + op.create_index(op.f('ix_template_has_plugins_template_id'), 'template_has_plugins', ['template_id'], unique=False) + op.create_table('template_has_tags', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('template_id', sa.Integer(), nullable=True), + sa.Column('tag_id', sa.Integer(), nullable=True), + sa.ForeignKeyConstraint(['tag_id'], ['tags.id'], ), + sa.ForeignKeyConstraint(['template_id'], ['templates.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_template_has_tags_id'), 'template_has_tags', ['id'], unique=False) + op.create_index(op.f('ix_template_has_tags_tag_id'), 'template_has_tags', ['tag_id'], unique=False) + op.create_index(op.f('ix_template_has_tags_template_id'), 'template_has_tags', ['template_id'], unique=False) + op.create_table('users_has_projects', + sa.Column('id', sa.Integer(), nullable=False), + sa.Column('oid', sa.String(), nullable=True), + sa.Column('project_id', sa.Integer(), nullable=True), + sa.Column('created_at', sa.DateTime(), nullable=True), + sa.ForeignKeyConstraint(['oid'], ['users.oid'], ), + sa.ForeignKeyConstraint(['project_id'], ['projects.id'], ), + sa.PrimaryKeyConstraint('id') + ) + op.create_index(op.f('ix_users_has_projects_id'), 'users_has_projects', ['id'], unique=False) + op.create_index(op.f('ix_users_has_projects_oid'), 'users_has_projects', ['oid'], unique=False) + op.create_index(op.f('ix_users_has_projects_project_id'), 'users_has_projects', ['project_id'], unique=False) + # ### end Alembic commands ### + + +def downgrade() -> None: + # ### commands auto generated by Alembic - please adjust! ### + op.drop_index(op.f('ix_users_has_projects_project_id'), table_name='users_has_projects') + op.drop_index(op.f('ix_users_has_projects_oid'), table_name='users_has_projects') + op.drop_index(op.f('ix_users_has_projects_id'), table_name='users_has_projects') + op.drop_table('users_has_projects') + op.drop_index(op.f('ix_template_has_tags_template_id'), table_name='template_has_tags') + op.drop_index(op.f('ix_template_has_tags_tag_id'), table_name='template_has_tags') + op.drop_index(op.f('ix_template_has_tags_id'), table_name='template_has_tags') + op.drop_table('template_has_tags') + op.drop_index(op.f('ix_template_has_plugins_template_id'), table_name='template_has_plugins') + op.drop_index(op.f('ix_template_has_plugins_plugin_id'), table_name='template_has_plugins') + op.drop_index(op.f('ix_template_has_plugins_id'), table_name='template_has_plugins') + op.drop_table('template_has_plugins') + op.drop_index(op.f('ix_template_has_credentials_template_id'), table_name='template_has_credentials') + op.drop_index(op.f('ix_template_has_credentials_id'), table_name='template_has_credentials') + op.drop_table('template_has_credentials') + op.drop_index(op.f('ix_representation_project_id'), table_name='representation') + op.drop_index(op.f('ix_representation_name'), table_name='representation') + op.drop_index(op.f('ix_representation_id'), table_name='representation') + op.drop_table('representation') + op.drop_index(op.f('ix_project_has_credentials_project_id'), table_name='project_has_credentials') + op.drop_index(op.f('ix_project_has_credentials_id'), table_name='project_has_credentials') + op.drop_table('project_has_credentials') + op.drop_index(op.f('ix_output_chat_messages_project_id'), table_name='output_chat_messages') + op.drop_index(op.f('ix_output_chat_messages_id'), table_name='output_chat_messages') + op.drop_table('output_chat_messages') + op.drop_index(op.f('ix_exports_project_id'), table_name='exports') + op.drop_index(op.f('ix_exports_id'), table_name='exports') + op.drop_table('exports') + op.drop_index(op.f('ix_dev_chat_messages_project_id'), table_name='dev_chat_messages') + op.drop_index(op.f('ix_dev_chat_messages_id'), table_name='dev_chat_messages') + op.drop_table('dev_chat_messages') + op.drop_index(op.f('ix_analytics_logs_user_id'), table_name='analytics_logs') + op.drop_index(op.f('ix_analytics_logs_project_id'), table_name='analytics_logs') + op.drop_index(op.f('ix_analytics_logs_name'), table_name='analytics_logs') + op.drop_index(op.f('ix_analytics_logs_id'), table_name='analytics_logs') + op.drop_table('analytics_logs') + op.drop_index(op.f('ix_users_oid'), table_name='users') + op.drop_table('users') + op.drop_index(op.f('ix_templates_name'), table_name='templates') + op.drop_index(op.f('ix_templates_id'), table_name='templates') + op.drop_table('templates') + op.drop_index(op.f('ix_tags_name'), table_name='tags') + op.drop_index(op.f('ix_tags_id'), table_name='tags') + op.drop_table('tags') + op.drop_index(op.f('ix_projects_name'), table_name='projects') + op.drop_index(op.f('ix_projects_id'), table_name='projects') + op.drop_table('projects') + op.drop_index(op.f('ix_plugins_name'), table_name='plugins') + op.drop_index(op.f('ix_plugins_id'), table_name='plugins') + op.drop_table('plugins') + # ### end Alembic commands ### diff --git a/docker-compose.yml b/docker-compose.yml index b99f652..ac4cfb6 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -12,6 +12,10 @@ services: - ./server/app:/code/server/app environment: - DB_CONNECTION_STRING=postgresql://${DB_USER}:${DB_PASSWORD}@${DB_HOST}:${DB_PORT}/${DB_NAME} + - DB_USER=${DB_USER} + - DB_PASSWORD=${DB_PASSWORD} + - DB_HOST=${DB_HOST} + - DB_PORT=${DB_PORT} - KAFKA_BROKER=${KAFKA_BROKER} - KAFKA_USE_SASL=${KAFKA_USE_SASL} - KAFKA_PRODUCER_USERNAME=${KAFKA_PRODUCER_USERNAME} diff --git a/scripts/backup.sql b/scripts/backup.sql index 361f7fe..0d8f0a2 100644 --- a/scripts/backup.sql +++ b/scripts/backup.sql @@ -1327,4 +1327,3 @@ ALTER TABLE ONLY public.users_has_projects -- -- PostgreSQL database dump complete -- - diff --git a/scripts/base_queries.sql b/scripts/base_queries.sql deleted file mode 100644 index bf170aa..0000000 --- a/scripts/base_queries.sql +++ /dev/null @@ -1,14 +0,0 @@ -INSERT INTO public.templates(id, name, description, project_class, created_at, updated_at, engine_url, file_upload, audio_upload, has_credentials, verified, sort) VALUES (1, 'Custom Template', 'Use this to create your own template. You can see an example here: https://dev.azure.com/pbyc/PbyC/_git/Examples','custom', '2023-05-10 07:32:03.987439', '2023-05-04 13:47:56.492011', true, true, true, false, false, -1); - -INSERT INTO public.templates(id, name, description, project_class, created_at, updated_at, engine_url, file_upload, audio_upload, has_credentials, verified, sort) VALUES (2, 'JB App','Use this to create apps using JugalBandi','jb_app', '2023-05-10 07:32:03.987439', '2023-05-04 13:47:56.492011', false, false, false, false, true, 0); - -INSERT INTO public.plugins(id, name, description, pypi_url, dsl, doc, icon, created_at, updated_at) VALUES (1, 'Knowledge Base', 'Fetch data from a trusted or validated knowledgebase', 'https://pwrplugins.z29.web.core.windows.net/plugins/knowledgebase/rag.dsl', '', '', '', '2023-11-24 09:05:53.86371', '2023-11-24 09:05:53.86371'); - -INSERT INTO public.plugins(id, name, description, pypi_url, dsl, doc, icon, created_at, updated_at) VALUES (2, 'Reasoning Engine', 'Using advanced capabilities of AI models (LLMs) to reason over knowledge based on a task', 'https://pwrplugins.z29.web.core.windows.net/plugins/reason_engine/llm.dsl' , '', '', '','2023-11-24 09:05:53.86371' , '2023-11-24 09:05:53.86371'); - - -INSERT INTO public.template_has_plugins( - id, template_id, plugin_id) - VALUES (1, 2, 1); - -INSERT INTO public.template_has_plugins(id, template_id, plugin_id) VALUES (1, 2, 2); \ No newline at end of file diff --git a/scripts/create-migration.sh b/scripts/create-migration.sh new file mode 100644 index 0000000..25886db --- /dev/null +++ b/scripts/create-migration.sh @@ -0,0 +1,26 @@ +#!/bin/bash + +msg=$1 + +python3 -m venv .venv +source .venv/bin/activate +pip3 install alembic psycopg2-binary + + +if grep -qi microsoft /proc/version && grep -q WSL2 /proc/version; then + PWRHOST=$(hostname -I | awk '{print $1}') + export PWR_KAFKA_BROKER=${PWRHOST}:9092 + export POSTGRES_DATABASE_HOST=${PWRHOST} + echo "Setting Kakfa & Postgres host by WSL2 IP: ${PWRHOST}" +else + export PWR_KAFKA_BROKER=kafka:9092 + export POSTGRES_DATABASE_HOST=postgres + echo "Setting Kakfa & Postgres host by docker-compose service name: kafka, postgres" +fi + + +set -a +source .env-dev +set +a + +alembic revision --autogenerate -m "$msg" \ No newline at end of file diff --git a/scripts/upgrade-db.sh b/scripts/upgrade-db.sh index d78343b..3d61df0 100644 --- a/scripts/upgrade-db.sh +++ b/scripts/upgrade-db.sh @@ -4,16 +4,15 @@ python3 -m venv .venv source .venv/bin/activate pip3 install alembic psycopg2-binary -# source ./set_jbhost.sh if grep -qi microsoft /proc/version && grep -q WSL2 /proc/version; then - JBHOST=$(hostname -I | awk '{print $1}') - export JB_KAFKA_BROKER=${JBHOST}:9092 - export JB_POSTGRES_DATABASE_HOST=${JBHOST} - echo "Setting Kakfa & Postgres host by WSL2 IP: ${JBHOST}" + PWRHOST=$(hostname -I | awk '{print $1}') + export PWR_KAFKA_BROKER=${PWRHOST}:9092 + export POSTGRES_DATABASE_HOST=${PWRHOST} + echo "Setting Kakfa & Postgres host by WSL2 IP: ${PWRHOST}" else - export JB_KAFKA_BROKER=kafka:9092 - export JB_POSTGRES_DATABASE_HOST=localhost + export PWR_KAFKA_BROKER=kafka:9092 + export POSTGRES_DATABASE_HOST=localhost echo "Setting Kakfa & Postgres host by docker-compose service name: kafka, postgres" fi diff --git a/server/app/alembic/env.py b/server/app/alembic/env.py index 180c60b..740fed2 100644 --- a/server/app/alembic/env.py +++ b/server/app/alembic/env.py @@ -18,7 +18,16 @@ # this is the Alembic Config object, which provides # access to the values within the .ini file in use. config = context.config -connection_string=os.environ['DB_CONNECTION_STRING'] +if os.environ.get("DB_CONNECTION_STRING"): + connection_string=os.environ['DB_CONNECTION_STRING'] +else: + # construct the postgres connection string from environment variables + db_user = os.getenv("DB_USER") + db_password = os.getenv("DB_PASSWORD") + db_host = os.getenv("DB_HOST") + db_port = os.getenv("DB_PORT") + db_name = os.getenv("DB_NAME") + connection_string = f"postgresql://{db_user}:{db_password}@{db_host}:{db_port}/{db_name}" config.set_main_option('sqlalchemy.url', connection_string) # Interpret the config file for Python logging. diff --git a/server/app/database.py b/server/app/database.py index 1d04a6c..6286002 100644 --- a/server/app/database.py +++ b/server/app/database.py @@ -6,11 +6,17 @@ load_dotenv() -SQLALCHEMY_DATABASE_URL = "sqlite:///./sql_app.db" +SQLALCHEMY_DATABASE_URL = "" if os.environ.get("DB_CONNECTION_STRING"): SQLALCHEMY_DATABASE_URL = os.environ["DB_CONNECTION_STRING"] else: - print("DB_CONNECTION_STRING not set. Using default sqlite database.") + # construct the postgres connection string from environment variables + db_user = os.getenv("DB_USER") + db_password = os.getenv("DB_PASSWORD") + db_host = os.getenv("DB_HOST") + db_port = os.getenv("DB_PORT") + db_name = os.getenv("DB_NAME") + SQLALCHEMY_DATABASE_URL = f"postgresql://{db_user}:{db_password}@{db_host}:{db_port}/{db_name}" connect_args = {} if SQLALCHEMY_DATABASE_URL.startswith("sqlite"): diff --git a/server/app/main.py b/server/app/main.py index 7fcb80b..d5ebc2e 100644 --- a/server/app/main.py +++ b/server/app/main.py @@ -42,7 +42,7 @@ # import sys # print(os.environ.get('DB_CONNECTION_STRING'), file=sys.stderr) -models.Base.metadata.create_all(bind=engine) +# models.Base.metadata.create_all(bind=engine) logger = logging.getLogger(__name__) logger.setLevel(level=logging.INFO)