summaryrefslogtreecommitdiff
path: root/love2dToAPK/tools/tools/zbstudio-win/lualibs/dist/depends.lua
diff options
context:
space:
mode:
Diffstat (limited to 'love2dToAPK/tools/tools/zbstudio-win/lualibs/dist/depends.lua')
-rw-r--r--love2dToAPK/tools/tools/zbstudio-win/lualibs/dist/depends.lua770
1 files changed, 770 insertions, 0 deletions
diff --git a/love2dToAPK/tools/tools/zbstudio-win/lualibs/dist/depends.lua b/love2dToAPK/tools/tools/zbstudio-win/lualibs/dist/depends.lua
new file mode 100644
index 0000000..2a6b7f0
--- /dev/null
+++ b/love2dToAPK/tools/tools/zbstudio-win/lualibs/dist/depends.lua
@@ -0,0 +1,770 @@
+-- Utility functions for dependencies
+
+module ("dist.depends", package.seeall)
+
+local cfg = require "dist.config"
+local mf = require "dist.manifest"
+local sys = require "dist.sys"
+local const = require "dist.constraints"
+local utils = require "dist.utils"
+local package = require "dist.package"
+
+-- Return all packages with specified names from manifest.
+-- Names can also contain version constraint (e.g. 'copas>=1.2.3', 'saci-1.0' etc.).
+function find_packages(package_names, manifest)
+ if type(package_names) == "string" then package_names = {package_names} end
+ manifest = manifest or mf.get_manifest()
+ assert(type(package_names) == "table", "depends.find_packages: Argument 'package_names' is not a table or string.")
+ assert(type(manifest) == "table", "depends.find_packages: Argument 'manifest' is not a table.")
+
+ local packages_found = {}
+ -- find matching packages in manifest
+ for _, pkg_to_find in pairs(package_names) do
+ local pkg_name, pkg_constraint = split_name_constraint(pkg_to_find)
+ pkg_name = utils.escape_magic(pkg_name):gsub("%%%*",".*")
+ for _, repo_pkg in pairs(manifest) do
+ if string.match(repo_pkg.name, "^" .. pkg_name .. "$") and (not pkg_constraint or satisfies_constraint(repo_pkg.version, pkg_constraint)) then
+ table.insert(packages_found, repo_pkg)
+ end
+ end
+ end
+ return packages_found
+end
+
+-- Return manifest consisting of packages installed in specified deploy_dir directory
+function get_installed(deploy_dir)
+ deploy_dir = deploy_dir or cfg.root_dir
+ assert(type(deploy_dir) == "string", "depends.get_installed: Argument 'deploy_dir' is not a string.")
+ deploy_dir = sys.abs_path(deploy_dir)
+
+ local distinfos_path = sys.make_path(deploy_dir, cfg.distinfos_dir)
+ local manifest = {}
+
+ if not sys.is_dir(distinfos_path) then return {} end
+
+ -- from all directories of packages installed in deploy_dir
+ for dir in sys.get_directory(distinfos_path) do
+
+ if dir ~= "." and dir ~= ".." and sys.is_dir(sys.make_path(distinfos_path, dir)) then
+ local pkg_dist_dir = sys.make_path(distinfos_path, dir)
+
+ -- load the dist.info file
+ for file in sys.get_directory(pkg_dist_dir) do
+ local pkg_dist_file = sys.make_path(pkg_dist_dir, file)
+
+ if sys.is_file(pkg_dist_file) then
+ table.insert(manifest, mf.load_distinfo(pkg_dist_file))
+ end
+ end
+
+ end
+
+ end
+ return manifest
+end
+
+-- If 'pkg.selected' == true then returns 'selected' else 'installed'.
+-- Used in error messages.
+local function selected_or_installed(pkg)
+ assert(type(pkg) == "table", "depends.selected_or_installed: Argument 'pkg' is not a table.")
+ if pkg.selected == true then
+ return "selected"
+ else
+ return "installed"
+ end
+end
+
+-- Return whether the 'package_name' is installed according to the the manifest 'installed_pkgs'
+-- If optional 'version_wanted' constraint is specified, then installed packages must
+-- also satisfy specified version constraint.
+-- If package is installed but doesn't satisfy version constraint, error message
+-- is returned as the second value.
+function is_installed(package_name, installed_pkgs, version_wanted)
+ assert(type(package_name) == "string", "depends.is_installed: Argument 'package_name' is not a string.")
+ assert(type(installed_pkgs) == "table", "depends.is_installed: Argument 'installed_pkgs' is not a table.")
+ assert(type(version_wanted) == "string" or type(version_wanted) == "nil", "depends.is_installed: Argument 'version_wanted' is not a string or nil.")
+
+ local pkg_is_installed, err = false, nil
+
+ for _, installed_pkg in pairs(installed_pkgs) do
+
+ -- check if package_name is in installed
+ if package_name == installed_pkg.name then
+
+ -- check if package is installed in satisfying version
+ if not version_wanted or satisfies_constraint(installed_pkg.version, version_wanted) then
+ pkg_is_installed = true
+ break
+ else
+ err = "Package '" .. package_name .. (version_wanted and " " .. version_wanted or "") .. "' needed, but " .. selected_or_installed(installed_pkg) .. " at version '" .. installed_pkg.version .. "'."
+ break
+ end
+ end
+
+ end
+ return pkg_is_installed, err
+end
+
+-- Check whether the package 'pkg' conflicts with 'installed_pkg' and return
+-- false or error message.
+local function packages_conflicts(pkg, installed_pkg)
+ assert(type(pkg) == "table", "depends.packages_conflicts: Argument 'pkg' is not a table.")
+ assert(type(installed_pkg) == "table", "depends.packages_conflicts: Argument 'installed_pkg' is not a table.")
+
+ -- check if pkg doesn't provide an already installed_pkg
+ if pkg.provides then
+ -- for all of pkg's provides
+ for _, provided_pkg in pairs(get_provides(pkg)) do
+ if provided_pkg.name == installed_pkg.name then
+ return "Package '" .. pkg_full_name(pkg.name, pkg.version, pkg.was_scm_version) .. "' provides '" .. pkg_full_name(provided_pkg.name, provided_pkg.version) .. "' but package '" .. pkg_full_name(installed_pkg.name, installed_pkg.version) .. "' is already " .. selected_or_installed(installed_pkg) .. "."
+ end
+ end
+ end
+
+ -- check for conflicts of package to install with installed package
+ if pkg.conflicts then
+ for _, conflict in pairs (pkg.conflicts) do
+ if conflict == installed_pkg.name then
+ return "Package '" .. pkg_full_name(pkg.name, pkg.version, pkg.was_scm_version) .. "' conflicts with already " .. selected_or_installed(installed_pkg) .. " package '" .. pkg_full_name(installed_pkg.name, installed_pkg.version) .. "'."
+ end
+ end
+ end
+
+ -- check for conflicts of installed package with package to install
+ if installed_pkg.conflicts then
+
+ -- direct conflicts with 'pkg'
+ for _, conflict in pairs (installed_pkg.conflicts) do
+ if conflict == pkg.name then
+ return "Already " .. selected_or_installed(installed_pkg) .. " package '" .. pkg_full_name(installed_pkg.name, installed_pkg.version) .. "' conflicts with package '" .. pkg_full_name(pkg.name, pkg.version, pkg.was_scm_version) .. "'."
+ end
+ end
+
+ -- conflicts with 'provides' of 'pkg' (packages provided by package to install)
+ if pkg.provides then
+ for _, conflict in pairs (installed_pkg.conflicts) do
+ -- for all of pkg's provides
+ for _, provided_pkg in pairs(get_provides(pkg)) do
+ if conflict == provided_pkg.name then
+ return "Already '" .. selected_or_installed(installed_pkg) .. " package '" .. pkg_full_name(installed_pkg.name, installed_pkg.version) .. "' conflicts with package '" .. pkg_full_name(provided_pkg.name, provided_pkg.version) .. "' provided by '" .. pkg_full_name(pkg.name, pkg.version, pkg.was_scm_version) .. "'."
+ end
+ end
+ end
+ end
+ end
+
+ -- no conflicts found
+ return false
+end
+
+-- Return table of package dependencies 'depends' with OS specific dependencies extracted.
+--
+-- OS specific dependencies are stored in a subtable with 'arch' as a key.
+-- E.g. this table containing OS specific dependencies:
+-- depends = {
+-- "lua~>5.1",
+-- "luadist-git>=0.1",
+-- Linux = {
+-- "iup>=3.6",
+-- "wxlua>=2.8.10.0",
+-- },
+-- Windows = {
+-- "luagd>=2.0.33r2",
+-- "luacom>=1.4.1",
+-- },
+-- }
+--
+-- ...will be on the 'Linux' architecture (determined by cfg.arch) converted into:
+-- depends = {
+-- "lua~>5.1",
+-- "luadist-git>=0.1",
+-- "iup>=3.6",
+-- "wxlua>=2.8.10.0",
+-- }
+function extract_os_specific_depends(depends)
+ assert(type(depends) == "table", "depends.extract_os_specific_depends: Argument 'depends' is not a table.")
+ local extracted = {}
+ for k, depend in pairs(depends) do
+ -- if 'depend' is a table, then it must be a table of OS specific
+ -- dependencies, so extract it if it's for this architecture
+ if type(depend) == "table" then
+ if k == cfg.arch then
+ for _, os_specific_depend in pairs(depend) do
+ table.insert(extracted, os_specific_depend)
+ end
+ end
+ else
+ table.insert(extracted, depend)
+ end
+ end
+ return extracted
+end
+
+-- Return all packages needed in order to install package 'pkg'
+-- and with specified 'installed' packages in the system using 'manifest'.
+-- 'pkg' can also contain version constraint (e.g. 'copas>=1.2.3', 'saci-1.0' etc.).
+--
+-- This function also downloads packages to get information about their dependencies.
+-- Directory where the package was downloaded is stored in 'download_dir' attribute
+-- of that package in the table of packages returned by this function.
+--
+-- Optional argument 'dependency_manifest' is a table of dependencies examined
+-- from previous installations etc. It can be used to speed-up the dependency
+-- resolving procedure for example.
+--
+-- When optional 'force_no_download' parameter is set to true, then information
+-- about packages won't be downloaded during dependency resolving, assuming that
+-- entries in the provided manifest are already complete.
+--
+-- When optional 'suppress_printing' parameter is set to true, then messages
+-- for the user won't be printed during dependency resolving.
+--
+-- Optional argument 'deploy_dir' is used just as a temporary place to place
+-- the downloaded packages into.
+--
+-- 'dependency_parents' is table of all packages encountered so far when resolving dependencies
+-- and is used to detect and deal with circular dependencies. Leave it 'nil'
+-- and it will do its job just fine :-).
+--
+-- 'tmp_installed' is internal table used in recursion and should be left 'nil' when
+-- calling this function from other context. It is used for passing the changes
+-- in installed packages between the recursive calls of this function.
+--
+-- TODO: refactor this spaghetti code!
+local function get_packages_to_install(pkg, installed, manifest, dependency_manifest, force_no_download, suppress_printing, deploy_dir, dependency_parents, tmp_installed)
+ manifest = manifest or mf.get_manifest()
+ dependency_manifest = dependency_manifest or {}
+ force_no_download = force_no_download or false
+ suppress_printing = suppress_printing or false
+ deploy_dir = deploy_dir or cfg.root_dir
+ dependency_parents = dependency_parents or {}
+
+ -- set helper table 'tmp_installed'
+ tmp_installed = tmp_installed or utils.deepcopy(installed)
+
+ assert(type(pkg) == "string", "depends.get_packages_to_install: Argument 'pkg' is not a string.")
+ assert(type(installed) == "table", "depends.get_packages_to_install: Argument 'installed' is not a table.")
+ assert(type(manifest) == "table", "depends.get_packages_to_install: Argument 'manifest' is not a table.")
+ assert(type(dependency_manifest) == "table", "depends.get_packages_to_install: Argument 'dependency_manifest' is not a table.")
+ assert(type(force_no_download) == "boolean", "depends.get_packages_to_install: Argument 'force_no_download' is not a boolean.")
+ assert(type(suppress_printing) == "boolean", "depends.get_packages_to_install: Argument 'suppress_printing' is not a boolean.")
+ assert(type(deploy_dir) == "string", "depends.get_packages_to_install: Argument 'deploy_dir' is not a string.")
+ assert(type(dependency_parents) == "table", "depends.get_packages_to_install: Argument 'dependency_parents' is not a table.")
+ assert(type(tmp_installed) == "table", "depends.get_packages_to_install: Argument 'tmp_installed' is not a table.")
+ deploy_dir = sys.abs_path(deploy_dir)
+
+ --[[ for future debugging:
+ print('resolving: '.. pkg)
+ print(' installed: ', utils.table_tostring(installed))
+ print(' tmp_installed: ', utils.table_tostring(tmp_installed))
+ --]]
+
+ -- check if package is already installed
+ local pkg_name, pkg_constraint = split_name_constraint(pkg)
+ local pkg_is_installed, err = is_installed(pkg_name, tmp_installed, pkg_constraint)
+ if pkg_is_installed then return {} end
+ if err then return nil, err end
+
+ -- table of packages needed to be installed (will be returned)
+ local to_install = {}
+
+ -- find out available versions of 'pkg' and insert them into manifest
+ if not force_no_download then
+ local versions, err = package.retrieve_versions(pkg, manifest, suppress_printing)
+ if not versions then return nil, err end
+ for _, version in pairs(versions) do
+ table.insert(manifest, version)
+ end
+ end
+
+ -- find candidates & sort them
+ local candidates_to_install = find_packages(pkg, manifest)
+ if #candidates_to_install == 0 then
+ return nil, "No suitable candidate for '" .. pkg .. "' found."
+ end
+ candidates_to_install = sort_by_versions(candidates_to_install)
+
+ for _, pkg in pairs(candidates_to_install) do
+
+ --[[ for future debugging:
+ print(' candidate: '.. pkg.name..'-'..pkg.version)
+ print(' installed: ', utils.table_tostring(installed))
+ print(' tmp_installed: ', utils.table_tostring(tmp_installed))
+ print(' to_install: ', utils.table_tostring(to_install))
+ print(' -is installed: ', is_installed(pkg.name, tmp_installed, pkg_constraint))
+ --]]
+
+ -- if there's an error from the previous candidate, print the reason for trying another one
+ if not suppress_printing and err then print(" - trying another candidate due to: " .. err) end
+
+ -- clear the state from the previous candidate
+ pkg_is_installed, err = false, nil
+
+ -- check whether this package has already been added to 'tmp_installed' by another of its candidates
+ pkg_is_installed, err = is_installed(pkg.name, tmp_installed, pkg_constraint)
+ if pkg_is_installed then break end
+
+ -- preserve information about the 'scm' version, because pkg.version
+ -- will be rewritten by information taken from pkg's dist.info file
+ local was_scm_version = (pkg.version == "scm")
+
+ -- Try to obtain cached dependency information from the dependency manifest
+ if dependency_manifest[pkg.name .. "-" .. pkg.version] and cfg.dep_cache then
+ pkg = dependency_manifest[pkg.name .. "-" .. pkg.version]
+ else
+ -- download info about the package if not already downloaded and downloading not prohibited
+ if not (pkg.download_dir or force_no_download) then
+ local path_or_err
+ pkg, path_or_err = package.retrieve_pkg_info(pkg, deploy_dir, suppress_printing)
+ if not pkg then
+ err = "Error when resolving dependencies: " .. path_or_err
+ else
+ -- set path to downloaded package - used to indicate that the
+ -- package was already downloaded, to delete unused but downloaded
+ -- packages and also to install choosen packages
+ pkg.download_dir = path_or_err
+ end
+ end
+ end
+
+ if pkg and was_scm_version then pkg.was_scm_version = true end
+
+ -- check arch & type
+ if not err then
+ if not (pkg.arch == "Universal" or pkg.arch == cfg.arch) or
+ not (pkg.type == "all" or pkg.type == "source" or pkg.type == cfg.type) then
+ err = "Package '" .. pkg_full_name(pkg.name, pkg.version) .. "' doesn't have required arch and type."
+ end
+ end
+
+ -- checks for conflicts with other installed (or previously selected) packages
+ if not err then
+ for _, installed_pkg in pairs(tmp_installed) do
+ err = packages_conflicts(pkg, installed_pkg)
+ if err then break end
+ end
+ end
+
+ -- if pkg passed all of the above tests
+ if not err then
+
+ -- check if pkg's dependencies are satisfied
+ if pkg.depends then
+
+ -- insert pkg into the stack of circular dependencies detection
+ table.insert(dependency_parents, pkg.name)
+
+ -- extract all OS specific dependencies of pkg
+ pkg.depends = extract_os_specific_depends(pkg.depends)
+
+ -- for all dependencies of pkg
+ for _, depend in pairs(pkg.depends) do
+ local dep_name = split_name_constraint(depend)
+
+ -- detect circular dependencies using 'dependency_parents'
+ local is_circular_dependency = false
+ for _, parent in pairs(dependency_parents) do
+ if dep_name == parent then
+ is_circular_dependency = true
+ break
+ end
+ end
+
+ -- if circular dependencies not detected
+ if not is_circular_dependency then
+
+ -- recursively call this function on the candidates of this pkg's dependency
+ local depends_to_install, dep_err = get_packages_to_install(depend, installed, manifest, dependency_manifest, force_no_download, suppress_printing, deploy_dir, dependency_parents, tmp_installed)
+
+ -- if any suitable dependency packages were found, insert them to the 'to_install' table
+ if depends_to_install then
+ for _, depend_to_install in pairs(depends_to_install) do
+
+ -- add some meta information
+ if not depend_to_install.selected_by then
+ depend_to_install.selected_by = pkg.name .. "-" .. pkg.version
+ end
+
+ table.insert(to_install, depend_to_install)
+ table.insert(tmp_installed, depend_to_install)
+ table.insert(installed, depend_to_install)
+ end
+ else
+ err = "Error getting dependency of '" .. pkg_full_name(pkg.name, pkg.version) .. "': " .. dep_err
+ break
+ end
+
+ -- if circular dependencies detected
+ else
+ err = "Error getting dependency of '" .. pkg_full_name(pkg.name, pkg.version) .. "': '" .. dep_name .. "' is a circular dependency."
+ break
+ end
+ end
+
+ -- remove last package from the stack of circular dependencies detection
+ table.remove(dependency_parents)
+ end
+
+ -- if no error occured
+ if not err then
+ -- add pkg and it's provides to the fake table of installed packages, with
+ -- property 'selected' set, indicating that the package isn't
+ -- really installed in the system, just selected to be installed (this is used e.g. in error messages)
+ pkg.selected = true
+ table.insert(tmp_installed, pkg)
+ if pkg.provides then
+ for _, provided_pkg in pairs(get_provides(pkg)) do
+ provided_pkg.selected = true
+ table.insert(tmp_installed, provided_pkg)
+ end
+ end
+ -- add pkg to the table of packages to install
+ table.insert(to_install, pkg)
+
+ -- if some error occured
+ else
+ -- delete the downloaded package
+ if pkg.download_dir and not cfg.debug then sys.delete(pkg.download_dir) end
+
+ -- set tables of 'packages to install' and 'installed packages' to their original state
+
+ to_install = {}
+ tmp_installed = utils.deepcopy(installed)
+ -- add provided packages to installed ones
+ for _, installed_pkg in pairs(tmp_installed) do
+ for _, pkg in pairs(get_provides(installed_pkg)) do
+ table.insert(tmp_installed, pkg)
+ end
+ end
+ end
+
+ -- if error occured
+ else
+ -- delete the downloaded package
+ if pkg and pkg.download_dir and not cfg.debug then sys.delete(pkg.download_dir) end
+
+ -- if pkg is already installed, skip checking its other candidates
+ if pkg_is_installed then break end
+ end
+ end
+
+ -- if package is not installed and no suitable candidates were found, return the last error
+ if #to_install == 0 and not pkg_is_installed then
+ return nil, err
+ else
+ return to_install
+ end
+end
+
+-- Resolve dependencies and return all packages needed in order to install
+-- 'packages' into the system with already 'installed' packages, using 'manifest'.
+-- Also return the table of the dependencies determined during the process
+-- as the second return value.
+--
+-- Optional argument 'dependency_manifest' is a table of dependencies examined
+-- from previous installations etc. It can be used to speed-up the dependency
+-- resolving procedure for example.
+--
+-- Optional argument 'deploy_dir' is used as a temporary place to place the
+-- downloaded packages into.
+--
+-- When optional 'force_no_download' parameter is set to true, then information
+-- about packages won't be downloaded during dependency resolving, assuming that
+-- entries in manifest are complete.
+--
+-- When optional 'suppress_printing' parameter is set to true, then messages
+-- for the user won't be printed during dependency resolving.
+function get_depends(packages, installed, manifest, dependency_manifest, deploy_dir, force_no_download, suppress_printing)
+ if not packages then return {} end
+ manifest = manifest or mf.get_manifest()
+ dependency_manifest = dependency_manifest or {}
+ deploy_dir = deploy_dir or cfg.root_dir
+ force_no_download = force_no_download or false
+ suppress_printing = suppress_printing or false
+ if type(packages) == "string" then packages = {packages} end
+
+ assert(type(packages) == "table", "depends.get_depends: Argument 'packages' is not a table or string.")
+ assert(type(installed) == "table", "depends.get_depends: Argument 'installed' is not a table.")
+ assert(type(manifest) == "table", "depends.get_depends: Argument 'manifest' is not a table.")
+ assert(type(dependency_manifest) == "table", "depends.get_depends: Argument 'dependency_manifest' is not a table.")
+ assert(type(deploy_dir) == "string", "depends.get_depends: Argument 'deploy_dir' is not a string.")
+ assert(type(force_no_download) == "boolean", "depends.get_depends: Argument 'force_no_download' is not a boolean.")
+ assert(type(suppress_printing) == "boolean", "depends.get_depends: Argument 'suppress_printing' is not a boolean.")
+ deploy_dir = sys.abs_path(deploy_dir)
+
+ local tmp_installed = utils.deepcopy(installed)
+
+ -- add provided packages to installed ones
+ for _, installed_pkg in pairs(tmp_installed) do
+ for _, pkg in pairs(get_provides(installed_pkg)) do
+ table.insert(tmp_installed, pkg)
+ end
+ end
+
+ -- If 'pkg' contains valid (architecture specific) path separator,
+ -- it is treated like a path to already downloaded package and
+ -- we assume that user wants to use this specific version of the
+ -- module to be installed. Hence, we will add information about
+ -- this version into the manifest and also remove references to
+ -- any other versions of this module from the manifest. This will
+ -- enforce the version of the module required by the user.
+ for k, pkg in pairs(packages) do
+ if pkg:find(sys.path_separator()) then
+ local pkg_dir = sys.abs_path(pkg)
+ local pkg_info, err = mf.load_distinfo(sys.make_path(pkg_dir, "dist.info"))
+ if not pkg_info then return nil, err end
+
+ -- add information about location of the package, also to prevent downloading it again
+ pkg_info.download_dir = pkg_dir
+ -- mark package to skip deleting its directory after installation
+ pkg_info.preserve_pkg_dir = true
+
+ -- set default arch/type if not explicitly stated and package is of source type
+ if package.is_source_type(pkg_dir) then
+ pkg_info = package.ensure_source_arch_and_type(pkg_info)
+ elseif not (pkg_info.arch and pkg_info.type) then
+ return nil, pkg_dir .. ": binary package missing arch or type in 'dist.info'."
+ end
+
+ -- update manifest
+ manifest = utils.filter(manifest, function(p) return p.name ~= pkg_info.name and true end)
+ table.insert(manifest, pkg_info)
+
+ -- update packages to install
+ pkg = pkg_info.name .. "-" .. pkg_info.version
+ packages[k] = pkg
+ end
+ end
+
+ local to_install = {}
+
+ -- get packages needed to satisfy the dependencies
+ for _, pkg in pairs(packages) do
+
+ local needed_to_install, err = get_packages_to_install(pkg, tmp_installed, manifest, dependency_manifest, force_no_download, suppress_printing, deploy_dir)
+
+ -- if everything's fine
+ if needed_to_install then
+
+ for _, needed_pkg in pairs(needed_to_install) do
+
+ -- TODO: why not to use 'installed' instead of 'tmp_installed'?
+ -- It's because provides aren't searched for by find()
+ -- function inside the update_dependency_manifest().
+ dependency_manifest = update_dependency_manifest(needed_pkg, tmp_installed, needed_to_install, dependency_manifest)
+
+ table.insert(to_install, needed_pkg)
+ table.insert(tmp_installed, needed_pkg)
+ -- add provides of needed_pkg to installed ones
+ for _, provided_pkg in pairs(get_provides(needed_pkg)) do
+ -- copy 'selected' property
+ provided_pkg.selected = needed_pkg.selected
+ table.insert(tmp_installed, provided_pkg)
+ end
+ end
+ -- if error occured
+ else
+ -- delete already downloaded packages
+ for _, pkg in pairs(to_install) do
+ if pkg.download_dir and not cfg.debug then sys.delete(pkg.download_dir) end
+ end
+ return nil, "Cannot resolve dependencies for '" .. pkg .. "': ".. err
+ end
+ end
+
+ return to_install, dependency_manifest
+end
+
+-- Return table of packages provided by specified package (from it's 'provides' field)
+function get_provides(package)
+ assert(type(package) == "table", "depends.get_provides: Argument 'package' is not a table.")
+ if not package.provides then return {} end
+
+ local provided = {}
+ for _, provided_name in pairs(package.provides) do
+ local pkg = {}
+ pkg.name, pkg.version = split_name_constraint(provided_name)
+ pkg.type = package.type
+ pkg.arch = package.arch
+ pkg.provided = package.name .. "-" .. package.version
+ table.insert(provided, pkg)
+ end
+ return provided
+end
+
+-- Return package name and version constraint from full package version constraint specification
+-- E. g.:
+-- for 'luaexpat-1.2.3' return: 'luaexpat' , '1.2.3'
+-- for 'luajit >= 1.2' return: 'luajit' , '>=1.2'
+function split_name_constraint(version_constraint)
+ assert(type(version_constraint) == "string", "depends.split_name_constraint: Argument 'version_constraint' is not a string.")
+
+ local split = version_constraint:find("[%s=~<>-]+%d") or version_constraint:find("[%s=~<>-]+scm")
+
+ if split then
+ return version_constraint:sub(1, split - 1), version_constraint:sub(split):gsub("[%s-]", "")
+ else
+ return version_constraint, nil
+ end
+end
+
+-- Return only packages that can be installed on the specified architecture and type
+function filter_packages_by_arch_and_type(packages, req_arch, req_type)
+ assert(type(packages) == "table", "depends.filter_packages_by_arch_and_type: Argument 'packages' is not a table.")
+ assert(type(req_arch) == "string", "depends.filter_packages_by_arch_and_type: Argument 'req_arch' is not a string.")
+ assert(type(req_type) == "string", "depends.filter_packages_by_arch_and_type: Argument 'pkg_type' is not a string.")
+
+ return utils.filter(packages,
+ function (pkg)
+ return (pkg.arch == "Universal" or pkg.arch == req_arch) and
+ (pkg.type == "all" or pkg.type == "source" or pkg.type == req_type)
+ end)
+end
+
+-- Return only packages that contain one of the specified strings in their 'name-version'.
+-- Case is ignored. If no strings are specified, return all the packages.
+-- Argument 'search_in_desc' specifies if search also in description of packages.
+function filter_packages_by_strings(packages, strings, search_in_desc)
+ if type(strings) == "string" then strings = {strings} end
+ assert(type(packages) == "table", "depends.filter_packages_by_strings: Argument 'packages' is not a table.")
+ assert(type(strings) == "table", "depends.filter_packages_by_strings: Argument 'strings' is not a string or table.")
+
+ if #strings ~= 0 then
+ return utils.filter(packages,
+ function (pkg)
+ for _,str in pairs(strings) do
+ local name = pkg.name .. "-" .. pkg.version
+ if search_in_desc then
+ name = name .. " " .. (pkg.desc or "")
+ end
+ if string.find(string.lower(name), string.lower(str), 1 ,true) ~= nil then return true end
+ end
+ end)
+ else
+ return packages
+ end
+end
+
+
+-- Return full package name and version string (e.g. 'luajit-2.0'). When version
+-- is nil or '' then return only name (e.g. 'luajit') and when name is nil or ''
+-- then return '<unknown>'. Optional 'was_scm_version' argument is a boolean,
+-- stating whether the package was originally selected for installation as a 'scm' version.
+function pkg_full_name(name, version, was_scm_version)
+ name = name or ""
+ version = version or ""
+ was_scm_version = was_scm_version or false
+ if type(version) == "number" then version = tostring(version) end
+
+ assert(type(name) == "string", "depends.pkg_full_name: Argument 'name' is not a string.")
+ assert(type(version) == "string", "depends.pkg_full_name: Argument 'version' is not a string.")
+
+ if was_scm_version then version = version .. " [scm version]" end
+
+ if name == "" then
+ return "<unknown>"
+ else
+ return name .. ((version ~= "") and "-" .. version or "")
+ end
+end
+
+-- Return table of packages, sorted descendingly by versions (newer ones are moved to the top).
+function sort_by_versions(packages)
+ assert(type(packages) == "table", "depends.sort_by_versions: Argument 'packages' is not a table.")
+ return utils.sort(packages, function (a, b) return compare_versions(a.version, b.version) end)
+end
+
+-- Return table of packages, sorted alphabetically by name and then descendingly by version.
+function sort_by_names(packages)
+ assert(type(packages) == "table", "depends.sort_by_names: Argument 'packages' is not a table.")
+ return utils.sort(packages, function (a, b)
+ if a.name == b.name then
+ return compare_versions(a.version, b.version)
+ else
+ return a.name < b.name
+ end
+ end)
+end
+
+-- Return if version satisfies the specified constraint
+function satisfies_constraint(version, constraint)
+ assert(type(version) == "string", "depends.satisfies_constraint: Argument 'version' is not a string.")
+ assert(type(constraint) == "string", "depends.satisfies_constraint: Argument 'constraint' is not a string.")
+ return const.constraint_satisfied(version, constraint)
+end
+
+-- For package versions, return whether: 'version_a' > 'version_b'
+function compare_versions(version_a, version_b)
+ assert(type(version_a) == "string", "depends.compare_versions: Argument 'version_a' is not a string.")
+ assert(type(version_b) == "string", "depends.compare_versions: Argument 'version_b' is not a string.")
+ return const.compareVersions(version_a, version_b)
+end
+
+-- Returns 'dep_manifest' updated with information about the 'pkg'.
+-- 'installed' is table with installed packages
+-- 'to_install' is table with packages that are selected for installation
+-- Packages satisfying the dependencies will be searched for in these two tables.
+function update_dependency_manifest(pkg, installed, to_install, dep_manifest)
+ dep_manifest = dep_manifest or {}
+ assert(type(pkg) == "table", "depends.update_dependency_manifest: Argument 'pkg' is not a table.")
+ assert(type(installed) == "table", "depends.update_dependency_manifest: Argument 'installed' is not a table.")
+ assert(type(to_install) == "table", "depends.update_dependency_manifest: Argument 'to_install' is not a table.")
+ assert(type(dep_manifest) == "table", "depends.update_dependency_manifest: Argument 'dep_manifest' is not a table.")
+
+ local name_ver = pkg.name .. "-" .. (pkg.was_scm_version and "scm" or pkg.version)
+
+ -- add to manifest
+ if not dep_manifest[name_ver] then
+ dep_manifest[name_ver] = {}
+ dep_manifest[name_ver].name = pkg.name
+ dep_manifest[name_ver].version = pkg.version
+ dep_manifest[name_ver].was_scm_version = pkg.was_scm_version
+ dep_manifest[name_ver].arch = pkg.arch
+ dep_manifest[name_ver].type = pkg.type
+ dep_manifest[name_ver].path = pkg.path
+ dep_manifest[name_ver].depends = pkg.depends
+ dep_manifest[name_ver].conflicts = pkg.conflicts
+ dep_manifest[name_ver].provides = pkg.provides
+ dep_manifest[name_ver].license = pkg.license
+ dep_manifest[name_ver].desc = pkg.desc
+ dep_manifest[name_ver].url = pkg.url
+ dep_manifest[name_ver].author = pkg.author
+ dep_manifest[name_ver].maintainer = pkg.maintainer
+
+ -- add information which dependency is satisfied by which package
+ if pkg.depends then
+
+ -- TODO: Won't it be better to add OS-specific 'satisfied_by' metadata in a format like OS-specific 'depends' ?
+ local all_deps = extract_os_specific_depends(pkg.depends)
+
+ dep_manifest[name_ver].satisfied_by = {}
+ for _, depend in pairs(all_deps) do
+
+ -- find package satisfying the dependency
+ local satisfying = find_packages(depend, installed)[1] or find_packages(depend, to_install)[1]
+ satisfying = satisfying.name .. "-" .. satisfying.version
+ dep_manifest[name_ver].satisfied_by[depend] = satisfying
+
+ -- check whether the satisfying package isn't provided by other one
+ local provided_by = utils.filter(installed, function(pkg)
+ return pkg.provides and utils.contains(pkg.provides, satisfying)
+ end)
+ if #provided_by == 0 then
+ provided_by = utils.filter(to_install, function(pkg)
+ return pkg.provides and utils.contains(pkg.provides, satisfying)
+ end)
+ end
+
+ if #provided_by ~= 0 then
+ if not dep_manifest[name_ver].satisfying_provided_by then
+ dep_manifest[name_ver].satisfying_provided_by = {}
+ end
+ dep_manifest[name_ver].satisfying_provided_by[satisfying] = provided_by[1].name .. "-" .. provided_by[1].version
+ end
+ end
+
+ end
+ end
+
+ return dep_manifest
+end