From 50b5755d3e9a2c6b2040d2d167accd02208ec687 Mon Sep 17 00:00:00 2001 From: Hisham Date: Thu, 20 Oct 2016 22:16:01 -0400 Subject: Add test for #632. --- spec/install_spec.lua | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/spec/install_spec.lua b/spec/install_spec.lua index 8d43ca2a..74f8f493 100644 --- a/spec/install_spec.lua +++ b/spec/install_spec.lua @@ -88,6 +88,19 @@ describe("LuaRocks install tests #blackbox #b_install", function() assert.is.truthy(lfs.attributes(testing_paths.testing_sys_tree .. "/lib/luarocks/rocks/luasec")) end) + it('LuaRocks install - handle relative path in --tree #632', function() + local relative_path = "./temp_dir_"..math.random(100000) + if test_env.TEST_TARGET_OS == "windows" then + relative_path = relative_path:gsub("/", "\\") + end + test_env.remove_dir(relative_path) + assert.is.falsy(lfs.attributes(relative_path)) + assert.is_true(run.luarocks_bool("install luafilesystem --tree="..relative_path)) + assert.is.truthy(lfs.attributes(relative_path)) + test_env.remove_dir(relative_path) + assert.is.falsy(lfs.attributes(relative_path)) + end) + it('LuaRocks install - handle versioned modules when installing another version with --keep #268', function() assert.is_true(run.luarocks_bool("install luafilesystem")) assert.is.truthy(lfs.attributes(testing_paths.testing_sys_tree .. "/lib/lua/"..env_variables.LUA_VERSION.."/lfs."..test_env.lib_extension)) -- cgit v1.2.3-55-g6feb From 51aa62b855241a12866d3d008e7019689c24d130 Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 15:26:35 +0300 Subject: Fix absolute path detection on Windows Do not use "^[%.a-zA-Z]?:?[\\/]" as a pattern when checking if a path is absolute on Windows. Check if there is a directory separator after an optional drive letter instead. --- src/luarocks/fs/win32.lua | 39 +++++++++++++++++++++++++++++++-------- 1 file changed, 31 insertions(+), 8 deletions(-) diff --git a/src/luarocks/fs/win32.lua b/src/luarocks/fs/win32.lua index 14804713..cfc28d35 100644 --- a/src/luarocks/fs/win32.lua +++ b/src/luarocks/fs/win32.lua @@ -33,7 +33,24 @@ function win32.quiet_stderr(cmd) return cmd.." 2> NUL" end -local drive_letter = "[%.a-zA-Z]?:?[\\/]" +-- Split path into root and the rest. +-- Root part consists of an optional drive letter (e.g. "C:") +-- and an optional directory separator. +local function split_root(path) + local root = "" + + if path:match("^.:") then + root = path:sub(1, 2) + path = path:sub(3) + end + + if path:match("^[\\/]") then + root = path:sub(1, 1) + path = path:sub(2) + end + + return root, path +end --- Quote argument for shell processing. Fixes paths on Windows. -- Adds double quotes and escapes. @@ -41,8 +58,9 @@ local drive_letter = "[%.a-zA-Z]?:?[\\/]" -- @return string: Quoted argument. function win32.Q(arg) assert(type(arg) == "string") - -- Quote DIR for Windows - if arg:match("^"..drive_letter) then + -- Use Windows-specific directory separator for paths. + -- Paths should be converted to absolute by now. + if split_root(arg) ~= "" then arg = arg:gsub("/", "\\") end if arg == "\\" then @@ -62,8 +80,9 @@ end -- @return string: Quoted argument. function win32.Qb(arg) assert(type(arg) == "string") - -- Quote DIR for Windows - if arg:match("^"..drive_letter) then + -- Use Windows-specific directory separator for paths. + -- Paths should be converted to absolute by now. + if split_root(arg) ~= "" then arg = arg:gsub("/", "\\") end if arg == "\\" then @@ -88,10 +107,14 @@ function win32.absolute_name(pathname, relative_to) assert(type(relative_to) == "string" or not relative_to) relative_to = relative_to or fs.current_dir() - if pathname:match("^"..drive_letter) then + local root, rest = split_root(pathname) + if root:match("[\\/]$") then + -- It's an absolute path already. return pathname else - return relative_to .. "/" .. pathname + -- It's a relative path, join it with base path. + -- This drops drive letter from paths like "C:foo". + return relative_to .. "/" .. rest end end @@ -100,7 +123,7 @@ end -- @param pathname string: pathname to use. -- @return string: The root of the given pathname. function win32.root_of(pathname) - return (fs.absolute_name(pathname):match("^("..drive_letter..")")) + return (split_root(fs.absolute_name(pathname))) end --- Create a wrapper to make a script executable from the command-line. -- cgit v1.2.3-55-g6feb From 0574fb3c6d015b80c1f0330e2bf5a630c5ec306a Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 17:24:01 +0300 Subject: Refactor store_results in luarocks.manif Instead of passing a function and its arguments to a helper function for it to call it, make a copy of result, and return it, call the function on the outer level and pass the result to helper. --- src/luarocks/manif.lua | 44 ++++++++++++++++++++++---------------------- 1 file changed, 22 insertions(+), 22 deletions(-) diff --git a/src/luarocks/manif.lua b/src/luarocks/manif.lua index 2d983a85..15a27c48 100644 --- a/src/luarocks/manif.lua +++ b/src/luarocks/manif.lua @@ -161,31 +161,28 @@ function manif.load_manifest(repo_url, lua_version) return manif_core.manifest_loader(pathname, repo_url, lua_version) end ---- Output a table listing items of a package. --- @param itemsfn function: a function for obtaining items of a package. --- pkg and version will be passed to it; it should return a table with --- items as keys. --- @param pkg string: package name --- @param version string: package version --- @param tbl table: the package matching table: keys should be item names --- and values arrays of strings with packages names in "name/version" format. -local function store_package_items(itemsfn, pkg, version, tbl) - assert(type(itemsfn) == "function") - assert(type(pkg) == "string") +--- Update storage table to account for items provided by a package. +-- @param storage table: a table storing items in the following format: +-- keys are item names and values are arrays of packages providing each item, +-- where a package is specified as string `name/version`. +-- @param items table: a table mapping item names to paths. +-- @param name string: package name. +-- @param version string: package version. +local function store_package_items(storage, name, version, items) + assert(type(storage) == "table") + assert(type(items) == "table") + assert(type(name) == "string") assert(type(version) == "string") - assert(type(tbl) == "table") - local pkg_version = pkg.."/"..version - local result = {} + local package_identifier = name.."/"..version - for item, path in pairs(itemsfn(pkg, version)) do - result[item] = path - if not tbl[item] then - tbl[item] = {} + for item_name, path in pairs(items) do + if not storage[item_name] then + storage[item_name] = {} end - table.insert(tbl[item], pkg_version) + + table.insert(storage[item_name], package_identifier) end - return result end --- Sort function for ordering rock identifiers in a manifest's @@ -340,8 +337,11 @@ local function store_results(results, manifest, dep_handler) if not rock_manifest then return nil, "rock_manifest file not found for "..name.." "..version.." - not a LuaRocks 2 tree?" end - entrytable.modules = store_package_items(repos.package_modules, name, version, manifest.modules) - entrytable.commands = store_package_items(repos.package_commands, name, version, manifest.commands) + + entrytable.modules = repos.package_modules(name, version) + store_package_items(manifest.modules, name, version, entrytable.modules) + entrytable.commands = repos.package_commands(name, version) + store_package_items(manifest.commands, name, version, entrytable.commands) end table.insert(versiontable, entrytable) end -- cgit v1.2.3-55-g6feb From 6a6d4e1c2f97b11caf826641b7dc1640effe283e Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 17:32:43 +0300 Subject: Refactor store_results in luarocks.manif (2) Do not accept a hook function to call after storing results. Simply call it on the outside after calling store_results. --- src/luarocks/manif.lua | 29 ++++++++--------------------- 1 file changed, 8 insertions(+), 21 deletions(-) diff --git a/src/luarocks/manif.lua b/src/luarocks/manif.lua index 15a27c48..4122364a 100644 --- a/src/luarocks/manif.lua +++ b/src/luarocks/manif.lua @@ -318,12 +318,10 @@ end -- @param results table: The search results as returned by search.disk_search. -- @param manifest table: A manifest table (must contain repository, modules, commands tables). -- It will be altered to include the search results. --- @param dep_handler: dependency handler function -- @return boolean or (nil, string): true in case of success, or nil followed by an error message. -local function store_results(results, manifest, dep_handler) +local function store_results(results, manifest) assert(type(results) == "table") assert(type(manifest) == "table") - assert((not dep_handler) or type(dep_handler) == "function") for name, versions in pairs(results) do local pkgtable = manifest.repository[name] or {} @@ -349,9 +347,6 @@ local function store_results(results, manifest, dep_handler) end manifest.repository[name] = pkgtable end - if dep_handler then - dep_handler(manifest) - end sort_package_matching_table(manifest.modules) sort_package_matching_table(manifest.commands) return true @@ -385,25 +380,19 @@ function manif.make_manifest(repo, deps_mode, remote) manif_core.cache_manifest(repo, nil, manifest) - local dep_handler = nil - if not remote then - dep_handler = function(manifest) - update_dependencies(manifest, deps_mode) - end - end - local ok, err = store_results(results, manifest, dep_handler) + local ok, err = store_results(results, manifest) if not ok then return nil, err end if remote then local cache = {} for luaver in util.lua_versions() do local vmanifest = { repository = {}, modules = {}, commands = {} } - local dep_handler = function(manifest) - filter_by_lua_version(manifest, luaver, repo, cache) - end - local ok, err = store_results(results, vmanifest, dep_handler) + local ok, err = store_results(results, vmanifest) + filter_by_lua_version(vmanifest, luaver, repo, cache) save_table(repo, "manifest-"..luaver, vmanifest) end + else + update_dependencies(manifest, deps_mode) end return save_table(repo, "manifest", manifest) @@ -445,12 +434,10 @@ function manif.update_manifest(name, version, repo, deps_mode) local results = {[name] = {[version] = {{arch = "installed", repo = repo}}}} - local dep_handler = function(manifest) - update_dependencies(manifest, deps_mode) - end - local ok, err = store_results(results, manifest, dep_handler) + local ok, err = store_results(results, manifest) if not ok then return nil, err end + update_dependencies(manifest, deps_mode) return save_table(repo, "manifest", manifest) end -- cgit v1.2.3-55-g6feb From b09ffc7d1c207a5cc4dda7b6ba1f52ede222327b Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 17:57:42 +0300 Subject: Rename a variable in manif.update_manifest `repo` sometimes means root dir and sometimes means rock dir. At least do not reuse the same variable for both. --- src/luarocks/manif.lua | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/luarocks/manif.lua b/src/luarocks/manif.lua index 4122364a..d697a93c 100644 --- a/src/luarocks/manif.lua +++ b/src/luarocks/manif.lua @@ -414,31 +414,31 @@ end function manif.update_manifest(name, version, repo, deps_mode) assert(type(name) == "string") assert(type(version) == "string") - repo = path.rocks_dir(repo or cfg.root_dir) + local rocks_dir = path.rocks_dir(repo or cfg.root_dir) assert(type(deps_mode) == "string") if deps_mode == "none" then deps_mode = cfg.deps_mode end - local manifest, err = manif.load_manifest(repo) + local manifest, err = manif.load_manifest(rocks_dir) if not manifest then util.printerr("No existing manifest. Attempting to rebuild...") - local ok, err = manif.make_manifest(repo, deps_mode) + local ok, err = manif.make_manifest(rocks_dir, deps_mode) if not ok then return nil, err end - manifest, err = manif.load_manifest(repo) + manifest, err = manif.load_manifest(rocks_dir) if not manifest then return nil, err end end - local results = {[name] = {[version] = {{arch = "installed", repo = repo}}}} + local results = {[name] = {[version] = {{arch = "installed", repo = rocks_dir}}}} local ok, err = store_results(results, manifest) if not ok then return nil, err end update_dependencies(manifest, deps_mode) - return save_table(repo, "manifest", manifest) + return save_table(rocks_dir, "manifest", manifest) end function manif.zip_manifests() -- cgit v1.2.3-55-g6feb From 692c3250cdac66f3dd1379784e57f5e7fa0c53b9 Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 18:00:12 +0300 Subject: manif.update_manifest: load manifest as local --- src/luarocks/manif.lua | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/luarocks/manif.lua b/src/luarocks/manif.lua index d697a93c..6b42912d 100644 --- a/src/luarocks/manif.lua +++ b/src/luarocks/manif.lua @@ -419,7 +419,7 @@ function manif.update_manifest(name, version, repo, deps_mode) if deps_mode == "none" then deps_mode = cfg.deps_mode end - local manifest, err = manif.load_manifest(rocks_dir) + local manifest, err = manif_core.load_local_manifest(rocks_dir) if not manifest then util.printerr("No existing manifest. Attempting to rebuild...") local ok, err = manif.make_manifest(rocks_dir, deps_mode) -- cgit v1.2.3-55-g6feb From fb8a377a8735382921f72cd97bb98a0aef38de72 Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 18:03:30 +0300 Subject: manif.update_manifest: do not update manifest if rebuilt --- src/luarocks/manif.lua | 12 ++++-------- 1 file changed, 4 insertions(+), 8 deletions(-) diff --git a/src/luarocks/manif.lua b/src/luarocks/manif.lua index 6b42912d..9213f362 100644 --- a/src/luarocks/manif.lua +++ b/src/luarocks/manif.lua @@ -422,14 +422,10 @@ function manif.update_manifest(name, version, repo, deps_mode) local manifest, err = manif_core.load_local_manifest(rocks_dir) if not manifest then util.printerr("No existing manifest. Attempting to rebuild...") - local ok, err = manif.make_manifest(rocks_dir, deps_mode) - if not ok then - return nil, err - end - manifest, err = manif.load_manifest(rocks_dir) - if not manifest then - return nil, err - end + -- Manifest built by `manif.make_manifest` should already + -- include information about given name and version, + -- no need to update it. + return manif.make_manifest(rocks_dir, deps_mode) end local results = {[name] = {[version] = {{arch = "installed", repo = rocks_dir}}}} -- cgit v1.2.3-55-g6feb From 7ce4a5c9accb22eb1fc422036c72f809e8c026b9 Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Fri, 28 Oct 2016 18:36:21 +0300 Subject: Update manifest after removal without rebuilding Rename `manif.update_manifest` to `manif.add_to_manifest`. Add `manif.remove_from_manifest` that performs reverse action. Use it in `repos.delete_version` to avoid rebuilding manifest everytime a package is removed. --- src/luarocks/manif.lua | 89 ++++++++++++++++++++++++++++++++++++++++++++++---- src/luarocks/repos.lua | 4 +-- 2 files changed, 84 insertions(+), 9 deletions(-) diff --git a/src/luarocks/manif.lua b/src/luarocks/manif.lua index 9213f362..c4c52bd2 100644 --- a/src/luarocks/manif.lua +++ b/src/luarocks/manif.lua @@ -185,6 +185,37 @@ local function store_package_items(storage, name, version, items) end end +--- Update storage table removing items provided by a package. +-- @param storage table: a table storing items in the following format: +-- keys are item names and values are arrays of packages providing each item, +-- where a package is specified as string `name/version`. +-- @param items table: a table mapping item names to paths. +-- @param name string: package name. +-- @param version string: package version. +local function remove_package_items(storage, name, version, items) + assert(type(storage) == "table") + assert(type(items) == "table") + assert(type(name) == "string") + assert(type(version) == "string") + + local package_identifier = name.."/"..version + + for item_name, path in pairs(items) do + local all_identifiers = storage[item_name] + + for i, identifier in ipairs(all_identifiers) do + if identifier == package_identifier then + table.remove(all_identifiers, i) + break + end + end + + if #all_identifiers == 0 then + storage[item_name] = nil + end + end +end + --- Sort function for ordering rock identifiers in a manifest's -- modules table. Rocks are ordered alphabetically by name, and then -- by version which greater first. @@ -398,10 +429,8 @@ function manif.make_manifest(repo, deps_mode, remote) return save_table(repo, "manifest", manifest) end ---- Load a manifest file from a local repository and add to the repository --- information with regard to the given name and version. --- A file called 'manifest' will be written in the root of the given --- repository directory. +--- Update manifest file for a local repository +-- adding information about a version of a package installed in that repository. -- @param name string: Name of a package from the repository. -- @param version string: Version of a package from the repository. -- @param repo string or nil: Pathname of a local repository. If not given, @@ -409,14 +438,14 @@ end -- @param deps_mode string: Dependency mode: "one" for the current default tree, -- "all" for all trees, "order" for all trees with priority >= the current default, -- "none" for using the default dependency mode from the configuration. --- @return boolean or (nil, string): True if manifest was generated, +-- @return boolean or (nil, string): True if manifest was updated successfully, -- or nil and an error message. -function manif.update_manifest(name, version, repo, deps_mode) +function manif.add_to_manifest(name, version, repo, deps_mode) assert(type(name) == "string") assert(type(version) == "string") local rocks_dir = path.rocks_dir(repo or cfg.root_dir) assert(type(deps_mode) == "string") - + if deps_mode == "none" then deps_mode = cfg.deps_mode end local manifest, err = manif_core.load_local_manifest(rocks_dir) @@ -437,6 +466,52 @@ function manif.update_manifest(name, version, repo, deps_mode) return save_table(rocks_dir, "manifest", manifest) end +--- Update manifest file for a local repository +-- removing information about a version of a package. +-- @param name string: Name of a package removed from the repository. +-- @param version string: Version of a package removed from the repository. +-- @param repo string or nil: Pathname of a local repository. If not given, +-- the default local repository is used. +-- @param deps_mode string: Dependency mode: "one" for the current default tree, +-- "all" for all trees, "order" for all trees with priority >= the current default, +-- "none" for using the default dependency mode from the configuration. +-- @return boolean or (nil, string): True if manifest was updated successfully, +-- or nil and an error message. +function manif.remove_from_manifest(name, version, repo, deps_mode) + assert(type(name) == "string") + assert(type(version) == "string") + local rocks_dir = path.rocks_dir(repo or cfg.root_dir) + assert(type(deps_mode) == "string") + + if deps_mode == "none" then deps_mode = cfg.deps_mode end + + local manifest, err = manif_core.load_local_manifest(rocks_dir) + if not manifest then + util.printerr("No existing manifest. Attempting to rebuild...") + -- Manifest built by `manif.make_manifest` should already + -- include up-to-date information, no need to update it. + return manif.make_manifest(rocks_dir, deps_mode) + end + + local package_entry = manifest.repository[name] + + local version_entry = package_entry[version][1] + remove_package_items(manifest.modules, name, version, version_entry.modules) + remove_package_items(manifest.commands, name, version, version_entry.commands) + + package_entry[version] = nil + manifest.dependencies[name][version] = nil + + if not next(package_entry) then + -- No more versions of this package. + manifest.repository[name] = nil + manifest.dependencies[name] = nil + end + + update_dependencies(manifest, deps_mode) + return save_table(rocks_dir, "manifest", manifest) +end + function manif.zip_manifests() for ver in util.lua_versions() do local file = "manifest-"..ver diff --git a/src/luarocks/repos.lua b/src/luarocks/repos.lua index c5f157c4..d4d9694e 100644 --- a/src/luarocks/repos.lua +++ b/src/luarocks/repos.lua @@ -312,7 +312,7 @@ function repos.deploy_files(name, version, wrap_bin_scripts, deps_mode) return nil, err end - return manif.update_manifest(name, version, nil, deps_mode) + return manif.add_to_manifest(name, version, nil, deps_mode) end --- Delete a package from the local repository. @@ -398,7 +398,7 @@ function repos.delete_version(name, version, deps_mode, quick) return true end - return manif.make_manifest(cfg.rocks_dir, deps_mode) + return manif.remove_from_manifest(name, version, nil, deps_mode) end return repos -- cgit v1.2.3-55-g6feb From f04b02b27fac9c409d0da5628026648f8329a99b Mon Sep 17 00:00:00 2001 From: Peter Melnichenko Date: Sat, 29 Oct 2016 16:03:25 +0300 Subject: Improve performance of manif_core.get_versions To get all installed versions of a package using deps_mode, instead of deeply merging all manifests of corresponding trees, shallowly merge only tables related to that package. This affects speed of manifest manipulation and dependency resolution. Minimal performance testing suggests 3x-4x speed-up on reinstallation of a package using `luarocks make`. --- src/luarocks/manif_core.lua | 21 ++++++++++----------- 1 file changed, 10 insertions(+), 11 deletions(-) diff --git a/src/luarocks/manif_core.lua b/src/luarocks/manif_core.lua index 5c8928d4..82e7ea4d 100644 --- a/src/luarocks/manif_core.lua +++ b/src/luarocks/manif_core.lua @@ -88,20 +88,19 @@ end function manif_core.get_versions(name, deps_mode) assert(type(name) == "string") assert(type(deps_mode) == "string") - - local manifest = {} + + local version_set = {} path.map_trees(deps_mode, function(tree) - local loaded = manif_core.load_local_manifest(path.rocks_dir(tree)) - if loaded then - util.deep_merge(manifest, loaded) + local manifest = manif_core.load_local_manifest(path.rocks_dir(tree)) + + if manifest and manifest.repository[name] then + for version in pairs(manifest.repository[name]) do + version_set[version] = true + end end end) - - local item = next(manifest) and manifest.repository[name] - if item then - return util.keys(item) - end - return {} + + return util.keys(version_set) end return manif_core -- cgit v1.2.3-55-g6feb From bcdb901611b05870692de69bd2a7cd1eec88a114 Mon Sep 17 00:00:00 2001 From: Hisham Date: Sat, 29 Oct 2016 17:47:27 -0200 Subject: Adjust changes by @mpeterv in #638 to new locations. --- src/luarocks/manif/writer.lua | 189 +++++++++++++++++++++++++++--------------- src/luarocks/repos.lua | 4 +- 2 files changed, 126 insertions(+), 67 deletions(-) diff --git a/src/luarocks/manif/writer.lua b/src/luarocks/manif/writer.lua index 1eb5ee7c..df971fca 100644 --- a/src/luarocks/manif/writer.lua +++ b/src/luarocks/manif/writer.lua @@ -13,33 +13,60 @@ local path = require("luarocks.path") local persist = require("luarocks.persist") local manif = require("luarocks.manif") ---- Output a table listing items of a package. --- @param itemsfn function: a function for obtaining items of a package. --- pkg and version will be passed to it; it should return a table with --- items as keys. --- @param pkg string: package name --- @param version string: package version --- @param tbl table: the package matching table: keys should be item names --- and values arrays of strings with packages names in "name/version" format. -local function store_package_items(itemsfn, pkg, version, tbl) - assert(type(itemsfn) == "function") - assert(type(pkg) == "string") +--- Update storage table to account for items provided by a package. +-- @param storage table: a table storing items in the following format: +-- keys are item names and values are arrays of packages providing each item, +-- where a package is specified as string `name/version`. +-- @param items table: a table mapping item names to paths. +-- @param name string: package name. +-- @param version string: package version. +local function store_package_items(storage, name, version, items) + assert(type(storage) == "table") + assert(type(items) == "table") + assert(type(name) == "string") assert(type(version) == "string") - assert(type(tbl) == "table") - local pkg_version = pkg.."/"..version - local result = {} + local package_identifier = name.."/"..version - for item, path in pairs(itemsfn(pkg, version)) do - result[item] = path - if not tbl[item] then - tbl[item] = {} + for item_name, path in pairs(items) do + if not storage[item_name] then + storage[item_name] = {} end - table.insert(tbl[item], pkg_version) + + table.insert(storage[item_name], package_identifier) end - return result end +--- Update storage table removing items provided by a package. +-- @param storage table: a table storing items in the following format: +-- keys are item names and values are arrays of packages providing each item, +-- where a package is specified as string `name/version`. +-- @param items table: a table mapping item names to paths. +-- @param name string: package name. +-- @param version string: package version. +local function remove_package_items(storage, name, version, items) + assert(type(storage) == "table") + assert(type(items) == "table") + assert(type(name) == "string") + assert(type(version) == "string") + + local package_identifier = name.."/"..version + + for item_name, path in pairs(items) do + local all_identifiers = storage[item_name] + + for i, identifier in ipairs(all_identifiers) do + if identifier == package_identifier then + table.remove(all_identifiers, i) + break + end + end + + if #all_identifiers == 0 then + storage[item_name] = nil + end + end +end --- Process the dependencies of a manifest table to determine its dependency -- chains for loading modules. The manifest dependencies information is filled @@ -174,12 +201,10 @@ end -- @param results table: The search results as returned by search.disk_search. -- @param manifest table: A manifest table (must contain repository, modules, commands tables). -- It will be altered to include the search results. --- @param dep_handler: dependency handler function -- @return boolean or (nil, string): true in case of success, or nil followed by an error message. -local function store_results(results, manifest, dep_handler) +local function store_results(results, manifest) assert(type(results) == "table") assert(type(manifest) == "table") - assert((not dep_handler) or type(dep_handler) == "function") for name, versions in pairs(results) do local pkgtable = manifest.repository[name] or {} @@ -189,10 +214,15 @@ local function store_results(results, manifest, dep_handler) local entrytable = {} entrytable.arch = entry.arch if entry.arch == "installed" then - local rock_manifest, err = manif.load_rock_manifest(name, version) - if not rock_manifest then return nil, err end - entrytable.modules = store_package_items(repos.package_modules, name, version, manifest.modules) - entrytable.commands = store_package_items(repos.package_commands, name, version, manifest.commands) + local rock_manifest = manif.load_rock_manifest(name, version) + if not rock_manifest then + return nil, "rock_manifest file not found for "..name.." "..version.." - not a LuaRocks 2 tree?" + end + + entrytable.modules = repos.package_modules(name, version) + store_package_items(manifest.modules, name, version, entrytable.modules) + entrytable.commands = repos.package_commands(name, version) + store_package_items(manifest.commands, name, version, entrytable.commands) end table.insert(versiontable, entrytable) end @@ -200,9 +230,6 @@ local function store_results(results, manifest, dep_handler) end manifest.repository[name] = pkgtable end - if dep_handler then - dep_handler(manifest) - end sort_package_matching_table(manifest.modules) sort_package_matching_table(manifest.commands) return true @@ -265,7 +292,7 @@ end -- @param deps_mode string: Dependency mode: "one" for the current default tree, -- "all" for all trees, "order" for all trees with priority >= the current default, -- "none" for the default dependency mode from the configuration. --- @param versioned boolean: if versioned versions of the manifest should be created. +-- @param remote boolean: 'true' if making a manifest for a rocks server. -- @return boolean or (nil, string): True if manifest was generated, -- or nil and an error message. function writer.make_manifest(repo, deps_mode, remote) @@ -286,34 +313,26 @@ function writer.make_manifest(repo, deps_mode, remote) manif.cache_manifest(repo, nil, manifest) - local dep_handler = nil - if not remote then - dep_handler = function(manifest) - update_dependencies(manifest, deps_mode) - end - end - local ok, err = store_results(results, manifest, dep_handler) + local ok, err = store_results(results, manifest) if not ok then return nil, err end if remote then local cache = {} for luaver in util.lua_versions() do local vmanifest = { repository = {}, modules = {}, commands = {} } - local dep_handler = function(manifest) - filter_by_lua_version(manifest, luaver, repo, cache) - end - store_results(results, vmanifest, dep_handler) + local ok, err = store_results(results, vmanifest) + filter_by_lua_version(vmanifest, luaver, repo, cache) save_table(repo, "manifest-"..luaver, vmanifest) end + else + update_dependencies(manifest, deps_mode) end return save_table(repo, "manifest", manifest) end ---- Load a manifest file from a local repository and add to the repository --- information with regard to the given name and version. --- A file called 'manifest' will be written in the root of the given --- repository directory. +--- Update manifest file for a local repository +-- adding information about a version of a package installed in that repository. -- @param name string: Name of a package from the repository. -- @param version string: Version of a package from the repository. -- @param repo string or nil: Pathname of a local repository. If not given, @@ -321,38 +340,78 @@ end -- @param deps_mode string: Dependency mode: "one" for the current default tree, -- "all" for all trees, "order" for all trees with priority >= the current default, -- "none" for using the default dependency mode from the configuration. --- @return boolean or (nil, string): True if manifest was generated, +-- @return boolean or (nil, string): True if manifest was updated successfully, -- or nil and an error message. -function writer.update_manifest(name, version, repo, deps_mode) +function writer.add_to_manifest(name, version, repo, deps_mode) assert(type(name) == "string") assert(type(version) == "string") - repo = path.rocks_dir(repo or cfg.root_dir) + local rocks_dir = path.rocks_dir(repo or cfg.root_dir) assert(type(deps_mode) == "string") - + if deps_mode == "none" then deps_mode = cfg.deps_mode end - local manifest, err = manif.load_manifest(repo) + local manifest, err = manif.load_local_manifest(rocks_dir) if not manifest then util.printerr("No existing manifest. Attempting to rebuild...") - local ok, err = writer.make_manifest(repo, deps_mode) - if not ok then - return nil, err - end - manifest, err = manif.load_manifest(repo) - if not manifest then - return nil, err - end + -- Manifest built by `manif.make_manifest` should already + -- include information about given name and version, + -- no need to update it. + return manif.make_manifest(rocks_dir, deps_mode) end - local results = {[name] = {[version] = {{arch = "installed", repo = repo}}}} + local results = {[name] = {[version] = {{arch = "installed", repo = rocks_dir}}}} - local dep_handler = function(manifest) - update_dependencies(manifest, deps_mode) - end - local ok, err = store_results(results, manifest, dep_handler) + local ok, err = store_results(results, manifest) if not ok then return nil, err end - return save_table(repo, "manifest", manifest) + update_dependencies(manifest, deps_mode) + return save_table(rocks_dir, "manifest", manifest) +end + +--- Update manifest file for a local repository +-- removing information about a version of a package. +-- @param name string: Name of a package removed from the repository. +-- @param version string: Version of a package removed from the repository. +-- @param repo string or nil: Pathname of a local repository. If not given, +-- the default local repository is used. +-- @param deps_mode string: Dependency mode: "one" for the current default tree, +-- "all" for all trees, "order" for all trees with priority >= the current default, +-- "none" for using the default dependency mode from the configuration. +-- @return boolean or (nil, string): True if manifest was updated successfully, +-- or nil and an error message. +function writer.remove_from_manifest(name, version, repo, deps_mode) + assert(type(name) == "string") + assert(type(version) == "string") + local rocks_dir = path.rocks_dir(repo or cfg.root_dir) + assert(type(deps_mode) == "string") + + if deps_mode == "none" then deps_mode = cfg.deps_mode end + + local manifest, err = manif.load_local_manifest(rocks_dir) + if not manifest then + util.printerr("No existing manifest. Attempting to rebuild...") + -- Manifest built by `manif.make_manifest` should already + -- include up-to-date information, no need to update it. + return manif.make_manifest(rocks_dir, deps_mode) + end + + local package_entry = manifest.repository[name] + + local version_entry = package_entry[version][1] + remove_package_items(manifest.modules, name, version, version_entry.modules) + remove_package_items(manifest.commands, name, version, version_entry.commands) + + package_entry[version] = nil + manifest.dependencies[name][version] = nil + + if not next(package_entry) then + -- No more versions of this package. + manifest.repository[name] = nil + manifest.dependencies[name] = nil + end + + update_dependencies(manifest, deps_mode) + return save_table(rocks_dir, "manifest", manifest) end return writer diff --git a/src/luarocks/repos.lua b/src/luarocks/repos.lua index 9616fe03..bbc9da79 100644 --- a/src/luarocks/repos.lua +++ b/src/luarocks/repos.lua @@ -315,7 +315,7 @@ function repos.deploy_files(name, version, wrap_bin_scripts, deps_mode) end local writer = require("luarocks.manif.writer") - return writer.update_manifest(name, version, nil, deps_mode) + return writer.add_to_manifest(name, version, nil, deps_mode) end --- Delete a package from the local repository. @@ -400,7 +400,7 @@ function repos.delete_version(name, version, deps_mode, quick) end local writer = require("luarocks.manif.writer") - return writer.make_manifest(cfg.rocks_dir, deps_mode) + return writer.remove_from_manifest(name, version, nil, deps_mode) end return repos -- cgit v1.2.3-55-g6feb