Wine package guidelines (Русский)
CLR – Cross – Eclipse – Free Pascal – GNOME – Go – Haskell – Java – KDE – Kernel – Lisp – MinGW – Nonfree – OCaml – Perl – PHP – Python – Ruby – VCS – Web – Wine
Многие программы для Windows всё еще могут быть нужны в Linux и нам хочется, чтобы для них был создан пакет. Из-за разницы между данными операционными системами этот процесс иногда может быть запутанным. В этом руководстве мы поговорим о Win32 бинарных файлах, поскольку проекты с открытым исходным кодом обычно итак уже портированы на Linux.
Contents
Что нужно обязательно проверить
- Лицензия: позволяет ли лицензия перепаковывать программу?
- Установщик: есть ли возможность установить программу по тихому? А ещё лучше, существует ли версия без инсталлятора?
- Портативность и чистота: портативна ли программа? Чистая ли она?
Здесь под портативной программой подразумевается такая программа, которая никогда не изменяет реестр и не пишет что-либо за пределами её директории; под чистой программой подразумевается такая, которая никогда не записывает что-либо даже в свою директорию, но может записать свои настройки в папку пользователя. Программа может являться одновременно и чистой и портативной (например, она никогда не записывает настройки) или быть ни чистой, ни портативной (например, она записывает в свою директорию, а также записывает не в свою директорию, и изменяет реестр...)
Лицензия
Обычно лицензии находятся в текстовом файле в директории установки. Если вы не можете её найти, попробуйте пройтись по этапам установки. Если про перепаковку ничего не сказано, продолжаем. Значит автор не возражает. В противном случае лицензии обычно не разрешают удалять файлы или вообще не позволяют перепаковывать программу. В первом случае просто будьте аккуратны, чтобы процесс makepkg не потерял какой-либо файл, а удалить ненужные файлы (например, деинсталляторы) вы сможете на фазе post_install
; во втором случае весь процесс установки придётся делать в фазе post_install
. Фаза build
будет использоваться только для копирования устанавливаемых файлов.
Установщик
Гораздо проще работать со сжатыми файлами, такими как .zip
, чем с инсталляторами для Windows. Если у вас нет выбора, например автор настаивает на том, чтобы распространять свою программу только в качестве инсталлятора, поищите в Интернете возможно ли установить эту программу по тихому. Обычно MSFN является хорошим местом для поиска. Если вы не сможете найти способа как это сделать, попробуйте открыть инсталлятор с помощью различных распаковывающих утилит; это может сработать.
Портативность и чистота
Для портативной программы не нужна своя файловая система, эмулируемая Wine, поэтому проверьте на Portable Freeware есть ли портативная версия программы.
The guideline in short
The idea behind packaging a Windows program is to use the program's files as mere data that Wine will interpret, just like JVM and Java bytecode.
So we will install the program in /usr/share/"$pkgname"
and the program will write all what it needs in "$HOME"/."$pkgname"
. Everything will be prepared by a small script saved in /usr/bin/"$pkgname"
that will create the folder, prepare it if needed, and finally start the program.
In the next sections we will talk about every step.
This way every user will have their own settings and their decisions will not bother other users.
Installing
If the program has no installer, the installation is a mere decompression of a file; unpack it to "$pkgdir"/usr/share/$pkgname
, making sure that the permissions are correct. These commands will do:
$ find "$pkgdir"/usr/share -type f -exec chmod 644 "{}" \; $ find "$pkgdir"/usr/share -type d -exec chmod 755 "{}" \;
If the program cannot be installed the easy way, you need to create a Wine environment:
$ install -m755 -d "$srcdir"/tmp "$srcdir"/tmp/env "$srcdir"/tmp/local $ export WINEPREFIX="$srcdir"/tmp/env $ export XDG_DATA_HOME="$srcdir"/tmp/local $ wine "$srcdir"/installer.exe /silentoptions
We have not discussed portability yet, but if your program does not need the registry keys it modified, you can just copy the directory from the:
"$srcdir"/tmp/env/drive_c/Program\ Files/programname
Otherwise you need to copy all the registry files too and eventually the files the program installed around. The "$srcdir"/tmp/local
will contains menu icons and desktop files, you may want to copy them in the package. If there does not exist a way to install the program silently... Maybe you can make a .tar.gz
file and upload it somewhere? If nothing automated is possible, force the user to follow the installer and hope he does not mess up the installation, write some checks before blindly copying a folder that may not exist (e.g., the user pressed 'Cancel'.)
The /usr/bin script
This script prepares the settings folder and starts the program. If your program is portable, it will look like this:
#!/bin/bash unset WINEPREFIX if [ ! -d "$HOME"/.programname ] ; then mkdir -p "$HOME"/.programname #prepare the environment here fi WINEDEBUG=-all wine "$HOME"/.programname/programname "$@"
If it is clean, it will look like this:
#!/bin/bash export WINEPREFIX="$HOME"/.programname/wine if [ ! -d "$HOME"/.programname ] ; then mkdir -p "$HOME"/.programname/wine wineboot -u #copy the registry file if needed fi WINEDEBUG=-all wine /usr/share/programname "$@"
As you can see, in the second case there is no environment preparation. In fact a clean application will be started directly from /usr/share
since it will not need to write in its folder, so its settings will be written somewhere in the emulated file system.
If the application is neither clean neither portable the two ideas must be combined.
If the application does not write settings at all, skip the if
and start it from /usr/share
.
The task of preparing the environment may differ greatly between applications, but follow these rules of thumb: If the program:
- just needs to read a file, symlink it.
- needs to write in a file, copy it.
- does not use a file, ignore it.
Of course the minimum is just starting WINEDEBUG=-all wine /usr/share/programname "$@"
.
Usually the environment will be made by symlinking between the "$HOME"/.programname
directory and the /usr/share/programname
files. But since some Windows programs are very fickle about their paths, you may need to symlink directly in the "$HOME"/.programname/wine/drive_c/Program\ Files/programname
directory.
Of course those are just ideas to integrate Win32 applications in the Linux environment, do not forget your intelligence and gumption.
As example, μTorrent is by default a clean application, but with a easy step can be used as a portable one. Since it is a single file and it is pretty small creating its wine environment (about 5MB) it is probably an overkill. It is better to symlink the executable, create the empty settings.dat in order to use it portable in the $HOME/.utorrent
directory. With the added advantage that just visiting .utorrent
folder an user can see a copy of the .torrent
files she downloaded.
UnionFsFuse
You can consider using the UnionFsFuse program available in the official repositories as unionfs-fuse).
UnionFsFuse allows to keep the base directory in /usr/share
and put a copy of the files the application needed to write inside $HOME/.programname
almost automatically.
Using UnionFsFuse means an additional dependency and it requires the fuse module that not all users might load. Yet, it might be worthwhile if the application would need lots of symlinking or if it is unclear exactly what it needs to be written. Just ensure to mount and unmount the UnionFs correctly.
Among others, the GeneRallyAUR[ссылка недействительна: package not found] package uses this approach.
One example
We will make a package for eMule. According to Portable Freeware, eMule is not completely portable since it writes some (useless) keys in the registry.
On the other hand, it is not clean either since it writes its configuration files and puts its downloads in its installation folder.
Luckily there is an installer-less version available.
So we make our PKGBUILD; the only dependency is wine. The md5sums
should be added.
# Maintainer: You <youremail> pkgname=emule pkgver=0.49b pkgrel=1 pkgdesc="One of the biggest and most reliable peer-to-peer file sharing clients around the world." arch=(i686 x86_64) url="http://www.emule-project.net" license=('GPL') depends=() depends=(wine) makedepends=(unzip) source=(emule http://prdownloads.sourceforge.net/emule/eMule$pkgver.zip) noextract=() options=(!strip) build() { rm -f src/eMule"$pkgver"/license* #It is GPL install -d -m755 pkg/usr/share/emule cp -ra src/eMule"$pkgver"/* pkg/usr/share/emule find pkg/usr/share/emule -type d -exec chmod 755 "{}" \; find pkg/usr/share/emule -type f -exec chmod 644 "{}" \; install -d -m755 pkg/usr/bin install -m755 emule pkg/usr/bin }
Now we make our emule file, which according to build
, will be copied and made executable in /usr/bin
.
#!/bin/bash export WINEARCH=win32 WINEPREFIX="$HOME/.emule/wine" if [ ! -d "$HOME"/.emule ] ; then mkdir -p "$HOME"/.emule/wine || exit 1 #Each user will have its config, we copy the default file since emule #needs to write here. cp -r /usr/share/emule/config "$HOME"/.emule || exit 1 #We symlink the files emule needs to read to work ln -s /usr/share/emule/emule.exe "$HOME"/.emule/emule || exit 1 ln -s -T /usr/share/emule/lang "$HOME"/.emule/lang || exit 1 ln -s -T /usr/share/emule/webserver "$HOME"/.emule/webserver || exit 1 fi wine "$HOME"/.emule/emule "$@"
If you want to be more precise, you may add a message in the .install
file telling the user that they should disable search history since wine messes up that menu. You may even provide a default config file with the best settings. And that's it... run $ makepkg
, check the package folder to be sure, and install.
Gecko
Unless you know for sure, that software require browser of .NET runtime (packages wine_gecko[ссылка недействительна: replaced by wine-gecko] and wine-mono in official repositories), default wine installation prompts for Gecko/Mono are undesirable.
To disable HTML rendering, bytecode support and the dialogs, you need to use a dlloverride in your script. For Gecko:
export WINEDLLOVERRIDES="mshtml="
For Mono:
export WINEDLLOVERRIDES="mscoree="
For both:
export WINEDLLOVERRIDES="mscoree,mshtml="
You can also disable them via winecfg
: just set mscoree/mshtml to Disable.