diff --git a/.github/workflows/build-linux.yml b/.github/workflows/build-linux.yml index 38df80b75..d78959986 100644 --- a/.github/workflows/build-linux.yml +++ b/.github/workflows/build-linux.yml @@ -8,37 +8,37 @@ on: inputs: version_override: type: string - description: 'Version number override' + description: "Version number override" required: false run_unit_tests: type: boolean - description: 'Skip running unit tests' + description: "Skip running unit tests" required: false default: true runs_on: type: string - description: 'The GitHub hosted runner to use' + description: "The GitHub hosted runner to use" required: true OS: type: string description: > The operating system targeted by the build. - + There must be a corresponding Bundle_$OS.sh script file in ./Scripts required: true architecture: type: string - description: 'CPU architecture targeted by the build.' + description: "CPU architecture targeted by the build." required: true env: - DOTNET_CONFIGURATION: 'Release' - DOTNET_VERSION: '8.0.x' - RELEASE_NAME: 'chardonnay' + DOTNET_CONFIGURATION: "Release" + DOTNET_VERSION: "8.0.x" + RELEASE_NAME: "chardonnay" jobs: build: - name: '${{ inputs.OS }}-${{ inputs.architecture }}' + name: "${{ inputs.OS }}-${{ inputs.architecture }}" runs-on: ${{ inputs.runs_on }} steps: - uses: actions/checkout@v4 @@ -60,7 +60,7 @@ jobs: version="$(grep -Eio -m 1 '.*' ./Source/AppScaffolding/AppScaffolding.csproj | sed -r 's/<\/?Version>//g')" fi echo "version=${version}" >> "${GITHUB_OUTPUT}" - + - name: Unit test if: ${{ inputs.run_unit_tests }} working-directory: ./Source @@ -69,7 +69,7 @@ jobs: - name: Publish id: publish working-directory: ./Source - run: | + run: | if [[ "${{ inputs.OS }}" == "MacOS" ]] then display_os="macOS" @@ -78,13 +78,13 @@ jobs: display_os="Linux" RUNTIME_ID="linux-${{ inputs.architecture }}" fi - + OUTPUT="bin/Publish/${display_os}-${{ inputs.architecture }}-${{ env.RELEASE_NAME }}" - + echo "display_os=${display_os}" >> $GITHUB_OUTPUT echo "Runtime Identifier: $RUNTIME_ID" echo "Output Directory: $OUTPUT" - + dotnet publish \ LibationAvalonia/LibationAvalonia.csproj \ --runtime $RUNTIME_ID \ @@ -122,7 +122,7 @@ jobs: ${SCRIPT} "${BUNDLE_DIR}" "${{ steps.get_version.outputs.version }}" "${{ inputs.architecture }}" artifact=$(ls ./bundle) echo "artifact=${artifact}" >> "${GITHUB_OUTPUT}" - + - name: Publish bundle uses: actions/upload-artifact@v4 with: diff --git a/.github/workflows/build-windows.yml b/.github/workflows/build-windows.yml index 7b4c1ddf2..cff369b0e 100644 --- a/.github/workflows/build-windows.yml +++ b/.github/workflows/build-windows.yml @@ -8,21 +8,21 @@ on: inputs: version_override: type: string - description: 'Version number override' + description: "Version number override" required: false run_unit_tests: type: boolean - description: 'Skip running unit tests' + description: "Skip running unit tests" required: false default: true env: - DOTNET_CONFIGURATION: 'Release' - DOTNET_VERSION: '8.0.x' + DOTNET_CONFIGURATION: "Release" + DOTNET_VERSION: "8.0.x" jobs: build: - name: '${{ matrix.os }}-${{ matrix.release_name }}' + name: "${{ matrix.os }}-${{ matrix.release_name }}" runs-on: windows-latest strategy: matrix: @@ -112,4 +112,4 @@ jobs: name: ${{ steps.zip.outputs.artifact }}.zip path: ./Source/bin/Publish/${{ steps.zip.outputs.artifact }}.zip if-no-files-found: error - retention-days: 7 + retention-days: 7 diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index ce1186ac4..a67098ea0 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -8,22 +8,21 @@ on: inputs: version_override: type: string - description: 'Version number override' + description: "Version number override" required: false run_unit_tests: type: boolean - description: 'Skip running unit tests' + description: "Skip running unit tests" required: false - default: true + default: true jobs: - windows: uses: ./.github/workflows/build-windows.yml with: version_override: ${{ inputs.version_override }} run_unit_tests: ${{ inputs.run_unit_tests }} - + linux: strategy: matrix: @@ -36,7 +35,7 @@ jobs: OS: ${{ matrix.OS }} architecture: ${{ matrix.architecture }} run_unit_tests: ${{ inputs.run_unit_tests }} - + macos: strategy: matrix: @@ -47,4 +46,4 @@ jobs: runs_on: macos-latest OS: MacOS architecture: ${{ matrix.architecture }} - run_unit_tests: ${{ inputs.run_unit_tests }} + run_unit_tests: ${{ inputs.run_unit_tests }} diff --git a/.github/workflows/docker.yml b/.github/workflows/docker.yml index 7ea0fc0b4..d1a144797 100644 --- a/.github/workflows/docker.yml +++ b/.github/workflows/docker.yml @@ -8,7 +8,11 @@ on: inputs: version: type: string - description: 'Version number' + description: "Version number" + required: true + release: + type: boolean + description: "Is this a release build?" required: true secrets: docker_username: @@ -16,12 +20,10 @@ on: docker_token: required: true -env: - DOCKER_IMAGE: ${{ secrets.docker_username }}/libation - jobs: - docker: + build_and_push: runs-on: ubuntu-latest + steps: - name: Checkout uses: actions/checkout@v4 @@ -33,14 +35,29 @@ jobs: uses: docker/setup-buildx-action@v3 - name: Login to Docker Hub + if: ${{ inputs.release }} uses: docker/login-action@v3 with: username: ${{ secrets.docker_username }} password: ${{ secrets.docker_token }} - - name: Build and push - uses: docker/build-push-action@v5 + - name: Generate docker image tags + id: metadata + uses: docker/metadata-action@v5 + with: + flavor: | + latest=true + images: | + name=${{ secrets.docker_username }}/libation + tags: | + type=raw,value=${{ inputs.version }},enable=${{ inputs.release }} + + - name: Build and push image + uses: docker/build-push-action@v6 with: - push: true - build-args: 'FOLDER_NAME=Linux-chardonnay' - tags: ${{ env.DOCKER_IMAGE }}:latest,${{ env.DOCKER_IMAGE }}:${{ inputs.version }} + platforms: linux/amd64,linux/arm64 + push: ${{ steps.metadata.outputs.tags != ''}} + cache-from: type=gha + cache-to: type=gha,mode=max + tags: ${{ steps.metadata.outputs.tags }} + labels: ${{ steps.metadata.outputs.labels }} diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index fd1cc7314..7c6b136a6 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -5,7 +5,7 @@ name: release on: push: tags: - - 'v*' + - "v*" jobs: prerelease: runs-on: ubuntu-latest @@ -15,7 +15,7 @@ jobs: - name: Get tag version id: get_version run: | - export TAG='${{ github.ref_name }}' + export TAG="${{ github.ref_name }}" echo "version=${TAG#v}" >> "${GITHUB_OUTPUT}" docker: @@ -23,6 +23,7 @@ jobs: uses: ./.github/workflows/docker.yml with: version: ${{ needs.prerelease.outputs.version }} + release: true secrets: docker_username: ${{ secrets.DOCKERHUB_USERNAME }} docker_token: ${{ secrets.DOCKERHUB_TOKEN }} @@ -33,9 +34,9 @@ jobs: with: version_override: ${{ needs.prerelease.outputs.version }} run_unit_tests: false - + release: - needs: [prerelease,build] + needs: [prerelease, build] runs-on: ubuntu-latest steps: - name: Download artifacts @@ -55,7 +56,7 @@ jobs: - name: Upload release assets uses: dwenegar/upload-release-assets@v2 env: - GITHUB_TOKEN: '${{ secrets.GITHUB_TOKEN }}' + GITHUB_TOKEN: "${{ secrets.GITHUB_TOKEN }}" with: - release_id: '${{ steps.release.outputs.id }}' + release_id: "${{ steps.release.outputs.id }}" assets_path: ./artifacts diff --git a/.github/workflows/validate.yml b/.github/workflows/validate.yml index 0ae4c7123..27abc2750 100644 --- a/.github/workflows/validate.yml +++ b/.github/workflows/validate.yml @@ -1,5 +1,5 @@ # validate.yml -# Validates that Libation will build on a pull request or push to master. +# Validates that Libation will build on a pull request or push to master. --- name: validate @@ -12,3 +12,11 @@ on: jobs: build: uses: ./.github/workflows/build.yml + docker: + uses: ./.github/workflows/docker.yml + with: + version: ${GITHUB_SHA} + release: false + secrets: + docker_username: ${{ secrets.DOCKERHUB_USERNAME }} + docker_token: ${{ secrets.DOCKERHUB_TOKEN }} diff --git a/Docker/appsettings.json b/Docker/appsettings.json new file mode 100644 index 000000000..1a5525b3a --- /dev/null +++ b/Docker/appsettings.json @@ -0,0 +1,3 @@ +{ + "LibationFiles": "/config-internal" +} diff --git a/Docker/liberate.sh b/Docker/liberate.sh index 83249b871..73cd89667 100755 --- a/Docker/liberate.sh +++ b/Docker/liberate.sh @@ -1,68 +1,174 @@ #!/bin/bash -# Rewire echo to print date time -echo() { - if [[ -n $1 ]]; then - printf "$(date '+%F %T'): %s\n" "$1" +error() { + log "ERROR" "$1" +} + +warn() { + log "WARNING" "$1" +} + +info() { + log "info" "$1" +} + +debug() { + if [ "${LOG_LEVEL}" = "debug" ]; then + log "debug" "$1" + fi +} + +log() { + LEVEL=$1 + MESSAGE=$2 + printf "$(date '+%F %T') %s: %s\n" "${LEVEL}" "${MESSAGE}" +} + +init_config_file() { + FILE=$1 + FULLPATH=${LIBATION_CONFIG_DIR}/${FILE} + if [ -f ${FULLPATH} ]; then + info "loading ${FILE}" + cp ${FULLPATH} ${LIBATION_CONFIG_INTERNAL}/ + return 0 + else + warn "${FULLPATH} not found, creating empty file" + echo "{}" > ${LIBATION_CONFIG_INTERNAL}/${FILE} + return 1 + fi +} + +update_settings() { + FILE=$1 + KEY=$2 + VALUE=$3 + info "setting ${KEY} to ${VALUE}" + echo $(jq --arg k "${KEY}" --arg v "${VALUE}" '.[$k] = $v' ${LIBATION_CONFIG_INTERNAL}/${FILE}) > ${LIBATION_CONFIG_INTERNAL}/${FILE}.tmp + mv ${LIBATION_CONFIG_INTERNAL}/${FILE}.tmp ${LIBATION_CONFIG_INTERNAL}/${FILE} +} + +is_mounted() { + DIR=$1 + if grep -qs "${DIR} " /proc/mounts; + then + return 0 + else + return 1 + fi +} + +create_db() { + DBFILE=$1 + if [ -f "${DBFILE}" ]; then + warn "prexisting database found when creating" + return 0 + else + if ! touch "${DBFILE}"; then + error "unable to create database, check permissions on host" + exit 1 fi + return 1 + fi } -# ################################ -# Setup -# ################################ -echo "Starting" -if [[ -z "${SLEEP_TIME}" ]]; then - echo "No sleep time passed in. Will run once and exit." -else - echo "Sleep time is set to ${SLEEP_TIME}" -fi - -echo "" - -# Check if the config directory is passed in, and there is no link to it then create the link. -if [ -d "/config" ] && [ ! -d "/root/Libation" ]; then - echo "Linking config directory to the Libation config directory" - ln -s /config/ /root/Libation -fi - -# If no config error and exit -if [ ! -d "/config" ]; then - echo "ERROR: No /config directory. You must pass in a volume containing your config mapped to /config" - exit 1 -fi - -# If user passes in db from a /db/ folder and a db does not already exist / is not already linked -FILE=/db/LibationContext.db -if [ -f "${FILE}" ] && [ ! -f "/config/LibationContext.db" ]; then - echo "Linking passed in Libation database from /db/ to the Libation config directory" - ln -s $FILE /config/LibationContext.db -fi - -# Confirm we have a db in the config direcotry. -if [ ! -f "/config/LibationContext.db" ]; then - echo "ERROR: No Libation database detected, exiting." +setup_db() { + DBPATH=$1 + dbpattern="*.db" + + debug "using database directory ${DBPATH}" + + # Figure out the right databse file + if [[ -z "${LIBATION_DB_FILE}" ]]; + then + dbCount=$(find "${DBPATH}" -type f -name "${dbpattern}" | wc -l) + if [ "${dbCount}" -gt 1 ]; + then + error "too many database files found, set LIBATION_DB_FILE to the filename you wish to use" + exit 1 + elif [ "${dbCount}" -eq 1 ]; + then + files=( ${DBPATH}/${dbpattern} ) + FILE=${files[0]} + else + FILE="${DBPATH}/LibationContext.db" + fi + else + FILE="${DBPATH}/${LIBATION_DB_FILE}" + fi + + debug "planning to use database ${FILE}" + + if [ -f "${FILE}" ]; then + info "database found at ${FILE}" + elif [ ${LIBATION_CREATE_DB} = "true" ]; + then + warn "database not found, creating one at ${FILE}" + create_db ${FILE} + else + error "database not found and creation is disabled" exit 1 -fi - -# ################################ -# Loop and liberate -# ################################ -while true -do - echo "" - echo "Scanning accounts" - /libation/LibationCli scan - echo "Liberating books" - /libation/LibationCli liberate - echo "" + fi + ln -s "${FILE}" "${LIBATION_CONFIG_INTERNAL}/LibationContext.db" +} + +run() { + info "scanning accounts" + /libation/LibationCli scan + info "liberating books" + /libation/LibationCli liberate +} + +main() { + info "initializing libation" + init_config_file AccountsSettings.json + init_config_file Settings.json + + info "loading settings" + update_settings Settings.json Books /data + update_settings Settings.json InProgress /tmp + + info "loading database" + # If user provides a separate database mount, use that + if is_mounted "${LIBATION_DB_DIR}"; + then + DB_LOCATION=${LIBATION_DB_DIR} + # Otherwise, use the config directory + else + DB_LOCATION=${LIBATION_CONFIG_DIR} + fi + setup_db ${DB_LOCATION} + + # Try to warn if books dir wasn't mounted in + if ! is_mounted "${LIBATION_BOOKS_DIR}"; + then + warn "${LIBATION_BOOKS_DIR} does not appear to be mounted, books will not be saved" + fi + + # Let the user know what the run type will be + if [[ -z "${SLEEP_TIME}" ]]; then + SLEEP_TIME=-1 + fi + + if [ "${SLEEP_TIME}" -eq -1 ]; then + info "running once" + else + info "running every ${SLEEP_TIME}" + fi + + # loop + while true + do + run # Liberate only once if SLEEP_TIME was set to -1 - if [ "${SLEEP_TIME}" = -1 ]; then + if [ "${SLEEP_TIME}" -eq -1 ]; then break fi - echo "Sleeping for ${SLEEP_TIME}" sleep "${SLEEP_TIME}" -done + done + + info "exiting" +} -echo "Exiting" \ No newline at end of file +main diff --git a/Dockerfile b/Dockerfile index ef2748421..7e9eb9bed 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,22 +1,39 @@ # Dockerfile -FROM mcr.microsoft.com/dotnet/sdk:8.0 as build-env +FROM --platform=${BUILDPLATFORM} mcr.microsoft.com/dotnet/sdk:8.0 AS build +ARG TARGETARCH COPY Source /Source -RUN dotnet publish -c Release -o /Source/bin/Publish/Linux-chardonnay /Source/LibationCli/LibationCli.csproj -p:PublishProfile=/Source/LibationCli/Properties/PublishProfiles/LinuxProfile.pubxml -COPY Docker/liberate.sh /Source/bin/Publish/Linux-chardonnay - +RUN dotnet publish \ + /Source/LibationCli/LibationCli.csproj \ + --arch ${TARGETARCH} \ + --configuration Release \ + --output /Source/bin/Publish/Linux-chardonnay \ + -p:PublishProfile=/Source/LibationCli/Properties/PublishProfiles/LinuxProfile.pubxml FROM mcr.microsoft.com/dotnet/runtime:8.0 +ARG USER_UID=1001 +ARG USER_GID=1001 + +# Set the character set that will be used for folder and filenames when liberating +ENV LANG=C.UTF-8 +ENV LC_ALL=C.UTF-8 -ENV SLEEP_TIME "30m" +ENV SLEEP_TIME=-1 +ENV LIBATION_CONFIG_INTERNAL=/config-internal +ENV LIBATION_CONFIG_DIR=/config +ENV LIBATION_DB_DIR=/db +ENV LIBATION_DB_FILE= +ENV LIBATION_CREATE_DB=true +ENV LIBATION_BOOKS_DIR=/data -# Sets the character set that will be used for folder and filenames when liberating -ENV LANG C.UTF-8 -ENV LC_ALL C.UTF-8 -RUN mkdir /db /config /data +RUN apt-get update && apt-get -y upgrade && \ + apt-get install -y jq && \ + mkdir -m777 ${LIBATION_CONFIG_INTERNAL} ${LIBATION_BOOKS_DIR} -COPY --from=build-env /Source/bin/Publish/Linux-chardonnay /libation +COPY --from=build /Source/bin/Publish/Linux-chardonnay /libation +COPY Docker/* /libation +USER ${USER_UID}:${USER_GID} -CMD ["./libation/liberate.sh"] +CMD ["/libation/liberate.sh"] diff --git a/Documentation/Docker.md b/Documentation/Docker.md index 9d75b6dda..09ca1cdbf 100644 --- a/Documentation/Docker.md +++ b/Documentation/Docker.md @@ -3,46 +3,65 @@ ### If you found this useful, tell a friend. If you found this REALLY useful, you can click here to [PayPal.me](https://paypal.me/mcrackan?locale.x=en_us) ...or just tell more friends. As long as I'm maintaining this software, it will remain **free** and **open source**. - +> [!WARNING] +> ## Breaking Changes +> * The docker image now runs as user 1001 and group 1001. Make sure that the permissions on your volumes allow user 1001 to read and write to them. +> * `SLEEP_TIME` is now set to `-1` by default. This means the image will run once and exit. If you were relying on the previous default, you'll need to explicitly set the `SLEEP_TIME` environment variable to `30m` to replicate the previous behavior. +> * The docker image now ignores the values in `Settings.json` for `Books` and `InProgress`. You can now change the folder that books are saved to by using the `LIBATION_BOOKS_DIR` environment variable. # Disclaimer The docker image is provided as-is. We hope it can be useful to you but it is not officially supported. -### Setup -In order to use the docker image, you'll need to provide it with a copy of the `AccountsSettings.json`, `Settings.json`, and `LibationContext.db` files. These files can usually be found in the Libation folder in your user's home directory. If you haven't run Libation yet, you'll need to launch it to generate these files and setup your accounts. Once you have them, copy these files to a new location, such as `/opt/libation/config`. Before using them we'll need to make a couple edits so that the filepaths referenced are correct when running from the docker image. - -In Settings.json, make the following changes: -* Change `Books` to `/data` -* Change `InProgress` to `/tmp` * - -*You may have to paste the following at the end of your your Settings.json file if `InProgess` is not present: +### Configuration +Configuration in Libation is handled by two files, `AccountsSettings.json` and `Settings.json`. These files can usually be found in the Libation folder in your user's home directory. The easiest way to configure these is to run the desktop version of Libation and then copy them into a folder, such as `/opt/libation/config`, that you'll volume mount into the image. `Settings.json` is technically optional, and, if not provided, Libation will run using the default settings. Additionally, the `Books` and `InProgress` settings in `Settings.json` will be ignored and the image will instead substitute it's own values. +### Running +Once the configuration files are copied, the docker image can be run with the following command. ``` - "InProgress": "/tmp" +sudo docker run -d \ + -v /opt/libation/config:/config \ + -v /opt/libation/books:/data \ + --name libation \ + --restart=always \ + rmcrackan/libation:latest ``` -![image](https://github.com/patienttruth/Libation/assets/105557996/cf65a108-cadf-4284-9000-e7672c99c59b) +By default the container will scan for new books once and download any new ones. This is configurable by passing in a value for the `SLEEP_TIME` environment variable. For example, if you pass in `10m` it will keep running, scan for new books, and download them every 10 minutes. -### Running -Once the configuration files are copied and edited, the docker image can be run with the following command. ``` sudo docker run -d \ -v /opt/libation/config:/config \ -v /opt/libation/books:/data \ + -e SLEEP_TIME='10m' \ --name libation \ --restart=always \ - rmcrackan/libation + rmcrackan/libation:latest ``` -By default the container will scan for new books every 30 minutes and download any new ones. This is configurable by passing in a value for the `SLEEP_TIME` environment variable. Additionally, if you pass in `-1` it will scan and download books once and then exit. +### Environment Variables +| Env Var | Default | Description | +| -------- | ------- | ----------- | +| SLEEP_TIME | -1 | Length of time to sleep before doing another scan/download. Set to -1 to run one. | +| LIBATION_BOOKS_DIR | /data | Folder where books will be saved | +| LIBATION_CONFIG_DIR | /config | Folder to read configuration from. | +| LIBATION_DB_DIR | /db | Optional folder to load database from. If not mounted, will load database from `LIBATION_CONFIG_DIR`. | +| LIBATION_DB_FILE | | Name of database file to load. By default it will look for all `.db` files and load one if there is only one present. | +| LIBATION_CREATE_DB | true | Whether or not the image should create a database file if none are found. | + +### User +This docker image runs as user `1001`. In order for the image to function properly, user `1001` must be able to read and write the volumes that are mounted in. If they are not, you will see errors + +If you want to change the user the image runs as, you can specify `-u :`. For example, to run it as user `2000` and group `3000`, you could do the following: ``` sudo docker run -d \ + -u 2000:3000 \ -v /opt/libation/config:/config \ -v /opt/libation/books:/data \ - -e SLEEP_TIME='10m' \ --name libation \ --restart=always \ - rmcrackan/libation + rmcrackan/libation:latest ``` +### Advanced Database Options +The docker image supports an optional database mount location defined by `LIBATION_DB_DIR`. This allows the database to be mounted as read/write, while allowing the rest of the configuration files to be mounted as read only. This is specifically useful if running in Kubernetes where you can use Configmaps and Secrets to define the configuration. If the `LIBATION_DB_DIR` is mounted, it will be used, otherwise it will look for the database in `LIBATION_CONFIG_DIR`. If it does not find the database in the expected location, it will attempt to make an empty database there. \ No newline at end of file