mirror of
https://github.com/zulip/zulip.git
synced 2025-11-13 02:17:19 +00:00
github: Create action for generating DigitalOcean one click app image.
This commit is contained in:
24
.github/workflows/update-oneclick-apps.yml
vendored
Normal file
24
.github/workflows/update-oneclick-apps.yml
vendored
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
name: Update one click apps
|
||||||
|
on:
|
||||||
|
release:
|
||||||
|
types: [published]
|
||||||
|
jobs:
|
||||||
|
update-digitalocean-oneclick-app:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
- name: Update DigitalOcean one click app
|
||||||
|
env:
|
||||||
|
DIGITALOCEAN_API_KEY: ${{ secrets.DIGITALOCEAN_API_KEY }}
|
||||||
|
ZULIP_API_KEY: ${{ secrets.ZULIP_BOT_API_KEY }}
|
||||||
|
ZULIP_EMAIL: ${{ secrets.ZULIP_BOT_EMAIL }}
|
||||||
|
ZULIP_SITE: ${{ secrets.ZULIP_ORGANIZATION_URL }}
|
||||||
|
ZULIP_STREAM_TO_SEND_BOT_MESSAGES: ${{ secrets.ZULIP_STREAM_TO_SEND_BOT_MESSAGES }}
|
||||||
|
PYTHON_DIGITALOCEAN_REQUEST_TIMEOUT_SEC: 30
|
||||||
|
RELEASE_VERSION: ${{ github.event.release.tag_name }}
|
||||||
|
run: |
|
||||||
|
export PATH="$HOME/.local/bin:$PATH"
|
||||||
|
git clone https://github.com/zulip/marketplace-partners
|
||||||
|
pip3 install python-digitalocean zulip fab-classic
|
||||||
|
echo $PATH
|
||||||
|
python3 tools/oneclickapps/prepare_digital_ocean_one_click_app_release.py
|
||||||
@@ -51,6 +51,10 @@ preparing a new release.
|
|||||||
* Post the release by [editing the latest tag on
|
* Post the release by [editing the latest tag on
|
||||||
GitHub](https://github.com/zulip/zulip/tags); use the text from
|
GitHub](https://github.com/zulip/zulip/tags); use the text from
|
||||||
`changelog.md` for the release notes.
|
`changelog.md` for the release notes.
|
||||||
|
|
||||||
|
**Note:** This will trigger the [GitHub action](https://github.com/zulip/zulip/blob/master/tools/oneclickapps/README.md)
|
||||||
|
for updating DigitalOcean one-click app image. The action uses the latest release
|
||||||
|
tarball published on `zulip.org` for creating the image.
|
||||||
* Update the [Docker image](https://github.com/zulip/docker-zulip) and
|
* Update the [Docker image](https://github.com/zulip/docker-zulip) and
|
||||||
do a release of that.
|
do a release of that.
|
||||||
* Update the image of DigitalOcean one click app using
|
* Update the image of DigitalOcean one click app using
|
||||||
|
|||||||
73
tools/oneclickapps/README.md
Normal file
73
tools/oneclickapps/README.md
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
# One click app release automation
|
||||||
|
|
||||||
|
This directory contains scripts for automating the release of Zulip one click apps.
|
||||||
|
|
||||||
|
## DigitalOcean 1-Click Application
|
||||||
|
`prepare_digital_ocean_one_click_app_release.py` creates the image of DigitalOcean 1-Click
|
||||||
|
app from the latest Zulip release (fetched from https://www.zulip.org/dist/releases). It will
|
||||||
|
also create a test droplet from the image and send the image and droplet
|
||||||
|
details to a pre-configured Zulip stream. Anyone, whose key is added to the
|
||||||
|
Zulip DigitalOcean team can SSH into the droplet for testing.
|
||||||
|
|
||||||
|
### Running as GitHub action
|
||||||
|
|
||||||
|
`.github/workflows/update-oneclick-apps.yml` is configured to invoke
|
||||||
|
`prepare_digital_ocean_one_click_app_release.py` as a GitHub action during each Zulip
|
||||||
|
server release.
|
||||||
|
|
||||||
|
You also need to set the following secrets in your GitHub repository to make the action
|
||||||
|
work correctly. These secrets are passed as environment variables to the GitHub action.
|
||||||
|
|
||||||
|
* `DIGITALOCEAN_API_KEY` - DigitalOcean API KEY used for creating droplets, snapshots etc.
|
||||||
|
* `ZULIP_ORGANIZATION_URL` - The URL of the Zulip organization where the message containing
|
||||||
|
droplet/image details should be sent.
|
||||||
|
* `ZULIP_BOT_API_KEY` - The API key of the Zulip Bot used for sending messages.
|
||||||
|
* `ZULIP_STREAM_TO_SEND_BOT_MESSAGES` - The stream to which the messages should be sent.
|
||||||
|
* `ZULIP_BOT_EMAIL` - The email of the Zulip bot.
|
||||||
|
|
||||||
|
Also pass the following as environment variables in `.github/workflows/update-oneclick-apps.yml`.
|
||||||
|
* `PYTHON_DIGITALOCEAN_REQUEST_TIMEOUT_SEC` - This configures the maximum number of seconds
|
||||||
|
to wait before the requests made by `python-digitalocean` timeout. If not configured, it's
|
||||||
|
common for the requests to take 20+ minutes before getting timed out.
|
||||||
|
|
||||||
|
### Verifying the one click app image
|
||||||
|
* The action will send the image name and test droplet details to the stream configured in the
|
||||||
|
above steps.
|
||||||
|
* SSH into the test droplet by following the instructions in the message.
|
||||||
|
* After logging into the test droplet, exit the installer and run the following script.
|
||||||
|
|
||||||
|
https://raw.githubusercontent.com/digitalocean/marketplace-partners/master/scripts/img_check.sh
|
||||||
|
|
||||||
|
This script checks whether the image created is valid. It is also run by the DigitalOcean team
|
||||||
|
before they approve the image submission in the one click app marketplace.
|
||||||
|
* If there are no errors (you can ignore most of the warnings), exit the SSH connection and
|
||||||
|
reconnect.
|
||||||
|
* Populate the details asked by the installer and verify that the installer completes successfully.
|
||||||
|
If there are errors see the section below.
|
||||||
|
* Use the link generated by the installer to create a new Zulip organization. Do some basic
|
||||||
|
testing like sending a bunch of messages and reloading the webpage.
|
||||||
|
* If there are no issues, submit the image in the
|
||||||
|
[DigitalOcean vendor portal](https://marketplace.digitalocean.com/vendorportal). You need to be
|
||||||
|
added to the vendor portal team for doing this. Ask Tim to add you if required. During the submission,
|
||||||
|
make sure to update the blog post URL if it's a major release.
|
||||||
|
* Keep checking the vendor portal for change in status of the submission. DigitalOcean does nominally send
|
||||||
|
emails when there are updates on the submission, but we have found the emails to not always arrive.
|
||||||
|
* If there are any issues with submission, rebuild the image by manually invoking the script and
|
||||||
|
resubmit. The issues we have seen mostly in the past are caused by the dependencies getting outdated
|
||||||
|
by the time the DigitalOcean team run the checks. In that case you have to just rebuild the image
|
||||||
|
by invoking the script.
|
||||||
|
* Delete the test droplet `oneclickapp-{release_version}-test` after you have completed testing
|
||||||
|
by going to the [DigitalOcean Zulip team account](https://cloud.digitalocean.com/droplets?i=0242e0).
|
||||||
|
If there are other existing test droplets with the same name format but with with older release versions
|
||||||
|
feel free to delete them as well. These droplets are also tagged with `github-action` and`temporary`
|
||||||
|
tags.
|
||||||
|
|
||||||
|
**Errors**
|
||||||
|
|
||||||
|
If there are any errors while setting up the one click app installer, you have three options
|
||||||
|
* Include the fix in the Fabric script that setups the installer.
|
||||||
|
[01-initial-setup](https://raw.githubusercontent.com/zulip/marketplace-partners/master/marketplace_docs/templates/Fabric/scripts/01-initial-setup)
|
||||||
|
file should be a good place to include the fix. See
|
||||||
|
[zulip/marketplace-partners#4](https://github.com/zulip/marketplace-partners/pull/4/files) for an
|
||||||
|
example fix.
|
||||||
|
* Wait for the next release to fix the error.
|
||||||
@@ -0,0 +1,159 @@
|
|||||||
|
import os
|
||||||
|
import subprocess
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import List
|
||||||
|
|
||||||
|
import digitalocean
|
||||||
|
import zulip
|
||||||
|
from requests.adapters import HTTPAdapter
|
||||||
|
from requests.packages.urllib3.util.retry import Retry
|
||||||
|
|
||||||
|
manager = digitalocean.Manager(token=os.environ["DIGITALOCEAN_API_KEY"])
|
||||||
|
zulip_client = zulip.Client()
|
||||||
|
TEST_DROPLET_SUBDOMAIN = "oneclicktest"
|
||||||
|
|
||||||
|
|
||||||
|
def generate_ssh_keys() -> None:
|
||||||
|
subprocess.call(
|
||||||
|
["ssh-keygen", "-f", str(Path.home()) + "/.ssh/id_ed25519", "-P", "", "-t", "ed25519"]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def get_public_ssh_key() -> str:
|
||||||
|
try:
|
||||||
|
with open(str(Path.home()) + "/.ssh/id_ed25519.pub") as f:
|
||||||
|
return f.read()
|
||||||
|
except FileNotFoundError:
|
||||||
|
return ""
|
||||||
|
|
||||||
|
|
||||||
|
def sleep_until_droplet_action_is_completed(
|
||||||
|
droplet: digitalocean.Droplet, action_type: str
|
||||||
|
) -> None:
|
||||||
|
incomplete = True
|
||||||
|
while incomplete:
|
||||||
|
for action in droplet.get_actions():
|
||||||
|
action.load()
|
||||||
|
print(f"...[{action.type}]: {action.status}")
|
||||||
|
if action.type == action_type and action.status == "completed":
|
||||||
|
incomplete = False
|
||||||
|
break
|
||||||
|
if incomplete:
|
||||||
|
time.sleep(5)
|
||||||
|
droplet.load()
|
||||||
|
|
||||||
|
|
||||||
|
def set_api_request_retry_limits(api_object: digitalocean.baseapi.BaseAPI) -> None:
|
||||||
|
retry = Retry(connect=5, read=5, backoff_factor=0.1)
|
||||||
|
adapter = HTTPAdapter(max_retries=retry)
|
||||||
|
api_object._session.mount("https://", adapter)
|
||||||
|
|
||||||
|
|
||||||
|
def create_droplet(
|
||||||
|
name: str, ssh_keys: List[str], image: str = "ubuntu-18-04-x64"
|
||||||
|
) -> digitalocean.Droplet:
|
||||||
|
droplet = digitalocean.Droplet(
|
||||||
|
token=manager.token,
|
||||||
|
name=name,
|
||||||
|
region="nyc3",
|
||||||
|
size_slug="s-1vcpu-2gb",
|
||||||
|
image=image,
|
||||||
|
backups=False,
|
||||||
|
ssh_keys=ssh_keys,
|
||||||
|
tags=["github-action", "temporary"],
|
||||||
|
)
|
||||||
|
set_api_request_retry_limits(droplet)
|
||||||
|
droplet.create()
|
||||||
|
sleep_until_droplet_action_is_completed(droplet, "create")
|
||||||
|
return droplet
|
||||||
|
|
||||||
|
|
||||||
|
def create_ssh_key(name: str, public_key: str) -> digitalocean.SSHKey:
|
||||||
|
action_public_ssh_key_object = digitalocean.SSHKey(
|
||||||
|
name=name, public_key=public_key, token=manager.token
|
||||||
|
)
|
||||||
|
set_api_request_retry_limits(action_public_ssh_key_object)
|
||||||
|
action_public_ssh_key_object.create()
|
||||||
|
return action_public_ssh_key_object
|
||||||
|
|
||||||
|
|
||||||
|
def create_snapshot(droplet: digitalocean.Droplet, snapshot_name: str) -> None:
|
||||||
|
droplet.take_snapshot(snapshot_name, power_off=True)
|
||||||
|
droplet.load()
|
||||||
|
sleep_until_droplet_action_is_completed(droplet, "snapshot")
|
||||||
|
|
||||||
|
|
||||||
|
def create_dns_records(droplet: digitalocean.Droplet) -> None:
|
||||||
|
domain = digitalocean.Domain(token=manager.token, name="zulipdev.org")
|
||||||
|
set_api_request_retry_limits(domain)
|
||||||
|
domain.load()
|
||||||
|
|
||||||
|
oneclick_test_app_record_names = [TEST_DROPLET_SUBDOMAIN, f"*.{TEST_DROPLET_SUBDOMAIN}"]
|
||||||
|
for record in domain.get_records():
|
||||||
|
if (
|
||||||
|
record.name in oneclick_test_app_record_names
|
||||||
|
and record.domain == "zulipdev.org"
|
||||||
|
and record.type == "A"
|
||||||
|
):
|
||||||
|
record.destroy()
|
||||||
|
|
||||||
|
domain.load()
|
||||||
|
for record_name in oneclick_test_app_record_names:
|
||||||
|
domain.create_new_domain_record(type="A", name=record_name, data=droplet.ip_address)
|
||||||
|
|
||||||
|
|
||||||
|
def setup_one_click_app_installer(droplet: digitalocean.Droplet) -> None:
|
||||||
|
subprocess.call(
|
||||||
|
[
|
||||||
|
"fab",
|
||||||
|
"build_image",
|
||||||
|
"-H",
|
||||||
|
droplet.ip_address,
|
||||||
|
"--keepalive",
|
||||||
|
"5",
|
||||||
|
"--connection-attempts",
|
||||||
|
"10",
|
||||||
|
],
|
||||||
|
cwd="marketplace-partners/marketplace_docs/templates/Fabric",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def send_message(content: str) -> None:
|
||||||
|
request = {
|
||||||
|
"type": "stream",
|
||||||
|
"to": os.environ["ZULIP_STREAM_TO_SEND_BOT_MESSAGES"],
|
||||||
|
"topic": "DigitalOcean One Click App",
|
||||||
|
"content": content,
|
||||||
|
}
|
||||||
|
zulip_client.send_message(request)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
release_version = os.environ["RELEASE_VERSION"]
|
||||||
|
|
||||||
|
generate_ssh_keys()
|
||||||
|
action_public_ssh_key_object = create_ssh_key(
|
||||||
|
f"oneclickapp-{release_version}-image-generator-public-key", get_public_ssh_key()
|
||||||
|
)
|
||||||
|
|
||||||
|
image_generator_droplet = create_droplet(
|
||||||
|
f"oneclickapp-{release_version}-image-generator", [action_public_ssh_key_object]
|
||||||
|
)
|
||||||
|
|
||||||
|
setup_one_click_app_installer(image_generator_droplet)
|
||||||
|
|
||||||
|
oneclick_image_name = f"oneclickapp-{release_version}"
|
||||||
|
create_snapshot(image_generator_droplet, oneclick_image_name)
|
||||||
|
snapshot = image_generator_droplet.get_snapshots()[0]
|
||||||
|
send_message(f"One click app image `{oneclick_image_name}` created.")
|
||||||
|
|
||||||
|
image_generator_droplet.destroy()
|
||||||
|
action_public_ssh_key_object.destroy()
|
||||||
|
|
||||||
|
test_droplet_name = f"oneclickapp-{release_version}-test"
|
||||||
|
test_droplet = create_droplet(test_droplet_name, manager.get_all_sshkeys(), image=snapshot.id)
|
||||||
|
create_dns_records(test_droplet)
|
||||||
|
send_message(
|
||||||
|
f"Test droplet `{test_droplet_name}` created. SSH as root to {TEST_DROPLET_SUBDOMAIN}.zulipdev.org for testing."
|
||||||
|
)
|
||||||
Reference in New Issue
Block a user