Fix Mac dmg build (#138)
* [mac/build] update build steps * [macos] bundle app * [macos] build dmg * [ci] setup macOS CI * [i18n] TODO: rm WX locale files * [deps] set main branch to main * Fix osx-fix-libs.py Create symbolic links, to make libicu happy Rewrite the script in python3, as python2 is deprecated * Add write permission before install_name_tool when doing osx-bundle Fix wangqr/Aegisub#39 * Handle @loader_path in libboost on macOS See wangqr/Aegisub#39 * [tools/mac] use python3 * [ci/mac] install & using system deps * [ci/win] don't build fribidi:docs * [ci/mac] trying openal-soft * [ci/mac] use pulseaudio in CI * [ci/win] only run aeg's test * [ci/win] fix CI * [ci/win] fix CI: not use dict `{}` * [ci] run ci * [ci/win] don't build docs * [ci/win] remove args tail newline * [ci/win] false->disabled * Use md title format Co-authored-by: Ryan Lucia <ryan@luciaonline.net> * Recover file permissions. * [ci/win] disable fontconfig Co-Authored-By: Ryan Lucia <ryan@luciaonline.net> * [ci/win] disable libass:fontconfig Co-authored-by: wangqr <wangqr@wangqr.tk> Co-authored-by: Ryan Lucia <ryan@luciaonline.net>
This commit is contained in:
parent
0dffcec461
commit
e58e4a9149
9 changed files with 218 additions and 137 deletions
50
.github/workflows/ci.yml
vendored
50
.github/workflows/ci.yml
vendored
|
@ -14,13 +14,23 @@ jobs:
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
config:
|
config:
|
||||||
- {
|
- name: Windows MSVC Release
|
||||||
name: Windows MSVC Release,
|
os: windows-latest
|
||||||
os: windows-latest,
|
msvc: true
|
||||||
msvc: true,
|
buildtype: release
|
||||||
buildtype: release,
|
args: >-
|
||||||
args: '-Ddefault_library=static --force-fallback-for=zlib,harfbuzz,freetype2,fribidi,libpng -Dfreetype2:harfbuzz=disabled -Dharfbuzz:freetype=disabled -Dharfbuzz:cairo=disabled -Dharfbuzz:glib=disabled -Dharfbuzz:gobject=disabled'
|
-Ddefault_library=static
|
||||||
}
|
--force-fallback-for=zlib,harfbuzz,freetype2,fribidi,libpng
|
||||||
|
-Dfreetype2:harfbuzz=disabled
|
||||||
|
-Dharfbuzz:freetype=disabled
|
||||||
|
-Dharfbuzz:cairo=disabled
|
||||||
|
-Dharfbuzz:glib=disabled
|
||||||
|
-Dharfbuzz:gobject=disabled
|
||||||
|
-Dharfbuzz:tests=disabled
|
||||||
|
-Dharfbuzz:docs=disabled
|
||||||
|
-Dfribidi:tests=false
|
||||||
|
-Dfribidi:docs=false
|
||||||
|
-Dlibass:fontconfig=disabled
|
||||||
#- {
|
#- {
|
||||||
# name: Windows MinGW,
|
# name: Windows MinGW,
|
||||||
# os: windows-latest,
|
# os: windows-latest,
|
||||||
|
@ -42,13 +52,13 @@ jobs:
|
||||||
name: macOS Debug,
|
name: macOS Debug,
|
||||||
os: macos-latest,
|
os: macos-latest,
|
||||||
buildtype: debugoptimized,
|
buildtype: debugoptimized,
|
||||||
args: -Ddefault_library=static
|
args: -Ddefault_library=static -Dbuild_osx_bundle=true -Dlocal_boost=true
|
||||||
}
|
}
|
||||||
- {
|
- {
|
||||||
name: macOS Release,
|
name: macOS Release,
|
||||||
os: macos-latest,
|
os: macos-latest,
|
||||||
buildtype: release,
|
buildtype: release,
|
||||||
args: -Ddefault_library=static
|
args: -Ddefault_library=static -Dbuild_osx_bundle=true -Dlocal_boost=true
|
||||||
}
|
}
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
@ -83,9 +93,12 @@ jobs:
|
||||||
- name: Install dependencies (MacOS)
|
- name: Install dependencies (MacOS)
|
||||||
if: matrix.config.os == 'macos-latest'
|
if: matrix.config.os == 'macos-latest'
|
||||||
run: |
|
run: |
|
||||||
brew install luarocks nasm ninja
|
brew update
|
||||||
|
brew install luarocks ninja
|
||||||
luarocks install luafilesystem 1.8.0
|
luarocks install luafilesystem 1.8.0
|
||||||
luarocks install moonscript --dev
|
luarocks install moonscript --dev
|
||||||
|
brew install libass zlib ffms2 fftw hunspell
|
||||||
|
brew install pulseaudio # NO OpenAL in github CI
|
||||||
|
|
||||||
- name: Install dependencies (Linux)
|
- name: Install dependencies (Linux)
|
||||||
if: matrix.config.os == 'ubuntu-latest'
|
if: matrix.config.os == 'ubuntu-latest'
|
||||||
|
@ -100,8 +113,9 @@ jobs:
|
||||||
run: meson compile -C build
|
run: meson compile -C build
|
||||||
|
|
||||||
- name: Run test
|
- name: Run test
|
||||||
run: meson test -C build --verbose
|
run: meson test -C build --verbose "gtest main"
|
||||||
|
|
||||||
|
# Windows artifacts
|
||||||
- name: Generate Windows installer
|
- name: Generate Windows installer
|
||||||
if: matrix.config.os == 'windows-latest'
|
if: matrix.config.os == 'windows-latest'
|
||||||
run: meson compile win-installer -C build
|
run: meson compile win-installer -C build
|
||||||
|
@ -123,3 +137,17 @@ jobs:
|
||||||
with:
|
with:
|
||||||
name: ${{ matrix.config.name }} - portable
|
name: ${{ matrix.config.name }} - portable
|
||||||
path: build/aegisub-portable-64.zip
|
path: build/aegisub-portable-64.zip
|
||||||
|
|
||||||
|
# macOS artifacts
|
||||||
|
- name: Generate macOS installer
|
||||||
|
if: matrix.config.os == 'macos-latest'
|
||||||
|
run: |
|
||||||
|
meson compile osx-bundle -C build
|
||||||
|
meson compile osx-build-dmg -C build
|
||||||
|
|
||||||
|
- name: Upload artifacts - macOS dmg
|
||||||
|
uses: actions/upload-artifact@v2
|
||||||
|
if: matrix.config.os == 'macos-latest'
|
||||||
|
with:
|
||||||
|
name: ${{ matrix.config.name }} - installer
|
||||||
|
path: build/Aegisub-*.dmg
|
||||||
|
|
21
README.md
21
README.md
|
@ -48,15 +48,24 @@ A vaguely recent version of Xcode and the corresponding command-line tools are r
|
||||||
|
|
||||||
For personal usage, you can use pip and homebrew to install almost all of Aegisub's dependencies:
|
For personal usage, you can use pip and homebrew to install almost all of Aegisub's dependencies:
|
||||||
|
|
||||||
brew install libass nasm ninja boost zlib icu4c pkg-config ffms2 fftw hunspell gettext cmake
|
pip3 install meson
|
||||||
brew link --force gettext
|
brew install cmake ninja pkg-config libass boost zlib ffms2 fftw hunspell
|
||||||
export LDFLAGS="-L/usr/local/opt/icu4c/lib"
|
export LDFLAGS="-L/usr/local/opt/icu4c/lib"
|
||||||
export CPPFLAGS="-I/usr/local/opt/icu4c/include"
|
export CPPFLAGS="-I/usr/local/opt/icu4c/include"
|
||||||
export PKG_CONFIG_PATH="/usr/local/opt/icu4c/lib/pkgconfig"
|
export PKG_CONFIG_PATH="/usr/local/opt/icu4c/lib/pkgconfig"
|
||||||
pip install meson
|
|
||||||
|
|
||||||
Once the dependencies are installed, build Aegisub with `meson build && meson compile -C build`.
|
Once the dependencies are installed, build Aegisub with `meson build && meson compile -C build`.
|
||||||
|
|
||||||
|
#### Build dmg
|
||||||
|
|
||||||
|
```bash
|
||||||
|
meson build_static -Ddefault_library=static -Dbuildtype=debugoptimized -Dbuild_osx_bundle=true -Dlocal_boost=true
|
||||||
|
meson compile -C build_static
|
||||||
|
meson test -C build_static --verbose
|
||||||
|
meson compile osx-bundle -C build_static
|
||||||
|
meson compile osx-build-dmg -C build_static
|
||||||
|
```
|
||||||
|
|
||||||
## Updating Moonscript
|
## Updating Moonscript
|
||||||
|
|
||||||
From within the Moonscript repository, run `bin/moon bin/splat.moon -l moonscript moonscript/ > bin/moonscript.lua`.
|
From within the Moonscript repository, run `bin/moon bin/splat.moon -l moonscript moonscript/ > bin/moonscript.lua`.
|
||||||
|
|
|
@ -8,9 +8,13 @@ if host_machine.system() == 'windows'
|
||||||
run_target('win-portable', command: [portable_setup, meson.project_build_root(), meson.project_source_root()])
|
run_target('win-portable', command: [portable_setup, meson.project_build_root(), meson.project_source_root()])
|
||||||
elif host_machine.system() == 'darwin'
|
elif host_machine.system() == 'darwin'
|
||||||
fontconfig_conf = run_command('pkg-config', '--variable=confdir', 'fontconfig').stdout().strip()
|
fontconfig_conf = run_command('pkg-config', '--variable=confdir', 'fontconfig').stdout().strip()
|
||||||
|
bundle_app_sh = find_program(meson.project_source_root() / 'tools/osx-bundle.sh')
|
||||||
run_target('osx-bundle',
|
run_target('osx-bundle',
|
||||||
command: ['../tools/osx-bundle.sh', meson.project_source_root(), meson.project_build_root(), 'wx-config', fontconfig_conf, '',
|
command: [bundle_app_sh, meson.project_source_root(), meson.project_build_root(), 'wx-config', fontconfig_conf, '',
|
||||||
get_option('build_osx_bundle') ? 'TRUE' : 'FALSE'])
|
get_option('build_osx_bundle') ? 'TRUE' : 'FALSE'])
|
||||||
|
build_dmg_sh = find_program(meson.project_source_root() / 'tools/osx-dmg.sh')
|
||||||
|
run_target('osx-build-dmg',
|
||||||
|
command: [build_dmg_sh, meson.project_source_root(), meson.project_build_root(), meson.project_version()])
|
||||||
else
|
else
|
||||||
conf_pkg.set('AEGISUB_COMMAND', 'aegisub')
|
conf_pkg.set('AEGISUB_COMMAND', 'aegisub')
|
||||||
|
|
||||||
|
|
|
@ -34,7 +34,8 @@ Source: {#BUILD_ROOT}\po\zh_TW.gmo; DestDir: {app}\locale\zh_TW; DestName:
|
||||||
#endif
|
#endif
|
||||||
; END ENABLE_TRANSLATIONS
|
; END ENABLE_TRANSLATIONS
|
||||||
|
|
||||||
|
;; TODO: rm those lines
|
||||||
|
;; xref: [Update and review translations · Issue #132 · TypesettingTools/Aegisub](https://github.com/TypesettingTools/Aegisub/issues/132)
|
||||||
#ifdef ENABLE_WX_TRANSLATIONS
|
#ifdef ENABLE_WX_TRANSLATIONS
|
||||||
; wxWidgets localization (commented out ones are out of date; some don't have wxstd.mo)
|
; wxWidgets localization (commented out ones are out of date; some don't have wxstd.mo)
|
||||||
Source: src\mo\wxstd-ar.mo; DestDir: {app}\locale\ar; DestName: wxstd.mo; Flags: ignoreversion; Components: translations
|
Source: src\mo\wxstd-ar.mo; DestDir: {app}\locale\ar; DestName: wxstd.mo; Flags: ignoreversion; Components: translations
|
||||||
|
|
|
@ -88,7 +88,7 @@ void SubtitlesProvider::LoadSubtitles(AssFile *subs, int time) {
|
||||||
push_line(line.GetEntryData());
|
push_line(line.GetEntryData());
|
||||||
|
|
||||||
if (!subs->Attachments.empty()) {
|
if (!subs->Attachments.empty()) {
|
||||||
// TODO: some scripts may have a lot of attachments,
|
// TODO: some scripts may have a lot of attachments,
|
||||||
// so ideally we'd want to write only those actually used on the requested video frame,
|
// so ideally we'd want to write only those actually used on the requested video frame,
|
||||||
// but this would require some pre-parsing of the attached font files with FreeType,
|
// but this would require some pre-parsing of the attached font files with FreeType,
|
||||||
// which isn't probably trivial.
|
// which isn't probably trivial.
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
[wrap-git]
|
[wrap-git]
|
||||||
directory = harfbuzz
|
directory = harfbuzz
|
||||||
url = https://github.com/harfbuzz/harfbuzz
|
url = https://github.com/harfbuzz/harfbuzz
|
||||||
revision = master
|
revision = main
|
||||||
|
|
||||||
[provide]
|
[provide]
|
||||||
harfbuzz = libharfbuzz_dep
|
harfbuzz = libharfbuzz_dep
|
||||||
|
|
|
@ -4,7 +4,7 @@ set -e
|
||||||
|
|
||||||
SRC_DIR="${1}"
|
SRC_DIR="${1}"
|
||||||
BUILD_DIR="${2}"
|
BUILD_DIR="${2}"
|
||||||
WX_PREFIX=`${3} --prefix`
|
WX_PREFIX=""
|
||||||
FONTCONFIG_CONF_DIR="${4}"
|
FONTCONFIG_CONF_DIR="${4}"
|
||||||
DICT_DIR="${5}"
|
DICT_DIR="${5}"
|
||||||
MESON_BUILD_OSX_BUNDLE="${6}"
|
MESON_BUILD_OSX_BUNDLE="${6}"
|
||||||
|
@ -75,26 +75,28 @@ mkdir -vp "${PKG_DIR}/Contents/Resources/en.lproj"
|
||||||
#mv "${PKG_DIR}/Contents/Resources/sr_RS.lproj" "${PKG_DIR}/Contents/Resources/sr_YU.lproj"
|
#mv "${PKG_DIR}/Contents/Resources/sr_RS.lproj" "${PKG_DIR}/Contents/Resources/sr_YU.lproj"
|
||||||
#mv "${PKG_DIR}/Contents/Resources/sr_RS@latin.lproj" "${PKG_DIR}/Contents/Resources/sr_YU@latin.lproj"
|
#mv "${PKG_DIR}/Contents/Resources/sr_RS@latin.lproj" "${PKG_DIR}/Contents/Resources/sr_YU@latin.lproj"
|
||||||
|
|
||||||
echo
|
## TODO: rm those lines
|
||||||
echo "---- Copying WX locale files ----"
|
## xref: [Update and review translations · Issue #132 · TypesettingTools/Aegisub](https://github.com/TypesettingTools/Aegisub/issues/132)
|
||||||
|
# echo
|
||||||
for i in `ls -1 ${SRC_DIR}/po/*.mo|sed "s|po/\(.*\).mo|\1|"`; do
|
# echo "---- Copying WX locale files ----"
|
||||||
WX_MO="${WX_PREFIX}/share/locale/${i}/LC_MESSAGES/wxstd.mo"
|
#
|
||||||
|
# for i in `ls -1 ${SRC_DIR}/po/*.mo|sed "s|po/\(.*\).mo|\1|"`; do
|
||||||
if ! test -f "${WX_MO}"; then
|
# WX_MO="${WX_PREFIX}/share/locale/${i}/LC_MESSAGES/wxstd.mo"
|
||||||
WX_MO="${HOME_DIR}/wxstd/${i}.mo"
|
#
|
||||||
fi
|
# if ! test -f "${WX_MO}"; then
|
||||||
|
# WX_MO="${HOME_DIR}/wxstd/${i}.mo"
|
||||||
if test -f "${WX_MO}"; then
|
# fi
|
||||||
cp -v "${WX_MO}" "${PKG_DIR}/Contents/Resources/${i}.lproj/"
|
#
|
||||||
else
|
# if test -f "${WX_MO}"; then
|
||||||
echo "WARNING: \"$i\" locale in aegisub but no WX catalog found!"
|
# cp -v "${WX_MO}" "${PKG_DIR}/Contents/Resources/${i}.lproj/"
|
||||||
fi
|
# else
|
||||||
done
|
# echo "WARNING: \"$i\" locale in aegisub but no WX catalog found!"
|
||||||
|
# fi
|
||||||
|
# done
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "---- Fixing libraries ----"
|
echo "---- Fixing libraries ----"
|
||||||
sudo python "${SRC_DIR}/tools/osx-fix-libs.py" "${PKG_DIR}/Contents/MacOS/aegisub" || exit $?
|
sudo python3 "${SRC_DIR}/tools/osx-fix-libs.py" "${PKG_DIR}/Contents/MacOS/aegisub" || exit $?
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "Done creating \"${PKG_DIR}\""
|
echo "Done creating \"${PKG_DIR}\""
|
||||||
|
|
|
@ -15,37 +15,46 @@
|
||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
|
||||||
TMP_DMG="temp_dmg"
|
SRC_DIR="${1}"
|
||||||
PKG_DIR="Aegisub.app"
|
BUILD_DIR="${2}"
|
||||||
PKG_NAME="Aegisub-${1}"
|
AEGI_VER="${3}"
|
||||||
PKG_NAME_RW="Aegisub-${1}_rw.dmg"
|
|
||||||
PKG_NAME_VOLUME="Aegisub-${1}"
|
PKG_NAME="Aegisub-${AEGI_VER}"
|
||||||
|
PKG_NAME_VOLUME="${PKG_NAME}"
|
||||||
|
|
||||||
|
PKG_DIR="${BUILD_DIR}/Aegisub.app"
|
||||||
|
DMG_TMP_DIR="${BUILD_DIR}/temp_dmg"
|
||||||
|
DMG_PATH="${BUILD_DIR}/${PKG_NAME}.dmg"
|
||||||
|
DMG_RW_PATH="${BUILD_DIR}/${PKG_NAME}_rw.dmg"
|
||||||
|
|
||||||
|
|
||||||
if ! test -d "${PKG_DIR}"; then
|
if ! test -d "${PKG_DIR}"; then
|
||||||
echo "\"${PKG_DIR}\" does not exist, please run 'make osx-bundle'"
|
echo "\"${PKG_DIR}\" does not exist, please run 'make osx-bundle'"
|
||||||
exit 1;
|
exit 1;
|
||||||
fi
|
fi
|
||||||
|
|
||||||
rm -rf "${TMP_DMG}" "${PKG_NAME}.dmg"
|
|
||||||
mkdir -v "${TMP_DMG}"
|
|
||||||
echo
|
echo
|
||||||
echo "---- Copying ${1} into ${TMP_DMG}/ ----"
|
echo "---- Removing old \"${DMG_TMP_DIR}\", \"${DMG_PATH}\", \"${DMG_RW_PATH}\" ----"
|
||||||
cp -R "${PKG_DIR}" "${TMP_DMG}"
|
rm -rf "${DMG_TMP_DIR}" "${DMG_PATH}" "${DMG_RW_PATH}"
|
||||||
|
mkdir -v "${DMG_TMP_DIR}"
|
||||||
|
echo
|
||||||
|
echo "---- Copying ${AEGI_VER} into ${DMG_TMP_DIR}/ ----"
|
||||||
|
cp -R "${PKG_DIR}" "${DMG_TMP_DIR}"
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "---- Setting up ----"
|
echo "---- Setting up ----"
|
||||||
ln -vsf /Applications "${TMP_DMG}"
|
ln -vsf /Applications "${DMG_TMP_DIR}"
|
||||||
mkdir -v "${TMP_DMG}/.background"
|
mkdir -v "${DMG_TMP_DIR}/.background"
|
||||||
cp -v packages/osx_dmg/dmg_background.png "${TMP_DMG}/.background/background.png"
|
cp -v "${SRC_DIR}/packages/osx_dmg/dmg_background.png" "${DMG_TMP_DIR}/.background/background.png"
|
||||||
cp -v packages/osx_bundle/Contents/Resources/Aegisub.icns "${TMP_DMG}/.VolumeIcon.icns"
|
cp -v "${SRC_DIR}/packages/osx_bundle/Contents/Resources/Aegisub.icns" "${DMG_TMP_DIR}/.VolumeIcon.icns"
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "---- Creating image ----"
|
echo "---- Creating image ----"
|
||||||
/usr/bin/hdiutil create -srcfolder "${TMP_DMG}" -volname "${PKG_NAME}" -fs HFS+ -fsargs "-c c=64,a=16,e=16" -format UDRW "${PKG_NAME_RW}"
|
/usr/bin/hdiutil create -srcfolder "${DMG_TMP_DIR}" -volname "${PKG_NAME}" -fs HFS+ -fsargs "-c c=64,a=16,e=16" -format UDRW "${DMG_RW_PATH}"
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "---- Mounting image ----"
|
echo "---- Mounting image ----"
|
||||||
DEV_NAME=`/usr/bin/hdiutil attach -readwrite -noverify -noautoopen "${PKG_NAME_RW}" |awk '/GUID_partition_scheme/ {print $1}'`
|
DEV_NAME=`/usr/bin/hdiutil attach -readwrite -noverify -noautoopen "${DMG_RW_PATH}" |awk '/GUID_partition_scheme/ {print $1}'`
|
||||||
echo "Device name: ${DEV_NAME}"
|
echo "Device name: ${DEV_NAME}"
|
||||||
|
|
||||||
echo
|
echo
|
||||||
|
@ -61,7 +70,7 @@ if test -n "${SET_STYLE}"; then
|
||||||
echo "---- Running AppleScript to set style ----"
|
echo "---- Running AppleScript to set style ----"
|
||||||
SCRIPT_TMP=`mktemp /tmp/aegisub_dmg_as.XXX`
|
SCRIPT_TMP=`mktemp /tmp/aegisub_dmg_as.XXX`
|
||||||
|
|
||||||
sed -f scripts/osx-bundle.sed packages/osx_dmg/dmg_set_style.applescript > ${SCRIPT_TMP}
|
sed -f "${SRC_DIR}/scripts/osx-bundle.sed" "${SRC_DIR}/packages/osx_dmg/dmg_set_style.applescript" > ${SCRIPT_TMP}
|
||||||
|
|
||||||
/usr/bin/osacompile -o ${SCRIPT_TMP}.scpt ${SCRIPT_TMP}
|
/usr/bin/osacompile -o ${SCRIPT_TMP}.scpt ${SCRIPT_TMP}
|
||||||
|
|
||||||
|
@ -79,18 +88,18 @@ if test -n "${SET_STYLE}"; then
|
||||||
|
|
||||||
hdiutil detach "${DEV_NAME}"
|
hdiutil detach "${DEV_NAME}"
|
||||||
|
|
||||||
DEV_NAME=`/usr/bin/hdiutil attach -readwrite -noverify -noautoopen "${PKG_NAME_RW}" |awk '/GUID_partition_scheme/ {print $1}'`
|
DEV_NAME=`/usr/bin/hdiutil attach -readwrite -noverify -noautoopen "${DMG_RW_PATH}" |awk '/GUID_partition_scheme/ {print $1}'`
|
||||||
echo "Device name: ${DEV_NAME}"
|
echo "Device name: ${DEV_NAME}"
|
||||||
|
|
||||||
cp -v "/Volumes/${PKG_NAME_VOLUME}/.DS_Store" packages/osx_dmg/DS_Store
|
cp -v "/Volumes/${PKG_NAME_VOLUME}/.DS_Store" "${SRC_DIR}/packages/osx_dmg/DS_Store"
|
||||||
SetFile -a v packages/osx_dmg/DS_Store
|
SetFile -a v "${SRC_DIR}/packages/osx_dmg/DS_Store"
|
||||||
hdiutil detach "${DEV_NAME}"
|
hdiutil detach "${DEV_NAME}"
|
||||||
|
|
||||||
rm -rf "${TMP_DMG}" "${PKG_NAME_RW}" ${SCRIPT_TMP}.scpt ${SCRIPT_TMP}
|
rm -rf "${DMG_TMP_DIR}" "${DMG_RW_PATH}" ${SCRIPT_TMP}.scpt ${SCRIPT_TMP}
|
||||||
exit 0
|
exit 0
|
||||||
else
|
else
|
||||||
echo "---- Installing DS_Store ----"
|
echo "---- Installing DS_Store ----"
|
||||||
cp -v packages/osx_dmg/DS_Store "/Volumes/${PKG_NAME_VOLUME}/.DS_Store"
|
cp -v "${SRC_DIR}/packages/osx_dmg/DS_Store" "/Volumes/${PKG_NAME_VOLUME}/.DS_Store"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo
|
echo
|
||||||
|
@ -100,11 +109,10 @@ echo /usr/bin/hdiutil detach "${DEV_NAME}" -force
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "---- Compressing ----"
|
echo "---- Compressing ----"
|
||||||
/usr/bin/hdiutil convert "${PKG_NAME_RW}" -format UDBZ -imagekey bzip2-level=9 -o "${PKG_NAME}.dmg"
|
/usr/bin/hdiutil convert "${DMG_RW_PATH}" -format UDBZ -imagekey bzip2-level=9 -o "${DMG_PATH}"
|
||||||
|
|
||||||
echo
|
echo "---- Removing temp dmg \"${DMG_RW_PATH}\" ----"
|
||||||
echo "---- Removing \"${TMP_DMG}\", \"${PKG_NAME_RW}\" ----"
|
rm -rf "${DMG_RW_PATH}"
|
||||||
rm -rf "${TMP_DMG}" "${PKG_NAME_RW}"
|
|
||||||
|
|
||||||
echo
|
echo
|
||||||
echo "Done!"
|
echo "Done!"
|
||||||
|
|
|
@ -5,99 +5,128 @@ import sys
|
||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
import stat
|
import stat
|
||||||
|
import subprocess
|
||||||
|
|
||||||
is_bad_lib = re.compile(r'(/usr/local|/opt)').match
|
is_bad_lib = re.compile(r'(/usr/local|/opt)').match
|
||||||
is_sys_lib = re.compile(r'(/usr|/System)').match
|
is_sys_lib = re.compile(r'(/usr|/System)').match
|
||||||
otool_libname_extract = re.compile(r'\s+(/.*?)[\(\s:]').search
|
otool_libname_extract = re.compile(r'\s+(/.*?)[(\s:]').search
|
||||||
|
otool_loader_path_extract = re.compile(r'\s+@loader_path/(.*?)[(\s:]').search
|
||||||
goodlist = []
|
goodlist = []
|
||||||
badlist = []
|
badlist = []
|
||||||
link_map = {}
|
link_map = {}
|
||||||
|
|
||||||
|
|
||||||
def otool(cmdline):
|
def otool(cmdline):
|
||||||
pipe = os.popen("otool " + cmdline, 'r')
|
with subprocess.Popen(['otool'] + cmdline, stdout=subprocess.PIPE,
|
||||||
output = pipe.readlines()
|
encoding='utf-8') as p:
|
||||||
pipe.close()
|
return p.stdout.readlines()
|
||||||
return output
|
|
||||||
|
|
||||||
def collectlibs(lib, masterlist, targetdir):
|
def collectlibs(lib, masterlist, targetdir):
|
||||||
global goodlist, link_map
|
global goodlist, link_map
|
||||||
liblist = otool("-L '" + lib + "'")
|
liblist = otool(['-L', lib])
|
||||||
locallist = []
|
locallist = []
|
||||||
|
|
||||||
for l in liblist:
|
for l in liblist:
|
||||||
lr = otool_libname_extract(l)
|
lr = otool_libname_extract(l)
|
||||||
if not lr: continue
|
if lr:
|
||||||
l = lr.group(1)
|
l = lr.group(1)
|
||||||
if is_bad_lib(l) and not l in badlist:
|
else:
|
||||||
badlist.append(l)
|
lr = otool_loader_path_extract(l)
|
||||||
if ((not is_sys_lib(l)) or is_bad_lib(l)) and not l in masterlist:
|
if lr:
|
||||||
locallist.append(l)
|
l = os.path.join(os.path.dirname(lib), lr.group(1))
|
||||||
print("found %s:" % l)
|
else:
|
||||||
|
continue
|
||||||
|
if is_bad_lib(l) and l not in badlist:
|
||||||
|
badlist.append(l)
|
||||||
|
if ((not is_sys_lib(l)) or is_bad_lib(l)) and l not in masterlist:
|
||||||
|
locallist.append(l)
|
||||||
|
print("found %s:" % l)
|
||||||
|
|
||||||
check = l
|
check = l
|
||||||
link_list = []
|
link_list = []
|
||||||
while check:
|
while check:
|
||||||
if os.path.isfile(check) and not os.path.islink(check):
|
basename = os.path.basename(check)
|
||||||
os.system("cp '%s' '%s'" % (check, targetdir))
|
target = os.path.join(targetdir, basename)
|
||||||
print(" FILE %s ... copied to target" % check)
|
|
||||||
if link_list:
|
|
||||||
for link in link_list:
|
|
||||||
link_map[link] = os.path.basename(check)
|
|
||||||
break
|
|
||||||
|
|
||||||
if os.path.islink(check):
|
|
||||||
print(" LINK %s" % check)
|
|
||||||
link_list.append(os.path.basename(check))
|
|
||||||
check = os.path.dirname(check) + "/" + os.readlink(check)
|
|
||||||
|
|
||||||
elif not l in goodlist and not l in masterlist:
|
if os.path.isfile(check) and not os.path.islink(check):
|
||||||
goodlist.append(l)
|
try:
|
||||||
masterlist.extend(locallist)
|
shutil.copy(check, target)
|
||||||
|
except PermissionError:
|
||||||
|
print(" FILE %s ... skipped" % check)
|
||||||
|
break
|
||||||
|
print(" FILE %s ... copied to target" % check)
|
||||||
|
if link_list:
|
||||||
|
for link in link_list:
|
||||||
|
link_map[link] = basename
|
||||||
|
break
|
||||||
|
|
||||||
for l in locallist:
|
if os.path.islink(check):
|
||||||
collectlibs(l, masterlist, targetdir)
|
link_dst = os.readlink(check)
|
||||||
|
try:
|
||||||
|
os.symlink(link_dst, target)
|
||||||
|
except FileExistsError:
|
||||||
|
print(" LINK %s ... existed" % check)
|
||||||
|
break
|
||||||
|
print(" LINK %s ... copied to target" % check)
|
||||||
|
link_list.append(basename)
|
||||||
|
check = os.path.join(os.path.dirname(check), link_dst)
|
||||||
|
|
||||||
exit;
|
elif l not in goodlist and l not in masterlist:
|
||||||
binname = sys.argv[1]
|
goodlist.append(l)
|
||||||
targetdir = os.path.dirname(binname)
|
masterlist.extend(locallist)
|
||||||
print("Searching for libraries in ", binname, "...")
|
|
||||||
libs = [binname]
|
for l in locallist:
|
||||||
collectlibs(sys.argv[1], libs, targetdir)
|
collectlibs(l, masterlist, targetdir)
|
||||||
|
|
||||||
|
|
||||||
print()
|
if __name__ == '__main__':
|
||||||
print("System libraries used...")
|
binname = sys.argv[1]
|
||||||
goodlist.sort()
|
targetdir = os.path.dirname(binname)
|
||||||
for l in goodlist:
|
print("Searching for libraries in", binname, "...")
|
||||||
print(l)
|
libs = [binname]
|
||||||
|
collectlibs(binname, libs, targetdir)
|
||||||
|
|
||||||
|
print()
|
||||||
|
print("System libraries used...")
|
||||||
|
goodlist.sort()
|
||||||
|
for l in goodlist:
|
||||||
|
print(l)
|
||||||
|
|
||||||
print()
|
print()
|
||||||
print("Fixing library install names...")
|
print("Fixing library install names...")
|
||||||
in_tool_cmdline = "install_name_tool "
|
in_tool_cmdline = ['install_name_tool']
|
||||||
for lib in libs:
|
for lib in libs:
|
||||||
libbase = os.path.basename(lib)
|
libbase = os.path.basename(lib)
|
||||||
if libbase in link_map:
|
if libbase in link_map:
|
||||||
libbase = link_map[libbase]
|
libbase = link_map[libbase]
|
||||||
in_tool_cmdline = in_tool_cmdline + ("-change '%s' '@executable_path/%s' " % (lib, libbase))
|
in_tool_cmdline = in_tool_cmdline + ['-change', lib,
|
||||||
for lib in libs:
|
'@executable_path/' + libbase]
|
||||||
libbase = os.path.basename(lib)
|
for lib in libs:
|
||||||
|
libbase = os.path.basename(lib)
|
||||||
|
|
||||||
if libbase in link_map:
|
if libbase in link_map:
|
||||||
libbase = link_map[libbase]
|
libbase = link_map[libbase]
|
||||||
print("%s -> @executable_path/%s (REMAPPED)" % (lib, libbase))
|
print("%s -> @executable_path/%s (REMAPPED)" % (lib, libbase))
|
||||||
else:
|
else:
|
||||||
print("%s -> @executable_path/%s" % (lib, libbase))
|
print("%s -> @executable_path/%s" % (lib, libbase))
|
||||||
|
|
||||||
os.system("%s -id '@executable_path/%s' '%s/%s'" % (in_tool_cmdline, libbase, targetdir, libbase))
|
targetlib = targetdir + '/' + libbase
|
||||||
sys.stdout.flush()
|
orig_permission = os.stat(targetlib).st_mode
|
||||||
|
if not(orig_permission & stat.S_IWUSR):
|
||||||
|
os.chmod(targetlib, orig_permission | stat.S_IWUSR)
|
||||||
|
subprocess.run(in_tool_cmdline + ['-id', '@executable_path/' + libbase,
|
||||||
|
targetlib])
|
||||||
|
if not(orig_permission & stat.S_IWUSR):
|
||||||
|
os.chmod(targetlib, orig_permission)
|
||||||
|
|
||||||
if badlist:
|
if badlist:
|
||||||
print()
|
print()
|
||||||
print("WARNING: The following libraries have blacklisted paths:")
|
print("WARNING: The following libraries have blacklisted paths:")
|
||||||
for lib in sorted(badlist):
|
for lib in sorted(badlist):
|
||||||
print(lib)
|
print(lib)
|
||||||
print("These paths normally have files from a package manager, which means that end result may not work if copied to another machine.")
|
print(
|
||||||
|
"These paths normally have files from a package manager, which means that end result may not work if copied to another machine.")
|
||||||
|
|
||||||
print()
|
print()
|
||||||
print("All done!")
|
print("All done!")
|
||||||
|
|
Loading…
Reference in a new issue