Skip to content
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

Xpack: wix toolset support #4788

Merged
merged 15 commits into from
May 27, 2024
269 changes: 269 additions & 0 deletions xmake/plugins/pack/wix/main.lua
Original file line number Diff line number Diff line change
@@ -0,0 +1,269 @@
--!A cross-platform build utility based on Lua
--
-- Licensed under the Apache License, Version 2.0 (the "License");
-- you may not use this file except in compliance with the License.
-- You may obtain a copy of the License at
--
-- http://www.apache.org/licenses/LICENSE-2.0
--
-- Unless required by applicable law or agreed to in writing, software
-- distributed under the License is distributed on an "AS IS" BASIS,
-- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-- See the License for the specific language governing permissions and
-- limitations under the License.
--
-- Copyright (C) 2015-present, TBOOX Open Source Group.
--
-- @author A2va
-- @file main.lua
--

import("lib.detect.find_tool")
import("private.action.require.impl.packagenv")
import("private.action.require.impl.install_packages")

import(".batchcmds")

-- get the wixtoolset
function _get_wix()

-- enter the environments of wix
local oldenvs = packagenv.enter("wixtoolset")
A2va marked this conversation as resolved.
Show resolved Hide resolved

-- find makensis
local packages = {}
local wix = find_tool("wix", {require_version = ">=4.0.0"})
if not wix then
table.join2(packages, install_packages("wixtoolset"))
end

-- enter the environments of installed packages
for _, instance in ipairs(packages) do
instance:envs_enter()
end

-- we need to force detect and flush detect cache after loading all environments
if not wix then
wix = find_tool("wix", {force = true})
end
assert(wix, "wix not found (ensure that wix is up to date)!")
return wix, oldenvs
end

-- translate the file path
function _translate_filepath(package, filepath)
return path.relative(filepath, package:install_rootdir())
end

-- get a table where the key is a directory and the value a list of files
-- used to regroup all files that are placed in the same directory under the same component.
function _get_cp_kind_table(package, cmds, opt)

local result = {}
for _, cmd in ipairs(cmds) do
if cmd.kind ~= "cp" then
goto continue
end

local option = table.join(cmd.opt or {}, opt)
local srcfiles = os.files(cmd.srcpath)
for _, srcfile in ipairs(srcfiles) do
-- the destination is directory? append the filename
local dstfile = cmd.dstpath
if #srcfiles > 1 or path.islastsep(dstfile) then
if option.rootdir then
dstfile = path.join(dstfile, path.relative(srcfile, option.rootdir))
else
dstfile = path.join(dstfile, path.filename(srcfile))
end
end
srcfile = path.normalize(srcfile)
local dstname = path.filename(dstfile)
local dstdir = path.normalize(path.directory(dstfile))
dstdir = _translate_filepath(package, dstdir)

if result[dstdir] then
table.insert(result[dstdir], {srcfile, dstname})
else
result[dstdir] = {{srcfile, dstname}}
end
end
::continue::
end
return result
end

function _get_other_commands(package, cmd, opt)
opt = table.join(cmd.opt or {}, opt)
local result = ""
local kind = cmd.kind

if kind == "rm" then
local subdirectory = _translate_filepath(package, path.directory(cmd.filepath))
subdirectory = subdirectory ~= "." and string.format([[Subdirectory="%s"]], subdirectory) or ""
local on = opt.install and [[On="install"]] or [[On="uninstall"]]
local filename = path.filename(cmd.filepath)

result = string.format([[<RemoveFile Directory="INSTALLFOLDER" Name="%s" %s %s/>]], filename, subdirectory, on)
elseif kind == "rmdir" then
local dir = _translate_filepath(package, cmd.dir)
local subdirectory = dir ~= "." and string.format([[Subdirectory="%s"]], dir) or ""
local on = opt.install and [[On="install"]] or [[On="uninstall"]]

result = string.format([[<RemoveFolder Directory="INSTALLFOLDER" %s %s/>]], subdirectory, on)
elseif kind == "mkdir" then
local dir = _translate_filepath(package, cmd.dir)
local subdirectory = dir ~= "." and string.format([[Subdirectory="%s"]], dir) or ""
result = string.format([[<CreateFolder Directory="INSTALLFOLDER" %s/>]], subdirectory)
end
return result
end

-- get the string of a wix feature
function _get_feature_string(name, opt)
local level = opt.default and 1 or 2
local description = opt.description or ""
local allow_absent = opt.force and "false" or "true"
local allow_advertise = opt.force and "false" or "true"
local typical_default = [[TypicalDefault="install"]]
local directory = opt.config_dir and [[ConfigurableDirectory="INSTALLFOLDER"]] or ""
local feature = string.format([[<Feature Id="%s" Title="%s" Description="%s" Level="%d" AllowAdvertise="%s" AllowAbsent="%s" %s %s>]], name, name, description, level, allow_advertise, allow_absent, typical_default, directory)
return feature
end

function _get_component_string(id, subdirectory)
local subdirectory = (subdirectory ~= "." and subdirectory ~= nil) and string.format([[Subdirectory="%s"]], subdirectory) or ""
return string.format([[<Component Id="%s" Guid="%s" Directory="INSTALLFOLDER" %s>]], id, hash.uuid(id), subdirectory)
end

-- build a feature from batchcmds
function _build_feature(package, opt)
opt = opt or {}
local default = opt.default or package:get("default")

local result = {}
table.insert(result, _get_feature_string(opt.name or package:title(), table.join(opt, {default = default, description = package:description()})))

local installcmds = batchcmds.get_installcmds(package):cmds()
local uninstallcmds = batchcmds.get_uninstallcmds(package):cmds()

local cp_table = _get_cp_kind_table(package, installcmds, opt)
table.remove_if(installcmds, function (_, cmd) return cmd.kind == "cp" end)

for dir, files in pairs(cp_table) do
local d = path.join(package:install_rootdir(), dir)
table.insert(result, _get_component_string(d:gsub(path.sep(), "_"), dir))
for _, file in ipairs(files) do
local srcfile = file[1]
local dstname = file[2]
table.insert(result, string.format([[<File Source="%s" Name="%s"/>]], srcfile, dstname))
end
table.insert(result, "</Component>")
end

table.insert(result, _get_component_string("OtherCmds"))
for _, cmd in ipairs(installcmds) do
table.insert(result, _get_other_commands(package, cmd, {install = true}))
end
for _, cmd in ipairs(uninstallcmds) do
table.insert(result, _get_other_commands(package, cmd, {install = false}))
end

table.insert(result, "</Component>")
table.insert(result, "</Feature>")
return result
end

-- add to path feature
function _add_to_path(package)
local result = {}
table.insert(result, _get_feature_string("PATH", {default = false, force = false, description = "Add to PATH"}))
table.insert(result, _get_component_string("PATH"))
table.insert(result, [[<Environment Id="PATH" Name="PATH" Value="[INSTALLFOLDER]bin" Permanent="false" Part="last" Action="set" System="true" />]])
table.insert(result, "</Component>")
table.insert(result, "</Feature>")
return result
end

-- get specvars
function _get_specvars(package)

local installcmds = batchcmds.get_installcmds(package):cmds()
local specvars = table.clone(package:specvars())

local features = {}
table.join2(features, _build_feature(package, {default = true, force = true, config_dir = true}))
table.join2(features, _add_to_path(package))

for name, component in table.orderpairs(package:components()) do
table.join2(features, _build_feature(component, {name = "Install " .. name}))
end

specvars.PACKAGE_WIX_CMDS = table.concat(features, "\n ")
specvars.PACKAGE_WIX_UPGRADECODE = hash.uuid(package:name())

-- company cannot be empty with wix
if package:get("company") == nil or package:get("company") == "" then
specvars.PACKAGE_COMPANY = package:name()
end
return specvars
end

function _pack_wix(wix, package)

-- install the initial specfile
local specfile = package:specfile()
if not os.isfile(specfile) then
local specfile_template = path.join(os.programdir(), "scripts", "xpack", "wix", "msi.wxs")
os.cp(specfile_template, specfile)
end

-- replace variables in specfile
local specvars = _get_specvars(package)
local pattern = package:extraconf("specfile", "pattern") or "%${([^\n]-)}"
io.gsub(specfile, "(" .. pattern .. ")", function(_, name)
name = name:trim()
local value = specvars[name]
if type(value) == "function" then
value = value()
end
if value ~= nil then
dprint(" > replace %s -> %s", name, value)
end
if type(value) == "table" then
dprint("invalid variable value", value)
end
return value
end)

local argv = {"build", specfile}
table.join2(argv, {"-ext", "WixToolset.UI.wixext"})
table.join2(argv, {"-o", package:outputfile()})

if package:arch() == "x64" then
table.join2(argv, {"-arch", "x64"})
elseif package:arch() == "x86" then
table.join2(argv, {"-arch", "x86"})
end

-- make package
os.vrunv(wix, argv)
end

function main(package)
-- only for windows
if not is_host("windows") then
return
end

cprint("packing %s", package:outputfile())

-- get wix
local wix, oldenvs = _get_wix()

-- pack nsis package
_pack_wix(wix.program, package)

-- done
os.setenvs(oldenvs)
end
3 changes: 3 additions & 0 deletions xmake/plugins/pack/xpack.lua
Original file line number Diff line number Diff line change
Expand Up @@ -230,6 +230,7 @@ function xpack:inputkind()
local inputkind = self:get("inputkind")
if inputkind == nil then
local inputkinds = {
wix = "binary",
nsis = "binary",
zip = "binary",
targz = "binary",
Expand Down Expand Up @@ -361,6 +362,7 @@ end
-- get the specfile path
function xpack:specfile()
local extensions = {
wix = ".wxs",
nsis = ".nsi",
srpm = ".spec",
rpm = ".spec",
Expand All @@ -375,6 +377,7 @@ function xpack:extension()
local extension = self:get("extension")
if extension == nil then
local extensions = {
wix = ".msi",
nsis = ".exe",
zip = ".zip",
targz = ".tar.gz",
Expand Down
34 changes: 34 additions & 0 deletions xmake/scripts/xpack/wix/msi.wxs
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
<!-- this is a WIX spec file -->
<!-- it is autogenerated by the xmake build system. -->
<!-- do not edit by hand. -->
<?define PackageName = "${PACKAGE_NAME}"?>
<?define PackageTitle = "${PACKAGE_TITLE}"?>
<?define PackageDescription = "${PACKAGE_DESCRIPTION}"?>
<?define PackageAuthor = "${PACKAGE_AUTHOR}"?>
<?define PackageMaintainer = "${PACKAGE_MAINTAINER}"?>
<?define PackageHomepage = "${PACKAGE_HOMEPAGE}"?>
<?define PackageCoypright = "${PACKAGE_COPYRIGHT}"?>
<?define PackageCompany = "${PACKAGE_COMPANY}"?>
<?define PackageIconFile = "${PACKAGE_ICONFILE}"?>
<?define PackageLicense = "${PACKAGE_LICENSE}"?>
<?define PackageLicenseFile = "${PACKAGE_LICENSEFILE}"?>
<?define PackageVersion = "${PACKAGE_VERSION}"?>

<?define UpgradeCode = "${PACKAGE_WIX_UPGRADECODE}"?>

<Wix xmlns="http://wixtoolset.org/schemas/v4/wxs" xmlns:ui="http://wixtoolset.org/schemas/v4/wxs/ui">
<Package Language="1033" Manufacturer="$(PackageCompany)" Name="$(PackageName)" UpgradeCode="$(UpgradeCode)" Version="$(PackageVersion)">
<MajorUpgrade DowngradeErrorMessage="A later version of $(PackageName) is already installed. Setup will now exit." />
<?if $(PackageIconFile) != "" ?>
<Icon Id="Icon.ico" SourceFile="$(PackageIconFile)"/>
<?endif ?>
<MediaTemplate EmbedCab="true" />
<ui:WixUI Id="WixUI_FeatureTree" InstallDirectory="INSTALLFOLDER" />
<StandardDirectory Id="ProgramFiles6432Folder">
<Directory Id="CompanyFolder" Name="$(PackageCompany)">
<Directory Id="INSTALLFOLDER" Name="$(PackageName)"/>
</Directory>
</StandardDirectory>
${PACKAGE_WIX_CMDS}
</Package>
</Wix>
Loading