From 34c9366f2b15288948824730c2c515d48d182601 Mon Sep 17 00:00:00 2001 From: Augustin Date: Wed, 5 Jan 2022 21:19:40 +0100 Subject: [PATCH] =?UTF-8?q?=F0=9F=90=99=20=20Bootstrap=20octavia-cli=20pro?= =?UTF-8?q?ject=20(#9070)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .github/workflows/gradle.yml | 70 +++++++++++++++++++++++ .gitignore | 1 + octavia-cli/.coveragerc | 3 + octavia-cli/.dockerignore | 3 + octavia-cli/.gitignore | 2 + octavia-cli/.python-version | 1 + octavia-cli/CHANGELOG.md | 0 octavia-cli/Dockerfile | 17 ++++++ octavia-cli/LICENSE | 21 +++++++ octavia-cli/README.md | 55 ++++++++++++++++++ octavia-cli/build.gradle | 9 +++ octavia-cli/octavia_cli/__init__.py | 0 octavia-cli/octavia_cli/entrypoint.py | 42 ++++++++++++++ octavia-cli/pytest.ini | 5 ++ octavia-cli/setup.py | 53 +++++++++++++++++ octavia-cli/unit_tests/test_entrypoint.py | 25 ++++++++ settings.gradle | 22 ++++--- 17 files changed, 320 insertions(+), 9 deletions(-) create mode 100644 octavia-cli/.coveragerc create mode 100644 octavia-cli/.dockerignore create mode 100644 octavia-cli/.gitignore create mode 100644 octavia-cli/.python-version create mode 100644 octavia-cli/CHANGELOG.md create mode 100644 octavia-cli/Dockerfile create mode 100644 octavia-cli/LICENSE create mode 100644 octavia-cli/README.md create mode 100644 octavia-cli/build.gradle create mode 100644 octavia-cli/octavia_cli/__init__.py create mode 100644 octavia-cli/octavia_cli/entrypoint.py create mode 100644 octavia-cli/pytest.ini create mode 100644 octavia-cli/setup.py create mode 100644 octavia-cli/unit_tests/test_entrypoint.py diff --git a/.github/workflows/gradle.yml b/.github/workflows/gradle.yml index 836b89f758a1..adc1ad47b14d 100644 --- a/.github/workflows/gradle.yml +++ b/.github/workflows/gradle.yml @@ -275,6 +275,76 @@ jobs: SLACK_TITLE: "Build failure" SLACK_FOOTER: "" + - name: Slack Notification - Success + if: success() && github.ref == 'refs/heads/master' + uses: rtCamp/action-slack-notify@master + env: + SLACK_WEBHOOK: ${{ secrets.BUILD_SLACK_WEBHOOK }} + SLACK_USERNAME: Buildbot + SLACK_TITLE: "Build Success" + SLACK_FOOTER: "" + octavia-cli-build: + runs-on: ubuntu-latest + name: "Octavia CLI: Build" + timeout-minutes: 90 + steps: + - name: Checkout Airbyte + uses: actions/checkout@v2 + + # this intentionally does not use restore-keys so we don't mess with gradle caching + - name: Gradle Caching + uses: actions/cache@v2 + with: + path: | + ~/.gradle/caches + ~/.gradle/wrapper + **/.venv + key: ${{ secrets.CACHE_VERSION }}-${{ runner.os }}-${{ hashFiles('**/*.gradle*') }}-${{ hashFiles('**/package-lock.json') }} + + - uses: actions/setup-java@v1 + with: + java-version: "17" + + # octavia-cli install and testing requires Python. + # We use 3.8 in this project because 3.7 is not supported on Apple M1. + - uses: actions/setup-python@v2 + with: + python-version: "3.8" + + - name: Set up CI Gradle Properties + run: | + mkdir -p ~/.gradle/ + cat > ~/.gradle/gradle.properties <> ~/.zshrc +source ~/.zshrc +octavia +```` + +# Current development status +Octavia is currently under development. +You can find a detailed and updated execution plan [here](https://docs.google.com/spreadsheets/d/1weB9nf0Zx3IR_QvpkxtjBAzyfGb7B0PWpsVt6iMB5Us/edit#gid=0). +We welcome community contributions! + +Summary of achievements: + +| Date | Milestone | +|------------|-------------------------------------| +| 2021-12-22 | Bootstrapping the project's code base | + +# Developing locally +1. Install Python 3.10.0. We suggest doing it through `pyenv` +2. Create a virtualenv: `python -m venv .venv` +3. Activate the virtualenv: `source .venv/bin/activate` +4. Install dev dependencies: `pip install -e .\[dev\]` +5. Install `pre-commit` hooks: `pre-commit install` +6. Run the test suite: `pytest --cov=octavia_cli unit_tests` +7. Iterate; please check the [Contributing](#contributing) for instructions on contributing. + +# Contributing +1. Please sign up to [Airbyte's Slack workspace](https://slack.airbyte.io/) and join the `#octavia-cli`. We'll sync up community efforts in this channel. +2. Read the [execution plan](https://docs.google.com/spreadsheets/d/1weB9nf0Zx3IR_QvpkxtjBAzyfGb7B0PWpsVt6iMB5Us/edit#gid=0) and find a task you'd like to work on. +3. Open a PR, make sure to test your code thoroughly. \ No newline at end of file diff --git a/octavia-cli/build.gradle b/octavia-cli/build.gradle new file mode 100644 index 000000000000..5ebd064d689b --- /dev/null +++ b/octavia-cli/build.gradle @@ -0,0 +1,9 @@ +plugins { + id 'airbyte-python' + id 'airbyte-docker' +} + +airbytePython { + moduleDirectory 'octavia_cli' +} + diff --git a/octavia-cli/octavia_cli/__init__.py b/octavia-cli/octavia_cli/__init__.py new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/octavia-cli/octavia_cli/entrypoint.py b/octavia-cli/octavia_cli/entrypoint.py new file mode 100644 index 000000000000..3d82bc32f5ea --- /dev/null +++ b/octavia-cli/octavia_cli/entrypoint.py @@ -0,0 +1,42 @@ +# +# Copyright (c) 2021 Airbyte, Inc., all rights reserved. +# + +import click + + +@click.group() +@click.option("--airbyte-url", envvar="AIRBYTE_URL", default="http://localhost:8000", help="The URL of your Airbyte instance.") +def octavia(airbyte_url): + # TODO: check if the airbyte_url is reachable + click.secho(f"🐙 - Octavia is targetting your Airbyte instance running at {airbyte_url}") + + +@octavia.command(help="Scaffolds a local project directories.") +def init(): + raise click.ClickException("The init command is not yet implemented.") + + +@octavia.command(name="list", help="List existing resources on the Airbyte instance.") +def _list(): + raise click.ClickException("The init command is not yet implemented.") + + +@octavia.command(name="import", help="Import an existing resources from the Airbyte instance.") +def _import(): + raise click.ClickException("The init command is not yet implemented.") + + +@octavia.command(help="Generate a YAML configuration file to manage a resource.") +def create(): + raise click.ClickException("The init command is not yet implemented.") + + +@octavia.command(help="Create or update resources according to YAML configurations.") +def apply(): + raise click.ClickException("The init command is not yet implemented.") + + +@octavia.command(help="Delete resources") +def delete(): + raise click.ClickException("The init command is not yet implemented.") diff --git a/octavia-cli/pytest.ini b/octavia-cli/pytest.ini new file mode 100644 index 000000000000..f5276a231cad --- /dev/null +++ b/octavia-cli/pytest.ini @@ -0,0 +1,5 @@ +[pytest] +log_cli = 1 +log_cli_level = INFO +log_cli_format = %(asctime)s [%(levelname)8s] %(message)s (%(filename)s:%(lineno)s) +log_cli_date_format=%Y-%m-%d %H:%M:%S diff --git a/octavia-cli/setup.py b/octavia-cli/setup.py new file mode 100644 index 000000000000..93534a6e851a --- /dev/null +++ b/octavia-cli/setup.py @@ -0,0 +1,53 @@ +# +# Copyright (c) 2021 Airbyte, Inc., all rights reserved. +# + +import pathlib + +from setuptools import find_packages, setup + +# The directory containing this file +HERE = pathlib.Path(__file__).parent + +# The text of the README file +README = (HERE / "README.md").read_text() + +setup( + name="octavia-cli", + version="0.1.0", + description="A command line interface to manage Airbyte configurations", + long_description=README, + author="Airbyte", + author_email="contact@airbyte.io", + license="MIT", + url="https://github.com/airbytehq/airbyte", + classifiers=[ + # This information is used when browsing on PyPi. + # Dev Status + "Development Status :: 3 - Alpha", + # Project Audience + "Intended Audience :: Developers", + "Topic :: Scientific/Engineering", + "Topic :: Software Development :: Libraries :: Python Modules", + "License :: OSI Approved :: MIT License", + # Python Version Support + "Programming Language :: Python :: 3.10", + ], + keywords="airbyte cli command-line-interface configuration", + project_urls={ + "Documentation": "https://docs.airbyte.io/", + "Source": "https://github.com/airbytehq/airbyte", + "Tracker": "https://github.com/airbytehq/airbyte/issues", + }, + packages=find_packages(exclude=("tests", "docs")), + install_requires=["click~=8.0.3"], + python_requires=">=3.8.12", + extras_require={ + "dev": ["MyPy~=0.812", "pytest~=6.2.5", "pytest-cov", "pytest-mock", "requests-mock", "pre-commit"], + "sphinx-docs": [ + "Sphinx~=4.2", + "sphinx-rtd-theme~=1.0", + ], + }, + entry_points={"console_scripts": ["octavia=octavia_cli.entrypoint:octavia"]}, +) diff --git a/octavia-cli/unit_tests/test_entrypoint.py b/octavia-cli/unit_tests/test_entrypoint.py new file mode 100644 index 000000000000..c8effc674c65 --- /dev/null +++ b/octavia-cli/unit_tests/test_entrypoint.py @@ -0,0 +1,25 @@ +# +# Copyright (c) 2021 Airbyte, Inc., all rights reserved. +# + +import pytest +from click.testing import CliRunner +from octavia_cli import entrypoint + + +def test_octavia(): + runner = CliRunner() + result = runner.invoke(entrypoint.octavia) + assert result.exit_code == 0 + assert result.output.startswith("Usage: octavia [OPTIONS] COMMAND [ARGS]...") + + +@pytest.mark.parametrize( + "command", + [entrypoint.init, entrypoint.apply, entrypoint.create, entrypoint.delete, entrypoint._list, entrypoint._import], +) +def test_not_implemented_commands(command): + runner = CliRunner() + result = runner.invoke(command) + assert result.exit_code == 1 + assert result.output.endswith("not yet implemented.\n") diff --git a/settings.gradle b/settings.gradle index b50a3f1933de..779ee999742f 100644 --- a/settings.gradle +++ b/settings.gradle @@ -19,27 +19,30 @@ sourceControl { rootProject.name = 'airbyte' -// SUB_BUILD is an enum of , PLATFORM, CONNECTORS_BASE. Blank is equivalent to all. +// SUB_BUILD is an enum of , PLATFORM, CONNECTORS_BASE, OCTAVIA_CLI. Blank is equivalent to all. if (!System.getenv().containsKey("SUB_BUILD")) { println("Building all of Airbyte.") } else { def subBuild = System.getenv().get("SUB_BUILD") println("Building Airbyte Sub Build: " + subBuild) - if (subBuild != "PLATFORM" && subBuild != "CONNECTORS_BASE") { - throw new IllegalArgumentException(String.format("%s is invalid. Must be unset or PLATFORM or CONNECTORS_BASE", subBuild)) + if (subBuild != "PLATFORM" && subBuild != "CONNECTORS_BASE" && subBuild != "OCTAVIA_CLI") { + throw new IllegalArgumentException(String.format("%s is invalid. Must be unset or PLATFORM or CONNECTORS_BASE or OCTAVIA_CLI", subBuild)) } } // shared -include ':airbyte-api' include ':airbyte-commons' -include ':airbyte-commons-docker' + +// shared by CONNECTORS_BASE and PLATFORM sub builds +include ':airbyte-api' include ':airbyte-commons-cli' +include ':airbyte-commons-docker' include ':airbyte-config:models' // reused by acceptance tests in connector base. include ':airbyte-db:lib' // reused by acceptance tests in connector base. include ':airbyte-json-validation' -include ':airbyte-protocol:models' include ':airbyte-metrics' +include ':airbyte-oauth' +include ':airbyte-protocol:models' include ':airbyte-queue' include ':airbyte-test-utils' @@ -60,7 +63,6 @@ if (!System.getenv().containsKey("SUB_BUILD") || System.getenv().get("SUB_BUILD" include ':airbyte-config:specs' include ':airbyte-container-orchestrator' include ':airbyte-webapp-e2e-tests' - include ':airbyte-oauth' include ':airbyte-scheduler:app' include ':airbyte-scheduler:client' include ':airbyte-server' @@ -86,8 +88,6 @@ if (!System.getenv().containsKey("SUB_BUILD") || System.getenv().get("SUB_BUILD" include ':airbyte-integrations:connector-templates:generator' include ':airbyte-integrations:bases:debezium' - include ':airbyte-oauth' - // Needed by normalization integration tests include ':airbyte-integrations:connectors:destination-bigquery' include ':airbyte-integrations:connectors:destination-jdbc' @@ -106,6 +106,10 @@ if (!System.getenv().containsKey("SUB_BUILD") || System.getenv().get("SUB_BUILD" include ':tools:code-generator' } +if (!System.getenv().containsKey("SUB_BUILD") || System.getenv().get("SUB_BUILD") == "OCTAVIA_CLI") { + include ':octavia-cli' +} + // connectors if (!System.getenv().containsKey("SUB_BUILD")) { // include all connector projects