Skip to content

add export #26

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 1 commit into from
Jan 9, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
173 changes: 173 additions & 0 deletions .github/workflows/bb-export.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,173 @@
name: Bytebase Export SQL
on:
pull_request:
types: [closed]
branches:
- main
paths:
- 'export/**'
workflow_dispatch:

jobs:
bytebase-export:
if: github.event.pull_request.merged == true
runs-on: ubuntu-latest
permissions:
pull-requests: write
issues: write
contents: read
steps:
- name: Checkout code
uses: actions/checkout@v4
with:
ref: ${{ github.event.pull_request.head.sha }}
fetch-depth: 0

- name: Login Bytebase
id: bytebase-login
uses: bytebase/login-action@0.0.2
with:
bytebase-url: ${{ secrets.BYTEBASE_URL }}
service-key: ${{ secrets.BYTEBASE_SERVICE_KEY }}
service-secret: ${{ secrets.BYTEBASE_SERVICE_SECRET }}

- name: Get changed files
id: changed-files
uses: tj-actions/changed-files@v42
with:
files: |
export/**/*.sql
since_last_remote_commit: true

- name: Process SQL files
id: process-sql
if: steps.changed-files.outputs.any_changed == 'true'
run: |
# Function to make API calls with error handling
call_api() {
local url="$1"
local method="$2"
local data="$3"
local description="$4"

echo "Making $description API call..."
response=$(curl -s -w "\n%{http_code}" \
--request "$method" "$url" \
--header "Authorization: Bearer ${{ steps.bytebase-login.outputs.token }}" \
--header "Content-Type: application/json" \
--data "$data")

status_code=$(echo "$response" | tail -n1)
body=$(echo "$response" | sed '$d')

echo "Status code: $status_code"
if [[ $status_code -lt 200 || $status_code -ge 300 ]]; then
echo "Failed $description. Status: $status_code"
echo "Response: $body"
return 1
fi

echo "$body"
return 0
}

# Process each SQL file
for file in ${{ steps.changed-files.outputs.all_changed_files }}; do
echo "Processing $file"

# Find and parse manifest.toml
DIR_PATH=$(dirname "$file")
MANIFEST_PATH=""
while [[ "$DIR_PATH" == export* ]]; do
if [[ -f "$DIR_PATH/manifest.toml" ]]; then
MANIFEST_PATH="$DIR_PATH/manifest.toml"
break
fi
DIR_PATH=$(dirname "$DIR_PATH")
done

if [[ -z "$MANIFEST_PATH" ]]; then
echo "Error: No manifest.toml found for $file"
exit 1
fi

# Parse manifest.toml
PROJECT=$(python3 -c "import toml; print(toml.load('$MANIFEST_PATH')['project'])")
INSTANCE=$(python3 -c "import toml; print(toml.load('$MANIFEST_PATH')['instance'])")
DATABASE=$(python3 -c "import toml; print(toml.load('$MANIFEST_PATH')['database'])")
FORMAT=$(python3 -c "import toml; config=toml.load('$MANIFEST_PATH'); print(config.get('format', 'JSON'))")

# Read SQL content and encode to base64
SQL_CONTENT=$(base64 < "$file")

# Generate UUID for step ID
STEP_ID=$(python3 -c "import uuid; print(str(uuid.uuid4()))")

BASE_URL="${{ steps.bytebase-login.outputs.api_url }}"

# 1. Create Sheet
sheet_data=$(call_api \
"$BASE_URL/v1/projects/$PROJECT/sheets" \
"POST" \
"{\"title\":\"\",\"content\":\"$SQL_CONTENT\",\"type\":\"TYPE_SQL\",\"source\":\"SOURCE_BYTEBASE_ARTIFACT\",\"visibility\":\"VISIBILITY_PUBLIC\"}" \
"Create Sheet") || exit 1

SHEET_NAME=$(echo "$sheet_data" | python3 -c "import sys, json; print(json.load(sys.stdin)['name'])")

# 2. Create Plan
plan_data=$(call_api \
"$BASE_URL/v1/projects/$PROJECT/plans" \
"POST" \
"{\"steps\":[{\"specs\":[{\"id\":\"$STEP_ID\",\"export_data_config\":{\"target\":\"/instances/$INSTANCE/databases/$DATABASE\",\"format\":\"$FORMAT\",\"sheet\":\"$SHEET_NAME\"}}]}],\"title\":\"Export data from $DATABASE\",\"description\":\"EXPORT\"}" \
"Create Plan") || exit 1

PLAN_NAME=$(echo "$plan_data" | python3 -c "import sys, json; print(json.load(sys.stdin)['name'])")

# 3. Create Issue
issue_data=$(call_api \
"$BASE_URL/v1/projects/$PROJECT/issues" \
"POST" \
"{\"approvers\":[],\"approvalTemplates\":[],\"subscribers\":[],\"title\":\"Issue: Export data from instances/$INSTANCE/databases/$DATABASE\",\"description\":\"SQL request from GitHub\",\"type\":\"DATABASE_DATA_EXPORT\",\"assignee\":\"\",\"plan\":\"$PLAN_NAME\"}" \
"Create Issue") || exit 1

# 4. Create Rollout
rollout_data=$(call_api \
"$BASE_URL/v1/projects/$PROJECT/rollouts" \
"POST" \
"{\"plan\":\"$PLAN_NAME\"}" \
"Create Rollout") || exit 1

# Extract issue link for PR comment
ISSUE_NUMBER=$(echo "$issue_data" | python3 -c "import sys, json; print(json.load(sys.stdin)['name'].split('/')[-1])")
ISSUE_LINK="${{ secrets.BYTEBASE_URL }}/projects/$PROJECT/issues/$ISSUE_NUMBER"
echo "ISSUE_LINK=$ISSUE_LINK" >> $GITHUB_ENV

echo "Successfully processed $file"
done

- name: Comment on PR
uses: actions/github-script@v7
if: always()
env:
CHANGED_FILES: ${{ steps.changed-files.outputs.all_changed_files }}
with:
script: |
const changedFiles = process.env.CHANGED_FILES || '';
let commentBody = `### SQL Export Summary\n\n`;

commentBody += `✅ **PR Status:** Merged\n\n`;

commentBody += `📝 **Processed SQL Files:**\n\n`;
if (changedFiles.trim()) {
commentBody += changedFiles.split(' ').map(f => `- ${f}`).join('\n');
} else {
commentBody += `None`;
}

commentBody += '\n\n**Status:** ${process.env.STATUS || 'Completed'}`;

await github.rest.issues.createComment({
...context.repo,
issue_number: context.issue.number,
body: commentBody
});
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
SELECT * FROM employee;
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
project = "sample-project"
instance = "prod-sample-instance"
database = "hr_prod"
format = "CSV" # Optional, defaults to "JSON" if not specified
Loading