diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 66373dad..da4344ad 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -29,16 +29,32 @@ jobs: BOOST_ROOT: ${{ github.workspace }}/lib/boost BOOST_URL: https://sourceforge.net/projects/boost/files/boost/1.76.0/boost_1_76_0.tar.bz2/download steps: + - name: Install Deno + uses: denoland/setup-deno@v2 + - name: Deactivate EOL conversion + shell: bash + run: | + git config --global core.autocrlf false + git config --global core.eol lf - name: Checkout repository uses: actions/checkout@v4 with: lfs: true + - uses: actions/setup-python@v5 + with: + python-version: '3.13' + - name: Install Python wheels + shell: bash + run: pip install -r requirements.txt - name: Restore Boost from cache uses: actions/cache@v4 id: cache-boost with: path: ${{ env.BOOST_ROOT }} key: ${{ env.BOOST_URL }} + - name: Lint + shell: bash + run: doit check-formatted - name: Download Boost if: steps.cache-boost.outputs.cache-hit != 'true' shell: bash @@ -49,14 +65,13 @@ jobs: fi mkdir -p $BOOST_ROOT curl --insecure -L $BOOST_URL | tar -xj --strip-components=1 -C $BOOST_ROOT - - name: Build Rhubarb + - name: Build and package shell: bash run: | JAVA_HOME=$JAVA_HOME_11_X64 - mkdir build - cd build - cmake ${{ matrix.cmakeOptions }} .. - cmake --build . --config Release --target package + mkdir rhubarb/build + (cd rhubarb/build && cmake ${{ matrix.cmakeOptions }} ..) + doit package - name: Run tests shell: bash run: | diff --git a/.gitignore b/.gitignore index 681f8b44..a996eb29 100644 --- a/.gitignore +++ b/.gitignore @@ -1,6 +1,7 @@ .vs/ .vscode/ *.user +artifacts/ build/ venv/ diff --git a/CMakeLists.txt b/CMakeLists.txt deleted file mode 100644 index d287005b..00000000 --- a/CMakeLists.txt +++ /dev/null @@ -1,38 +0,0 @@ -cmake_minimum_required(VERSION 3.2) - -include(app-info.cmake) - -project(${appName}) - -# Build and install main executable -add_subdirectory(rhubarb) - -# Build and install extras -add_subdirectory("extras/adobe-after-effects") -add_subdirectory("extras/magix-vegas") -add_subdirectory("extras/esoteric-software-spine") - -# Install misc. files -install(FILES README.adoc LICENSE.md CHANGELOG.md DESTINATION .) - -# Configure CPack -function(get_short_system_name variable) - if("${CMAKE_SYSTEM_NAME}" STREQUAL "Darwin") - set(${variable} "macOS" PARENT_SCOPE) - else() - set(${variable} "${CMAKE_SYSTEM_NAME}" PARENT_SCOPE) - endif() -endfunction() - -set(CPACK_PACKAGE_NAME ${appName}) -string(REPLACE " " "-" CPACK_PACKAGE_NAME "${CPACK_PACKAGE_NAME}") -get_short_system_name(CPACK_SYSTEM_NAME) -set(CPACK_PACKAGE_VERSION_MAJOR ${appVersionMajor}) -set(CPACK_PACKAGE_VERSION_MINOR ${appVersionMinor}) -set(CPACK_PACKAGE_VERSION_PATCH ${appVersionPatch}) -set(CPACK_PACKAGE_VERSION ${appVersion}) -set(CPACK_PACKAGE_FILE_NAME "${CPACK_PACKAGE_NAME}-${CPACK_PACKAGE_VERSION}-${CPACK_SYSTEM_NAME}") -set(CPACK_GENERATOR ZIP) - -# Run CPack -include(CPack) diff --git a/app-info.cmake b/app-info.cmake deleted file mode 100644 index 0c9fdf28..00000000 --- a/app-info.cmake +++ /dev/null @@ -1,8 +0,0 @@ -cmake_minimum_required(VERSION 3.2) - -set(appName "Rhubarb Lip Sync") -set(appVersionMajor 1) -set(appVersionMinor 13) -set(appVersionPatch 0) -set(appVersionSuffix "") -set(appVersion "${appVersionMajor}.${appVersionMinor}.${appVersionPatch}${appVersionSuffix}") diff --git a/app-info.toml b/app-info.toml new file mode 100644 index 00000000..05a75e46 --- /dev/null +++ b/app-info.toml @@ -0,0 +1,4 @@ +appName = "Rhubarb Lip Sync" + +# Can be any valid SemVer version, including suffixes +appVersion = "1.13.0" diff --git a/dodo.py b/dodo.py index 8dd92927..00d126a4 100644 --- a/dodo.py +++ b/dodo.py @@ -6,10 +6,14 @@ from gitignore_parser import parse_gitignore from typing import Dict, Optional, List from enum import Enum - +from shutil import rmtree, copy, copytree, make_archive +import platform +import tomllib root_dir = Path(__file__).parent rhubarb_dir = root_dir / 'rhubarb' +rhubarb_build_dir = rhubarb_dir / 'build' +extras_dir = root_dir / 'extras' def task_format(): @@ -71,6 +75,100 @@ def format(files: List[Path], formatter: Formatter, *, check_only: bool = False) raise ValueError(f'Unknown formatter: {formatter}') +def task_configure_rhubarb(): + """Configure CMake for the Rhubarb binary""" + + def configure_rhubarb(): + ensure_dir(rhubarb_build_dir) + subprocess.run(['cmake', '..'], cwd=rhubarb_build_dir, check=True) + + return {'basename': 'configure-rhubarb', 'actions': [configure_rhubarb]} + + +def task_build_rhubarb(): + """Build the Rhubarb binary""" + + def build_rhubarb(): + subprocess.run( + ['cmake', '--build', '.', '--config', 'Release'], cwd=rhubarb_build_dir, check=True + ) + + return {'basename': 'build-rhubarb', 'actions': [build_rhubarb]} + + +def task_build_spine(): + """Build Rhubarb for Spine""" + + def build_spine(): + onWindows = platform.system() == 'Windows' + subprocess.run( + ['gradlew.bat' if onWindows else './gradlew', 'build'], + cwd=extras_dir / 'esoteric-software-spine', + check=True, + shell=onWindows, + ) + + return {'basename': 'build-spine', 'actions': [build_spine]} + + +def task_package(): + """Package all artifacts into an archive file""" + + with open(root_dir / 'app-info.toml', 'rb') as file: + appInfo = tomllib.load(file) + + os_name = 'macOS' if platform.system() == 'Darwin' else platform.system() + file_name = f"{appInfo['appName'].replace(' ', '-')}-{appInfo['appVersion']}-{os_name}" + + artifacts_dir = ensure_empty_dir(root_dir / 'artifacts') + tree_dir = ensure_dir(artifacts_dir.joinpath(file_name)) + + def collect_artifacts(): + # Misc. files + copy(root_dir / 'README.adoc', tree_dir) + copy(root_dir / 'LICENSE.md', tree_dir) + copy(root_dir / 'CHANGELOG.md', tree_dir) + copytree(root_dir / 'img', tree_dir / 'img') + + # Rhubarb + subprocess.run( + ['cmake', '--install', '.', '--prefix', tree_dir], cwd=rhubarb_build_dir, check=True + ) + + # Adobe After Effects script + src = extras_dir / 'adobe-after-effects' + dst_extras_dir = ensure_dir(tree_dir / 'extras') + dst = ensure_dir(dst_extras_dir / 'adobe-after-effects') + copy(src / 'README.adoc', dst) + copy(src / 'Rhubarb Lip Sync.jsx', dst) + + # Rhubarb for Spine + src = extras_dir / 'esoteric-software-spine' + dst = ensure_dir(dst_extras_dir / 'esoteric-software-spine') + copy(src / 'README.adoc', dst) + for file in (src / 'build' / 'libs').iterdir(): + copy(file, dst) + + # Magix Vegas + src = extras_dir / 'magix-vegas' + dst = ensure_dir(dst_extras_dir / 'magix-vegas') + copy(src / 'README.adoc', dst) + copy(src / 'Debug Rhubarb.cs', dst) + copy(src / 'Debug Rhubarb.cs.config', dst) + copy(src / 'Import Rhubarb.cs', dst) + copy(src / 'Import Rhubarb.cs.config', dst) + + def pack_artifacts(): + zip_base_name = tree_dir + format = 'gztar' if platform.system() == 'Linux' else 'zip' + make_archive(zip_base_name, format, tree_dir) + + return { + 'actions': [collect_artifacts, pack_artifacts], + 'task_dep': ['build-rhubarb', 'build-spine'], + } + + @cache def get_files_by_formatters() -> Dict[Formatter, List[Path]]: """Returns a dict with all formattable code files grouped by formatter.""" @@ -115,3 +213,19 @@ def get_formatter(path: Path) -> Optional[Formatter]: return Formatter.PRETTIER case '.py': return Formatter.RUFF + + +def ensure_dir(dir: Path) -> Path: + """Makes sure the given directory exists.""" + + if not dir.exists(): + dir.mkdir() + return dir + + +def ensure_empty_dir(dir: Path) -> Path: + """Makes sure the given directory exists and is empty.""" + + if dir.exists(): + rmtree(dir) + return ensure_dir(dir) diff --git a/extras/adobe-after-effects/CMakeLists.txt b/extras/adobe-after-effects/CMakeLists.txt deleted file mode 100644 index e74b9fa4..00000000 --- a/extras/adobe-after-effects/CMakeLists.txt +++ /dev/null @@ -1,5 +0,0 @@ -cmake_minimum_required(VERSION 3.2) - -set(afterEffectsFiles "Rhubarb Lip Sync.jsx" "README.adoc") - -install(FILES ${afterEffectsFiles} DESTINATION "extras/adobe-after-effects") diff --git a/extras/esoteric-software-spine/CMakeLists.txt b/extras/esoteric-software-spine/CMakeLists.txt deleted file mode 100644 index 33d8b9c6..00000000 --- a/extras/esoteric-software-spine/CMakeLists.txt +++ /dev/null @@ -1,13 +0,0 @@ -cmake_minimum_required(VERSION 3.2) - -add_custom_target( - rhubarbForSpine - ALL - "./gradlew" "build" - WORKING_DIRECTORY "${CMAKE_CURRENT_SOURCE_DIR}" - COMMENT "Building Rhubarb for Spine through Gradle." -) - -install(DIRECTORY "build/libs/" DESTINATION "extras/esoteric-software-spine") - -install(FILES README.adoc DESTINATION "extras/esoteric-software-spine") diff --git a/extras/esoteric-software-spine/build.gradle.kts b/extras/esoteric-software-spine/build.gradle.kts index 41fad7b2..387062f2 100644 --- a/extras/esoteric-software-spine/build.gradle.kts +++ b/extras/esoteric-software-spine/build.gradle.kts @@ -8,13 +8,9 @@ plugins { fun getVersion(): String { // Dynamically read version from CMake file - val file = File(rootDir.parentFile.parentFile, "app-info.cmake") + val file = File(rootDir.parentFile.parentFile, "app-info.toml") val text = file.readText() - val major = Regex("""appVersionMajor\s+(\d+)""").find(text)!!.groupValues[1] - val minor = Regex("""appVersionMinor\s+(\d+)""").find(text)!!.groupValues[1] - val patch = Regex("""appVersionPatch\s+(\d+)""").find(text)!!.groupValues[1] - val suffix = Regex("""appVersionSuffix\s+"(.*?)"""").find(text)!!.groupValues[1] - return "$major.$minor.$patch$suffix" + return Regex("""appVersion\s*=\s*"(.*?)"(?:)""").find(text)!!.groupValues[1] } group = "com.rhubarb_lip_sync" diff --git a/extras/magix-vegas/CMakeLists.txt b/extras/magix-vegas/CMakeLists.txt deleted file mode 100644 index c796b4bb..00000000 --- a/extras/magix-vegas/CMakeLists.txt +++ /dev/null @@ -1,11 +0,0 @@ -cmake_minimum_required(VERSION 3.2) - -set(vegasFiles - "Debug Rhubarb.cs" - "Debug Rhubarb.cs.config" - "Import Rhubarb.cs" - "Import Rhubarb.cs.config" - "README.adoc" -) - -install(FILES ${vegasFiles} DESTINATION "extras/magix-vegas") diff --git a/package-osx.sh b/package-osx.sh deleted file mode 100755 index 11ae535d..00000000 --- a/package-osx.sh +++ /dev/null @@ -1,6 +0,0 @@ -#!/bin/sh -rm -rf build -mkdir build -cd build -cmake .. -G Xcode -cmake --build . --config Release --target package diff --git a/package-win.bat b/package-win.bat deleted file mode 100644 index c147b4e8..00000000 --- a/package-win.bat +++ /dev/null @@ -1,5 +0,0 @@ -rmdir /s /q build -mkdir build -cd build -cmake .. -G "Visual Studio 16 2019" -cmake --build . --config Release --target package diff --git a/rhubarb/CMakeLists.txt b/rhubarb/CMakeLists.txt index 69cc6ddf..c1e0acad 100644 --- a/rhubarb/CMakeLists.txt +++ b/rhubarb/CMakeLists.txt @@ -1,6 +1,13 @@ cmake_minimum_required(VERSION 3.10) -include("../app-info.cmake") +# Parse app info +file(READ "../app-info.toml" tomlContent) +string(REGEX MATCH "appName *= *\"[^\"]+\"" appName "${tomlContent}") +string(REGEX REPLACE ".*\"([^\"]+)\"" "\\1" appName "${appName}") +string(REGEX MATCH "appVersion *= *\"[^\"]+\"" appVersion "${tomlContent}") +string(REGEX REPLACE ".*\"([^\"]+)\"" "\\1" appVersion "${appVersion}") + +project("${appName}") # Support legacy OS X versions set(CMAKE_OSX_DEPLOYMENT_TARGET "10.10" CACHE STRING "Minimum OS X deployment version") @@ -94,6 +101,7 @@ set_target_properties(pocketSphinx PROPERTIES FOLDER lib) include_directories(SYSTEM "lib/tclap-1.2.1/include") # ... Google Test +set(INSTALL_GTEST OFF) # Prevent library files from ending up in our artifacts add_subdirectory("lib/googletest") target_compile_options(gmock PRIVATE ${disableWarningsFlags}) set_target_properties(gmock PROPERTIES FOLDER lib) @@ -327,7 +335,7 @@ target_link_libraries( ) # ... rhubarb-core -configure_file(src/core/app-info.cpp.in app-info.cpp ESCAPE_QUOTES) +configure_file(src/core/app-info.cpp.in app-info.cpp) add_library( rhubarb-core ${CMAKE_CURRENT_BINARY_DIR}/app-info.cpp @@ -560,7 +568,4 @@ endfunction() copy_and_install("lib/pocketsphinx-rev13216/model/en-us/*" "res/sphinx") copy_and_install("lib/cmusphinx-en-us-5.2/*" "res/sphinx/acoustic-model") - -copy_and_install("tests/resources/*" "tests/resources") - install(TARGETS rhubarb RUNTIME DESTINATION .)