diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 613224838..422266f64 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -56,9 +56,15 @@ jobs: tag_and_push langstream-control-plane tag_and_push langstream-api-gateway + - name: Package mini-langstream + run: | + version=${GITHUB_REF/refs\/tags\/v/} + ./bin/mini-langstream/package.sh $version + + - uses: ncipollo/release-action@v1 with: - artifacts: "langstream-cli/target/langstream-*.zip,helm/crds/*.yml" + artifacts: "langstream-cli/target/langstream-*.zip,helm/crds/*.yml,target/mini-langstream-*.zip" token: ${{ secrets.GITHUB_TOKEN }} generateReleaseNotes: true prerelease: false diff --git a/mini-langstream/README.md b/mini-langstream/README.md new file mode 100644 index 000000000..48f73dc98 --- /dev/null +++ b/mini-langstream/README.md @@ -0,0 +1,40 @@ +# mini-langstream + +## Get started + +### MacOS/Linux/WSL + +``` +curl -Ls "https://raw.githubusercontent.com/LangStream/langstream/main/bin/mini-langstream/get-mini-langstream.sh" | bash +``` + +## Start the cluster + +``` +$ mini-langstream start +``` + +## Use the CLI to deploy an application + +``` +$ mini-langstream cli apps deploy app -i \$(mini-langstream get-instance)" -s https://raw.githubusercontent.com/LangStream/langstream/main/examples/secrets/secrets.yaml -app https://github.com/LangStream/langstream/tree/main/examples/applications/python-processor-exclamation +``` + +## Start the cluster + +``` +$ mini-langstream start +``` + + +## Stop the cluster + +``` +$ mini-langstream delete +``` + +## Get help + +``` +mini-langstream help +``` diff --git a/mini-langstream/get-mini-langstream.sh b/mini-langstream/get-mini-langstream.sh new file mode 100755 index 000000000..fac8cc26f --- /dev/null +++ b/mini-langstream/get-mini-langstream.sh @@ -0,0 +1,188 @@ +#!/bin/bash +# +# +# Copyright DataStax, Inc. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +set -e + +track_last_command() { + last_command=$current_command + current_command=$BASH_COMMAND +} +trap track_last_command DEBUG + +echo_failed_command() { + local exit_code="$?" + if [[ "$exit_code" != "0" ]]; then + echo "'$last_command': command failed with exit code $exit_code." + fi +} + +trap echo_failed_command EXIT + +clear + + +echo " _ ____ _ "; +echo " | | __ _ _ __ __ _/ ___|| |_ _ __ ___ __ _ _ __ ___ "; +echo " | | / _\` | '_ \ / _\` \___ \| __| '__/ _ \/ _\` | '_ \` _ \ "; +echo " | |__| (_| | | | | (_| |___) | |_| | | __/ (_| | | | | | |"; +echo " |_____\__,_|_| |_|\__, |____/ \__|_| \___|\__,_|_| |_| |_|"; +echo " |___/ "; + + +get_latest_release_tarball_url() { + if ! command -v jq > /dev/null; then + echo "Not found." + echo "======================================================================================================" + echo " Please install jq on your system using your favourite package manager." + echo "" + echo " Restart after installing jq." + echo "======================================================================================================" + echo " In alternative you can set a fixed mini-langstream version by setting MINILANGSTREAM_URL." + echo "======================================================================================================" + echo "" + exit 1 + fi + curl -Ss https://api.github.com/repos/LangStream/langstream/releases/latest | jq -r '.assets[] | select((.name | contains("mini-langstream"))) | .browser_download_url' +} + + +echo "Installing $(tput setaf 6)mini-langstream$(tput setaf 7) please wait... " +# Local installation +BIN_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) +ROOT_DIR=$( cd -P "$( dirname "$BIN_DIR" )" >/dev/null 2>&1 && pwd ) +echo "" +echo "$(tput setaf 6)Checking archive:$(tput setaf 7)" +if [ -z "$MINILANGSTREAM_URL" ]; then + echo "$(tput setaf 2)MINILANGSTREAM_URL$(tput setaf 7) environment not set, checking for the latest release" + MINILANGSTREAM_URL=$(get_latest_release_tarball_url) + echo "$(tput setaf 2)[OK]$(tput setaf 7) - Using $MINILANGSTREAM_URL" +else + echo "$(tput setaf 2)[OK]$(tput setaf 7) - mini-langstream url set to $MINILANGSTREAM_URL (from MINILANGSTREAM_URL)" +fi +candidate_base_name=$(basename $MINILANGSTREAM_URL) + + +langstream_root_dir="$HOME/.mini-langstream" +mkdir -p $langstream_root_dir +langstream_downloads_dir="$langstream_root_dir/downloads" +mkdir -p $langstream_downloads_dir +langstream_candidates_dir="$langstream_root_dir/candidates" +mkdir -p $langstream_candidates_dir + +langstream_current_symlink="$langstream_candidates_dir/current" +mkdir -p $langstream_candidates_dir + +downloaded_zip_path=$langstream_downloads_dir/$candidate_base_name +downloaded_extracted_dir=${MINILANGSTREAM_URL//\.zip} +downloaded_extracted_path="$langstream_candidates_dir/$(basename $downloaded_extracted_dir)" + +darwin=false +case "$(uname)" in + Darwin*) + darwin=true + ;; +esac + +echo "$(tput setaf 2)[OK]$(tput setaf 7) - Ready to install $(basename $downloaded_extracted_dir)." + +if ! command -v unzip > /dev/null; then + echo "Not found." + echo "======================================================================================================" + echo " Please install unzip on your system using your favourite package manager." + echo "" + echo " Restart after installing unzip." + echo "======================================================================================================" + echo "" + exit 1 +fi +echo "$(tput setaf 2)[OK]$(tput setaf 7) - unzip command is available" + +if ! command -v curl > /dev/null; then + echo "Not found." + echo "" + echo "======================================================================================================" + echo " Please install curl on your system using your favourite package manager." + echo "" + echo " Restart after installing curl." + echo "======================================================================================================" + echo "" + exit 1 +fi +echo "$(tput setaf 2)[OK]$(tput setaf 7) - curl command is available" + +echo "" +echo "$(tput setaf 6)Downloading archive:$(tput setaf 7)" +if [ -f "$downloaded_zip_path" ]; then + echo "$(tput setaf 2)[OK]$(tput setaf 7) - Archive is already there" +else + curl --fail --location --progress-bar "$MINILANGSTREAM_URL" > "$downloaded_zip_path" + echo "$(tput setaf 2)[OK]$(tput setaf 7) - File downloaded" +fi + +# check integrity +ARCHIVE_OK=$(unzip -qt "$downloaded_zip_path" | grep 'No errors detected in compressed data') +if [[ -z "$ARCHIVE_OK" ]]; then + echo "Downloaded zip archive corrupt. Are you connected to the internet?" + exit +fi +echo "$(tput setaf 2)[OK]$(tput setaf 7) - Integrity of the archive checked" + +echo "" +echo "$(tput setaf 6)Extracting and installation:$(tput setaf 7)" +unzip -qo "$downloaded_zip_path" -d "$langstream_candidates_dir" +echo "$(tput setaf 2)[OK]$(tput setaf 7) - Extraction is successful" + +rm -rf $langstream_current_symlink +ln -s $downloaded_extracted_path $langstream_current_symlink + +echo "$(tput setaf 2)[OK]$(tput setaf 7) - mini-langstream installed at $langstream_candidates_dir" + +function inject_if_not_found() { + local file=$1 + touch "$file" + if [[ -z $(grep 'mini-langstream/candidates' "$file") ]]; then + echo -e "\n$init_snipped" >> "$file" + echo "$(tput setaf 2)[OK]$(tput setaf 7) - mini-langstream bin added to ${file}" + fi +} + + + + +bash_profile="${HOME}/.bash_profile" +bashrc="${HOME}/.bashrc" +zshrc="${ZDOTDIR:-${HOME}}/.zshrc" +init_snipped=$( cat << EOF +export PATH=\$PATH:$langstream_current_symlink/bin +EOF +) + +if [[ $darwin == true ]]; then + inject_if_not_found $bash_profile +else + inject_if_not_found $bashrc +fi + +if [[ -s "$zshrc" ]]; then + inject_if_not_found $zshrc +fi + +echo "$(tput setaf 2)[OK]$(tput setaf 7) - Installation Successful" +echo "Open $(tput setaf 2)a new terminal$(tput setaf 7) and run: $(tput setaf 3)mini-langstream start$(tput setaf 7)" +echo "" +echo "You can close this window." diff --git a/mini-langstream/mini-langstream b/mini-langstream/mini-langstream new file mode 100755 index 000000000..015145eb8 --- /dev/null +++ b/mini-langstream/mini-langstream @@ -0,0 +1,468 @@ +#!/bin/bash + +set -e +set -o errexit -o pipefail -o nounset +check_command() { + if ! command -v $1 &> /dev/null + then + echo "Command $1 could not be found. $2" + exit + fi +} + +check_command docker +check_command minikube +check_command helm +check_command kubectl +check_command langstream "Please install LangStream CLI using the instructions at https://github.com/LangStream/langstream#installation" + + +help() { + echo """ +mini-langstream, LangStream local cluster on Minikube. + +Syntax: mini-langstream [command] [options] + +Commands: + start Start the LangStream local cluster. + delete Delete the LangStream local cluster and local data. + langstream|cli Run the LangStream CLI with the given command. + get-instance Get the path to the LangStream app instance YAML file. + get-config Get the path to the Kubernetes configuration file. + get-values Get the path to the Helm values. + kubectl Run kubectl with the given command. + k9s Run K9s with the given command. + helm Run Helm with the given command. + help Print this help message. + dev Dev commands. It's required to run the command from the LangStream repository root directory. + start [--load] Start the LangStream local cluster in development mode. This will use the latest development images that are built locally. + --load Load LangStream images instead of building them. + load Load a specific Docker image into Minikube. + build Build docker image for a specific LangStream component and restart it. No arguments to rebuild all the images. + get-values Get the path to the Helm values. This values file is used when the cluster is started in dev mode. + +""" +} + + +# Constants +minikube_profile=mini-langstream +k8s_namespace=langstream +app_home=$HOME/.mini-langstream +data_dir=$app_home/data +conf_dir=$app_home/conf +kubeconfig_path=$data_dir/kube-config +cli_config=$data_dir/cli.yaml +app_instance_file=$conf_dir/app-instance.yaml +values_file=$conf_dir/values.yaml +dev_values_file=$conf_dir/dev-values.yaml + +mkdir -p $app_home +mkdir -p $data_dir +mkdir -p $conf_dir +touch $cli_config + + +cat > $app_instance_file < $dev_values_file < $values_file </dev/null + kubectl_cmd config set-context --current --namespace=$k8s_namespace >/dev/null + echo "✅" +} + +delete_minikube() { + echo -n "Deleting minikube: " + minikube_cmd delete --purge + echo "✅" +} + +helm_cmd() { + helm --kubeconfig $kubeconfig_path "$@" +} +kubectl_cmd() { + KUBECONFIG=$kubeconfig_path kubectl "$@" +} + +load_image_if_not_exists() { + local image=$1 + echo -n "Image $image: " + (minikube_cmd image list | grep -q $image) || load_image $image + echo "✅" +} +load_image() { + local image=$1 + minikube_cmd image load $image +} + +install_langstream() { + local dev="$1" + local load="$2" + helm_cmd repo add langstream https://datastax.github.io/langstream &> /dev/null || true + helm_cmd repo update &> /dev/null + if [ "$dev" == "true" ]; then + if [ "$load" == "false" ]; then + echo -n "Building images: " + build_no_restart + echo "✅" + else + load_image_if_not_exists langstream/langstream-control-plane:latest-dev + load_image_if_not_exists langstream/langstream-cli:latest-dev + load_image_if_not_exists langstream/langstream-deployer:latest-dev + load_image_if_not_exists langstream/langstream-runtime:latest-dev + load_image_if_not_exists langstream/langstream-api-gateway:latest-dev + fi + + echo -n "LangStream: " + helm_cmd upgrade --install langstream -n $k8s_namespace --create-namespace langstream/langstream --values $dev_values_file > /dev/null + else + echo -n "LangStream: " + helm_cmd upgrade --install langstream langstream/langstream -n $k8s_namespace --create-namespace --values $values_file > /dev/null + fi + echo "✅" + +} + +start_port_forward() { + local service=$1 + local file=$(mktemp) + + minikube_cmd service $1 --url -n $k8s_namespace &> $file & + grep -q 'http' <(tail -f $file) + url=$(grep -e http $file) + echo "$url" +} + +kafka_hostname=langstream-kafka + +install_kafka() { + if [ "$(docker ps -q -f name=$kafka_hostname)" ]; then + echo "Kafka: ✅" + return + fi + + if [ "$(docker ps -a -q -f name=$kafka_hostname)" ]; then + docker rm -f $kafka_hostname + fi + + + (docker run \ + -d \ + --name $kafka_hostname \ + -p 39092:39092 \ + -e KAFKA_LISTENERS=BROKER://0.0.0.0:19092,EXTERNAL://0.0.0.0:39092,CONTROLLER://0.0.0.0:9093 \ + -e KAFKA_ADVERTISED_LISTENERS=BROKER://localhost:19092,EXTERNAL://host.minikube.internal:39092 \ + -e KAFKA_INTER_BROKER_LISTENER_NAME=BROKER \ + -e KAFKA_CONTROLLER_LISTENER_NAMES=CONTROLLER \ + -e KAFKA_LISTENER_SECURITY_PROTOCOL_MAP=CONTROLLER:PLAINTEXT,BROKER:PLAINTEXT,EXTERNAL:PLAINTEXT \ + -e KAFKA_PROCESS_ROLES='controller,broker' \ + -e KAFKA_NODE_ID=1 \ + -e KAFKA_CONTROLLER_QUORUM_VOTERS='1@0.0.0.0:9093' \ + -e KAFKA_LOG_DIRS='/tmp/kraft-combined-logs' \ + -e CLUSTER_ID=ciWo7IWazngRchmPES6q5A== \ + -e KAFKA_OFFSETS_TOPIC_REPLICATION_FACTOR=1 \ + -v $data_dir/kafka:/var/lib/kafka/data \ + confluentinc/cp-kafka:7.5.0) > /dev/null + echo "Kafka: ✅" +} +delete_kafka() { + delete_docker_container $kafka_hostname + rm -rf $data_dir/kafka +} + +minio_hostname=langstream-minio + +install_minio() { + + if [ "$(docker ps -q -f name=$minio_hostname)" ]; then + echo "Minio: ✅" + return + fi + + if [ "$(docker ps -a -q -f name=$minio_hostname)" ]; then + docker rm -f $minio_hostname + fi + (docker run \ + -d \ + --name $minio_hostname \ + -p 9090:9090 \ + -p 9000:9000 \ + -v $data_dir/minio:/data \ + quay.io/minio/minio:latest \ + server /data --console-address :9090) > /dev/null + echo "Minio: ✅" +} +delete_minio() { + delete_docker_container $minio_hostname + rm -rf $data_dir/minio +} + +configure_cli() { + control_plane_url=$(start_port_forward langstream-control-plane) + api_gateway_url=$(start_port_forward langstream-api-gateway) + langstream -V + langstream profiles delete local-langstream-cluster &> /dev/null || true + langstream profiles create local-langstream-cluster --web-service-url $control_plane_url --api-gateway-url ${api_gateway_url/http/ws} --tenant default >/dev/null + echo "CLI: ✅" +} + +run_cli() { + langstream -p local-langstream-cluster "$@" +} +cleanup_docker_env() { + echo -n "Docker: " + docker system prune -f &> /dev/null + docker volume prune -f &> /dev/null + echo "✅" +} +start() { + local dev="false" + local load="false" + while [[ "$#" -gt 0 ]]; do + case $1 in + --dev) dev="true"; shift ;; + --load) load="true"; shift ;; + *) echo "Unknown parameter passed: $1"; exit 1 ;; + esac + done + cleanup_docker_env + start_minikube + install_minio + install_langstream "$dev" "$load" + install_kafka + configure_cli + echo "Ready 🚀" + echo "Deploy your first app with:" + echo "mini-langstream cli apps deploy app -i \$(mini-langstream get-instance)" -s https://raw.githubusercontent.com/LangStream/langstream/main/examples/secrets/secrets.yaml -app https://github.com/LangStream/langstream/tree/main/examples/applications/python-processor-exclamation + echo "mini-langstream cli apps get app" +} + +delete_docker_container() { + local container=$1 + echo -n "Deleting $container: " + docker rm -f $container &> /dev/null || true + echo "✅" +} + +delete() { + delete_kafka + delete_minikube + delete_minio +} + +handle_load() { + local image=$1 + if [[ $image == "control-plane" ]]; then + image=langstream/langstream-control-plane:latest-dev + elif [[ $image == "deployer" ]]; then + image=langstream/langstream-deployer:latest-dev + elif [[ $image == "api-gateway" ]]; then + image=langstream/langstream-api-gateway:latest-dev + elif [[ $image == "cli" ]]; then + image=langstream/langstream-cli:latest-dev + elif [[ $image == "runtime" ]]; then + image=langstream/langstream-runtime:latest-dev + fi + load_image $image + label="" + if [[ $image == *"langstream-control-plane"* ]]; then + label=langstream-control-plane + elif [[ $image == *"langstream-deployer"* ]]; then + label=langstream-deployer + elif [[ $image == *"langstream-api-gateway"* ]]; then + label=langstream-api-gateway + elif [[ $image == *"langstream-cli"* ]]; then + label=langstream-client + fi + if [ ! -z "$label" ]; then + kubectl_cmd -n $k8s_namespace delete pod -l app.kubernetes.io/name=$label + echo "Pod $label: ✅" + fi +} +build_no_restart() { + eval $(minikube_cmd docker-env) + ./docker/build.sh "$@" + eval $(minikube_cmd docker-env -u) +} + +handle_build() { + eval $(minikube_cmd docker-env) + ./docker/build.sh "$@" + eval $(minikube_cmd docker-env -u) + + docker_build_input=$1 + if [[ "$docker_build_input" == "control-plane" ]]; then + labels=(langstream-control-plane) + elif [[ "$docker_build_input" == "operator" || "$docker_build_input" == "deployer" ]]; then + labels=(langstream-deployer) + elif [[ "$docker_build_input" == "cli" ]]; then + labels=(langstream-client) + elif [[ "$docker_build_input" == "api-gateway" ]]; then + labels=(langstream-api-gateway) + elif [[ "$docker_build_input" == "runtime" ]]; then + labels=() + else + labels=(langstream-control-plane langstream-deployer langstream-api-gateway langstream-client) + fi + + if (( ${#labels[@]} )); then + for label in "${labels[@]}"; do + deploy_name=$(kubectl_cmd -n $k8s_namespace get deployment -l app.kubernetes.io/name=$label | awk '{print $1}' | grep langstream) + output=$(kubectl_cmd -n $k8s_namespace patch deployment $deploy_name -p "{\"spec\":{\"template\":{\"spec\":{\"containers\":[{\"name\":\"langstream\",\"image\":\"langstream/$label:latest-dev\"}]}}}}") + if echo "$output" | grep -q "(no change)"; then + kubectl_cmd -n $k8s_namespace delete pod -l app.kubernetes.io/name=$label > /dev/null + fi + echo "Pod $label: ✅" + done + fi +} + + +arg1=${1:-""} +if [ "$arg1" == "" ] || [ "$arg1" == "help" ]; then + help +elif [ "$arg1" == "start" ]; then + shift + start "$@" +elif [ "$arg1" == "delete" ]; then + shift + delete "$@" +elif [ "$arg1" == "langstream" ] || [ "$arg1" == "cli" ]; then + shift + run_cli "$@" +elif [ "$arg1" == "get-instance" ]; then + shift + echo "$(realpath $app_instance_file)" +elif [ "$arg1" == "get-config" ]; then + shift + echo "$kubeconfig_path" +elif [ "$arg1" == "kubectl" ]; then + shift + kubectl_cmd "$@" +elif [ "$arg1" == "k9s" ]; then + shift + KUBECONFIG=$kubeconfig_path k9s "$@" +elif [ "$arg1" == "helm" ]; then + shift + helm_cmd "$@" +elif [ "$arg1" == "get-values" ]; then + shift + echo "$(realpath $values_file)" +elif [ "$arg1" == "dev" ]; then + shift + devarg1=${1:-""} + if [ "$devarg1" == "start" ]; then + shift + start --dev "$@" + elif [ "$devarg1" == "load" ]; then + shift + handle_load "$@" + elif [ "$devarg1" == "build" ]; then + shift + handle_build "$@" + elif [ "$devarg1" == "get-values" ]; then + shift + echo "$(realpath $dev_values_file)" + else + echo "Unknown command $arg1" + help + exit 1 + fi +else + echo "Unknown command $arg1" + help + exit 1 +fi diff --git a/mini-langstream/package.sh b/mini-langstream/package.sh new file mode 100755 index 000000000..0f714bdcc --- /dev/null +++ b/mini-langstream/package.sh @@ -0,0 +1,32 @@ +#!/bin/bash +# +# Copyright DataStax, Inc. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +version=$1 +if [ -z "$version" ]; then + echo "version is not set" + exit 1 +fi +temp_dir=$(mktemp -d) +mkdir -p $temp_dir/bin +cp mini-langstream/mini-langstream $temp_dir/bin/mini-langstream + +mkdir -p target +target_filename=$(realpath target/mini-langstream-$version.zip) +rm -rf $target_filename +cd $temp_dir +zip -r $target_filename . +echo "Created $target_filename" \ No newline at end of file