diff --git a/Dockerfile b/Dockerfile index 122600bf94..2f9c4ecbcd 100644 --- a/Dockerfile +++ b/Dockerfile @@ -19,5 +19,7 @@ FROM debian:bullseye-slim as runtime RUN apt-get update; \ apt-get install -y --no-install-recommends libcurl4-openssl-dev ca-certificates WORKDIR app -COPY --from=builder /app/collab /app +COPY --from=builder /app/collab /app/collab +COPY --from=builder /app/crates/collab/migrations /app/migrations +ENV MIGRATIONS_PATH=/app/migrations ENTRYPOINT ["/app/collab"] diff --git a/Dockerfile.migrator b/Dockerfile.migrator deleted file mode 100644 index 482228a2eb..0000000000 --- a/Dockerfile.migrator +++ /dev/null @@ -1,15 +0,0 @@ -# syntax = docker/dockerfile:1.2 - -FROM rust:1.64-bullseye as builder -WORKDIR app -RUN --mount=type=cache,target=/usr/local/cargo/registry \ - --mount=type=cache,target=./target \ - cargo install sqlx-cli --root=/app --target-dir=/app/target --version 0.5.7 - -FROM debian:bullseye-slim as runtime -RUN apt-get update; \ - apt-get install -y --no-install-recommends libssl1.1 -WORKDIR app -COPY --from=builder /app/bin/sqlx /app -COPY ./crates/collab/migrations /app/migrations -ENTRYPOINT ["/app/sqlx", "migrate", "run"] diff --git a/Procfile b/Procfile index e1b87dd48b..d5db6fbd68 100644 --- a/Procfile +++ b/Procfile @@ -1,2 +1,2 @@ web: cd ../zed.dev && PORT=3000 npx vercel dev -collab: cd crates/collab && cargo run +collab: cd crates/collab && cargo run serve diff --git a/crates/collab/Procfile b/crates/collab/Procfile deleted file mode 100644 index ef8914fcc0..0000000000 --- a/crates/collab/Procfile +++ /dev/null @@ -1,2 +0,0 @@ -collab: ./target/release/collab -release: ./target/release/sqlx migrate run diff --git a/crates/collab/k8s/manifest.template.yml b/crates/collab/k8s/manifest.template.yml index 628cf92506..94af6cc152 100644 --- a/crates/collab/k8s/manifest.template.yml +++ b/crates/collab/k8s/manifest.template.yml @@ -54,6 +54,8 @@ spec: containers: - name: collab image: "${ZED_IMAGE_ID}" + args: + - serve ports: - containerPort: 8080 protocol: TCP diff --git a/crates/collab/k8s/migrate.template.yml b/crates/collab/k8s/migrate.template.yml index 9b1dc14d7e..3a848da85a 100644 --- a/crates/collab/k8s/migrate.template.yml +++ b/crates/collab/k8s/migrate.template.yml @@ -10,6 +10,8 @@ spec: containers: - name: migrator image: ${ZED_IMAGE_ID} + args: + - migrate env: - name: DATABASE_URL valueFrom: diff --git a/crates/collab/src/db.rs b/crates/collab/src/db.rs index b8c09d9afb..ce99b97348 100644 --- a/crates/collab/src/db.rs +++ b/crates/collab/src/db.rs @@ -6,8 +6,12 @@ use collections::HashMap; use futures::StreamExt; use serde::{Deserialize, Serialize}; pub use sqlx::postgres::PgPoolOptions as DbOptions; -use sqlx::{types::Uuid, FromRow, QueryBuilder}; -use std::{cmp, ops::Range, time::Duration}; +use sqlx::{ + migrate::{Migrate as _, Migration, MigrationSource}, + types::Uuid, + FromRow, QueryBuilder, +}; +use std::{cmp, ops::Range, path::Path, time::Duration}; use time::{OffsetDateTime, PrimitiveDateTime}; #[async_trait] @@ -173,6 +177,13 @@ pub trait Db: Send + Sync { fn as_fake(&self) -> Option<&FakeDb>; } +#[cfg(any(test, debug_assertions))] +pub const DEFAULT_MIGRATIONS_PATH: Option<&'static str> = + Some(concat!(env!("CARGO_MANIFEST_DIR"), "/migrations")); + +#[cfg(not(any(test, debug_assertions)))] +pub const DEFAULT_MIGRATIONS_PATH: Option<&'static str> = None; + pub struct PostgresDb { pool: sqlx::PgPool, } @@ -187,6 +198,47 @@ impl PostgresDb { Ok(Self { pool }) } + pub async fn migrate( + &self, + migrations_path: &Path, + ignore_checksum_mismatch: bool, + ) -> anyhow::Result> { + let migrations = MigrationSource::resolve(migrations_path) + .await + .map_err(|err| anyhow!("failed to load migrations: {err:?}"))?; + + let mut conn = self.pool.acquire().await?; + + conn.ensure_migrations_table().await?; + let applied_migrations: HashMap<_, _> = conn + .list_applied_migrations() + .await? + .into_iter() + .map(|m| (m.version, m)) + .collect(); + + let mut new_migrations = Vec::new(); + for migration in migrations { + match applied_migrations.get(&migration.version) { + Some(applied_migration) => { + if migration.checksum != applied_migration.checksum && !ignore_checksum_mismatch + { + Err(anyhow!( + "checksum mismatch for applied migration {}", + migration.description + ))?; + } + } + None => { + let elapsed = conn.apply(&migration).await?; + new_migrations.push((migration, elapsed)); + } + } + } + + Ok(new_migrations) + } + pub fn fuzzy_like_string(string: &str) -> String { let mut result = String::with_capacity(string.len() * 2 + 1); for c in string.chars() { @@ -1763,11 +1815,8 @@ mod test { use lazy_static::lazy_static; use parking_lot::Mutex; use rand::prelude::*; - use sqlx::{ - migrate::{MigrateDatabase, Migrator}, - Postgres, - }; - use std::{path::Path, sync::Arc}; + use sqlx::{migrate::MigrateDatabase, Postgres}; + use std::sync::Arc; use util::post_inc; pub struct FakeDb { @@ -2430,13 +2479,13 @@ mod test { let mut rng = StdRng::from_entropy(); let name = format!("zed-test-{}", rng.gen::()); let url = format!("postgres://postgres@localhost/{}", name); - let migrations_path = Path::new(concat!(env!("CARGO_MANIFEST_DIR"), "/migrations")); Postgres::create_database(&url) .await .expect("failed to create test db"); let db = PostgresDb::new(&url, 5).await.unwrap(); - let migrator = Migrator::new(migrations_path).await.unwrap(); - migrator.run(&db.pool).await.unwrap(); + db.migrate(Path::new(DEFAULT_MIGRATIONS_PATH.unwrap()), false) + .await + .unwrap(); Self { db: Some(Arc::new(db)), url, diff --git a/crates/collab/src/main.rs b/crates/collab/src/main.rs index 2c0fb69b67..0308b21e2b 100644 --- a/crates/collab/src/main.rs +++ b/crates/collab/src/main.rs @@ -9,12 +9,15 @@ mod db_tests; #[cfg(test)] mod integration_tests; +use anyhow::anyhow; use axum::{routing::get, Router}; use collab::{Error, Result}; use db::{Db, PostgresDb}; use serde::Deserialize; use std::{ + env::args, net::{SocketAddr, TcpListener}, + path::PathBuf, sync::Arc, time::Duration, }; @@ -34,22 +37,17 @@ pub struct Config { pub log_json: Option, } +#[derive(Default, Deserialize)] +pub struct MigrateConfig { + pub database_url: String, + pub migrations_path: Option, +} + pub struct AppState { db: Arc, config: Config, } -impl AppState { - async fn new(config: Config) -> Result> { - let db = PostgresDb::new(&config.database_url, 5).await?; - let this = Self { - db: Arc::new(db), - config, - }; - Ok(Arc::new(this)) - } -} - #[tokio::main] async fn main() -> Result<()> { if let Err(error) = env::load_dotenv() { @@ -59,24 +57,59 @@ async fn main() -> Result<()> { ); } - let config = envy::from_env::().expect("error loading config"); - init_tracing(&config); - let state = AppState::new(config).await?; + match args().skip(1).next().as_deref() { + Some("version") => { + println!("collab v{VERSION}"); + } + Some("migrate") => { + let config = envy::from_env::().expect("error loading config"); + let db = PostgresDb::new(&config.database_url, 5).await?; - let listener = TcpListener::bind(&format!("0.0.0.0:{}", state.config.http_port)) - .expect("failed to bind TCP listener"); - let rpc_server = rpc::Server::new(state.clone(), None); + let migrations_path = config + .migrations_path + .as_deref() + .or(db::DEFAULT_MIGRATIONS_PATH.map(|s| s.as_ref())) + .ok_or_else(|| anyhow!("missing MIGRATIONS_PATH environment variable"))?; - rpc_server.start_recording_project_activity(Duration::from_secs(5 * 60), rpc::RealExecutor); + let migrations = db.migrate(&migrations_path, false).await?; + for (migration, duration) in migrations { + println!( + "Ran {} {} {:?}", + migration.version, migration.description, duration + ); + } - let app = api::routes(&rpc_server, state.clone()) - .merge(rpc::routes(rpc_server)) - .merge(Router::new().route("/", get(handle_root))); + return Ok(()); + } + Some("serve") => { + let config = envy::from_env::().expect("error loading config"); + let db = PostgresDb::new(&config.database_url, 5).await?; - axum::Server::from_tcp(listener)? - .serve(app.into_make_service_with_connect_info::()) - .await?; + init_tracing(&config); + let state = Arc::new(AppState { + db: Arc::new(db), + config, + }); + let listener = TcpListener::bind(&format!("0.0.0.0:{}", state.config.http_port)) + .expect("failed to bind TCP listener"); + + let rpc_server = rpc::Server::new(state.clone(), None); + rpc_server + .start_recording_project_activity(Duration::from_secs(5 * 60), rpc::RealExecutor); + + let app = api::routes(&rpc_server, state.clone()) + .merge(rpc::routes(rpc_server)) + .merge(Router::new().route("/", get(handle_root))); + + axum::Server::from_tcp(listener)? + .serve(app.into_make_service_with_connect_info::()) + .await?; + } + _ => { + Err(anyhow!("usage: collab "))?; + } + } Ok(()) } diff --git a/script/bootstrap b/script/bootstrap index 4f6b9cc70d..e23f42e80e 100755 --- a/script/bootstrap +++ b/script/bootstrap @@ -7,7 +7,7 @@ echo "creating database..." script/sqlx database create echo "migrating database..." -script/sqlx migrate run +cargo run -p collab -- migrate echo "seeding database..." script/seed-db diff --git a/script/deploy b/script/deploy index e352df345b..f7da274841 100755 --- a/script/deploy +++ b/script/deploy @@ -1,12 +1,7 @@ #!/bin/bash -# Prerequisites: -# -# - Log in to the DigitalOcean API, either interactively, by running -# `doctl auth init`, or by setting the `DIGITALOCEAN_ACCESS_TOKEN` -# environment variable. - set -eu +source script/lib/deploy-helpers.sh if [[ $# < 2 ]]; then echo "Usage: $0 " @@ -15,28 +10,8 @@ fi export ZED_KUBE_NAMESPACE=$1 COLLAB_VERSION=$2 -ENV_FILE="crates/collab/k8s/environments/${ZED_KUBE_NAMESPACE}.sh" -if [[ ! -f $ENV_FILE ]]; then - echo "Invalid environment name '${ZED_KUBE_NAMESPACE}'" - exit 1 -fi -export $(cat $ENV_FILE) - -if [[ ! $COLLAB_VERSION =~ ^[0-9]+\.[0-9]+\.[0-9]+$ ]]; then - echo "Invalid version number '$COLLAB_VERSION'" - exit 1 -fi -TAG_NAMES=$(doctl registry repository list-tags collab --no-header --format Tag) -if ! $(echo "${TAG_NAMES}" | grep -Fqx v${COLLAB_VERSION}); then - echo "No such image tag: 'zed/collab:v${COLLAB_VERSION}'" - echo "Found tags" - echo "${TAG_NAMES}" - exit 1 -fi -export ZED_IMAGE_ID="registry.digitalocean.com/zed/collab:v${COLLAB_VERSION}" - -if [[ $(kubectl config current-context 2> /dev/null) != do-nyc1-zed-1 ]]; then - doctl kubernetes cluster kubeconfig save zed-1 -fi +export_vars_for_environment $ZED_KUBE_NAMESPACE +export ZED_IMAGE_ID=$(image_id_for_version $COLLAB_VERSION) +target_zed_kube_cluster envsubst < crates/collab/k8s/manifest.template.yml | kubectl apply -f - diff --git a/script/deploy-migration b/script/deploy-migration index 81a662db88..6812be3217 100755 --- a/script/deploy-migration +++ b/script/deploy-migration @@ -1,42 +1,20 @@ #!/bin/bash -# Prerequisites: -# -# - Log in to the DigitalOcean docker registry -# doctl registry login -# -# - Target the `zed-1` kubernetes cluster -# doctl kubernetes cluster kubeconfig save zed-1 - set -eu +source script/lib/deploy-helpers.sh -if [[ $# < 1 ]]; then - echo "Usage: $0 [production|staging|...]" +if [[ $# < 2 ]]; then + echo "Usage: $0 " exit 1 fi - export ZED_KUBE_NAMESPACE=$1 -ENV_FILE="crates/collab/k8s/environments/${ZED_KUBE_NAMESPACE}.sh" -if [[ ! -f $ENV_FILE ]]; then - echo "Invalid environment name '${ZED_KUBE_NAMESPACE}'" - exit 1 -fi +COLLAB_VERSION=$2 -if [[ -n $(git status --short) ]]; then - echo "Cannot deploy with uncommited changes" - exit 1 -fi - -git_sha=$(git rev-parse HEAD) -export ZED_IMAGE_ID=registry.digitalocean.com/zed/zed-migrator:${ZED_KUBE_NAMESPACE}-${git_sha} -export ZED_MIGRATE_JOB_NAME=zed-migrate-${git_sha} - -docker build . \ - --file ./Dockerfile.migrator \ - --tag $ZED_IMAGE_ID -docker push $ZED_IMAGE_ID +export_vars_for_environment $ZED_KUBE_NAMESPACE +export ZED_IMAGE_ID=$(image_id_for_version ${COLLAB_VERSION}) +export ZED_MIGRATE_JOB_NAME=zed-migrate-${COLLAB_VERSION} +target_zed_kube_cluster envsubst < crates/collab/k8s/migrate.template.yml | kubectl apply -f - - pod=$(kubectl --namespace=${ZED_KUBE_NAMESPACE} get pods --selector=job-name=${ZED_MIGRATE_JOB_NAME} --output=jsonpath='{.items[*].metadata.name}') echo "pod:" $pod diff --git a/script/lib/deploy-helpers.sh b/script/lib/deploy-helpers.sh new file mode 100644 index 0000000000..78ffa22671 --- /dev/null +++ b/script/lib/deploy-helpers.sh @@ -0,0 +1,43 @@ +# Prerequisites: +# +# - Log in to the DigitalOcean API, either interactively, by running +# `doctl auth init`, or by setting the `DIGITALOCEAN_ACCESS_TOKEN` +# environment variable. + +function export_vars_for_environment { + local environment=$1 + local env_file="crates/collab/k8s/environments/${environment}.sh" + if [[ ! -f $env_file ]]; then + echo "Invalid environment name '${environment}'" + exit 1 + fi + export $(cat $env_file) +} + +function image_id_for_version { + local version=$1 + if [[ ! ${version} =~ ^[0-9]+\.[0-9]+\.[0-9]+$ ]]; then + echo "Invalid version number '${version}'" + exit 1 + fi + TAG_NAMES=$(doctl registry repository list-tags collab --no-header --format Tag) + if ! $(echo "${TAG_NAMES}" | grep -Fqx v${version}); then + echo "No such image tag: 'zed/collab:v${version}'" + echo "Found tags" + echo "${TAG_NAMES}" + exit 1 + fi + + echo "registry.digitalocean.com/zed/collab:v${version}" +} + +function version_for_image_id { + local image_id=$1 + echo $image_id | cut -d: -f2 +} + +function target_zed_kube_cluster { + if [[ $(kubectl config current-context 2> /dev/null) != do-nyc1-zed-1 ]]; then + doctl kubernetes cluster kubeconfig save zed-1 + fi +}