mirror of
				https://github.com/neovim/neovim.git
				synced 2025-10-25 20:07:09 +00:00 
			
		
		
		
	 f05a2891d3
			
		
	
	f05a2891d3
	
	
	
		
			
			Problem:
Dirs "config", "packaging", and "third-party" are all closely related
but this is not obvious from the layout. This adds friction for new
contributors.
Solution:
- rename config/ to cmake.config/
- rename test/config/ to test/cmakeconfig/ because it is used in Lua
  tests: require('test.cmakeconfig.paths').
- rename packaging/ to cmake.packaging/
- rename third-party/ to cmake.deps/ (parallel with .deps/)
		
	
		
			
				
	
	
		
			344 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
			
		
		
	
	
			344 lines
		
	
	
		
			9.1 KiB
		
	
	
	
		
			Lua
		
	
	
	
	
	
| -- Usage:
 | |
| --    # bump to version
 | |
| --    nvim -es +"lua require('scripts.bump_deps').version(dependency, version_tag)"
 | |
| --
 | |
| --    # bump to commit
 | |
| --    nvim -es +"lua require('scripts.bump_deps').commit(dependency, commit_hash)"
 | |
| --
 | |
| --    # bump to HEAD
 | |
| --    nvim -es +"lua require('scripts.bump_deps').head(dependency)"
 | |
| --
 | |
| --    # submit PR
 | |
| --    nvim -es +"lua require('scripts.bump_deps').submit_pr()"
 | |
| --
 | |
| --    # create branch
 | |
| --    nvim -es +"lua require('scripts.bump_deps').create_branch()"
 | |
| 
 | |
| local M = {}
 | |
| 
 | |
| local _trace = false
 | |
| local required_branch_prefix = "bump-"
 | |
| local commit_prefix = "build(deps): "
 | |
| 
 | |
| -- Print message
 | |
| local function p(s)
 | |
| 	vim.cmd("set verbose=1")
 | |
| 	vim.api.nvim_echo({ { s, "" } }, false, {})
 | |
| 	vim.cmd("set verbose=0")
 | |
| end
 | |
| 
 | |
| local function die()
 | |
| 	p("")
 | |
| 	vim.cmd("cquit 1")
 | |
| end
 | |
| 
 | |
| -- Executes and returns the output of `cmd`, or nil on failure.
 | |
| -- if die_on_fail is true, process dies with die_msg on failure
 | |
| --
 | |
| -- Prints `cmd` if `trace` is enabled.
 | |
| local function _run(cmd, die_on_fail, die_msg)
 | |
| 	if _trace then
 | |
| 		p("run: " .. vim.inspect(cmd))
 | |
| 	end
 | |
| 	local rv = vim.trim(vim.fn.system(cmd)) or ""
 | |
| 	if vim.v.shell_error ~= 0 then
 | |
| 		if die_on_fail then
 | |
| 			if _trace then
 | |
| 				p(rv)
 | |
| 			end
 | |
| 			p(die_msg)
 | |
| 			die()
 | |
| 		end
 | |
| 		return nil
 | |
| 	end
 | |
| 	return rv
 | |
| end
 | |
| 
 | |
| -- Run a command, return nil on failure
 | |
| local function run(cmd)
 | |
| 	return _run(cmd, false, "")
 | |
| end
 | |
| 
 | |
| -- Run a command, die on failure with err_msg
 | |
| local function run_die(cmd, err_msg)
 | |
| 	return _run(cmd, true, err_msg)
 | |
| end
 | |
| 
 | |
| local function require_executable(cmd)
 | |
| 	local cmd_path = run_die({ "command", "-v", cmd }, cmd .. " not found!")
 | |
| 	run_die({ "test", "-x", cmd_path }, cmd .. " is not executable")
 | |
| end
 | |
| 
 | |
| local function rm_file_if_present(path_to_file)
 | |
| 	run({ "rm", "-f", path_to_file })
 | |
| end
 | |
| 
 | |
| local nvim_src_dir = vim.fn.getcwd()
 | |
| local temp_dir = nvim_src_dir .. "/tmp"
 | |
| run({ "mkdir", "-p", temp_dir })
 | |
| 
 | |
| local function get_dependency(dependency_name)
 | |
| 	local dependency_table = {
 | |
| 		["LuaJIT"] = {
 | |
| 			repo = "LuaJIT/LuaJIT",
 | |
| 			symbol = "LUAJIT",
 | |
| 		},
 | |
| 		["libuv"] = {
 | |
| 			repo = "libuv/libuv",
 | |
| 			symbol = "LIBUV",
 | |
| 		},
 | |
| 		["Luv"] = {
 | |
| 			repo = "luvit/luv",
 | |
| 			symbol = "LUV",
 | |
| 		},
 | |
| 		["tree-sitter"] = {
 | |
| 			repo = "tree-sitter/tree-sitter",
 | |
| 			symbol = "TREESITTER",
 | |
| 		},
 | |
| 	}
 | |
| 	local dependency = dependency_table[dependency_name]
 | |
| 	if dependency == nil then
 | |
| 		p("Not a dependency: " .. dependency_name)
 | |
| 		die()
 | |
| 	end
 | |
| 	dependency.name = dependency_name
 | |
| 	return dependency
 | |
| end
 | |
| 
 | |
| local function get_gh_commit_sha(repo, ref)
 | |
| 	require_executable("gh")
 | |
| 
 | |
| 	local sha = run_die(
 | |
| 		{ "gh", "api", "repos/" .. repo .. "/commits/" .. ref, "--jq", ".sha" },
 | |
| 		"Failed to get commit hash from GitHub. Not a valid ref?"
 | |
| 	)
 | |
| 	return sha
 | |
| end
 | |
| 
 | |
| local function get_archive_info(repo, ref)
 | |
| 	require_executable("curl")
 | |
| 
 | |
| 	local archive_name = ref .. ".tar.gz"
 | |
| 	local archive_path = temp_dir .. "/" .. archive_name
 | |
| 	local archive_url = "https://github.com/" .. repo .. "/archive/" .. archive_name
 | |
| 
 | |
| 	rm_file_if_present(archive_path)
 | |
| 	run_die({ "curl", "-sL", archive_url, "-o", archive_path }, "Failed to download archive from GitHub")
 | |
| 
 | |
| 	local archive_sha = run({ "sha256sum", archive_path }):gmatch("%w+")()
 | |
| 	return { url = archive_url, sha = archive_sha }
 | |
| end
 | |
| 
 | |
| local function write_cmakelists_line(symbol, kind, value)
 | |
| 	require_executable("sed")
 | |
| 
 | |
| 	local cmakelists_path = nvim_src_dir .. "/" .. "cmake.deps/CMakeLists.txt"
 | |
| 	run_die({
 | |
| 		"sed",
 | |
| 		"-i",
 | |
| 		"-e",
 | |
| 		"s/set(" .. symbol .. "_" .. kind .. ".*$" .. "/set(" .. symbol .. "_" .. kind .. " " .. value .. ")" .. "/",
 | |
| 		cmakelists_path,
 | |
| 	}, "Failed to write " .. cmakelists_path)
 | |
| end
 | |
| 
 | |
| local function explicit_create_branch(dep)
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	local checked_out_branch = run({ "git", "rev-parse", "--abbrev-ref", "HEAD" })
 | |
| 	if checked_out_branch ~= "master" then
 | |
| 		p("Not on master!")
 | |
| 		die()
 | |
| 	end
 | |
| 	run_die({ "git", "checkout", "-b", "bump-" .. dep }, "git failed to create branch")
 | |
| end
 | |
| 
 | |
| local function verify_branch(new_branch_suffix)
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	local checked_out_branch = run({ "git", "rev-parse", "--abbrev-ref", "HEAD" })
 | |
| 	if not checked_out_branch:match("^" .. required_branch_prefix) then
 | |
| 		p("Current branch '" .. checked_out_branch .. "' doesn't seem to start with " .. required_branch_prefix)
 | |
| 		p("Checking out to bump-" .. new_branch_suffix)
 | |
| 		explicit_create_branch(new_branch_suffix)
 | |
| 	end
 | |
| end
 | |
| 
 | |
| local function update_cmakelists(dependency, archive, comment)
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	verify_branch(dependency.name)
 | |
| 
 | |
| 	local changed_file = nvim_src_dir .. "/" .. "cmake.deps/CMakeLists.txt"
 | |
| 
 | |
| 	p("Updating " .. dependency.name .. " to " .. archive.url .. "\n")
 | |
| 	write_cmakelists_line(dependency.symbol, "URL", archive.url:gsub("/", "\\/"))
 | |
| 	write_cmakelists_line(dependency.symbol, "SHA256", archive.sha)
 | |
| 	run_die(
 | |
| 		{ "git", "commit", changed_file, "-m", commit_prefix .. "bump " .. dependency.name .. " to " .. comment },
 | |
| 		"git failed to commit"
 | |
| 	)
 | |
| end
 | |
| 
 | |
| local function verify_cmakelists_committed()
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	local cmakelists_path = nvim_src_dir .. "/" .. "cmake.deps/CMakeLists.txt"
 | |
| 	run_die({ "git", "diff", "--quiet", "HEAD", "--", cmakelists_path }, cmakelists_path .. " has uncommitted changes")
 | |
| end
 | |
| 
 | |
| local function warn_luv_symbol()
 | |
| 	p("warning: " .. get_dependency("Luv").symbol .. "_VERSION will not be updated")
 | |
| end
 | |
| 
 | |
| -- return first 9 chars of commit
 | |
| local function short_commit(commit)
 | |
| 	return string.sub(commit, 1, 9)
 | |
| end
 | |
| 
 | |
| -- TODO: remove hardcoded fork
 | |
| local function gh_pr(pr_title, pr_body)
 | |
| 	require_executable("gh")
 | |
| 
 | |
| 	local pr_url = run_die({
 | |
| 		"gh",
 | |
| 		"pr",
 | |
| 		"create",
 | |
| 		"--title",
 | |
| 		pr_title,
 | |
| 		"--body",
 | |
| 		pr_body,
 | |
| 	}, "Failed to create PR")
 | |
| 	return pr_url
 | |
| end
 | |
| 
 | |
| local function find_git_remote(fork)
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	local remotes = run({ "git", "remote", "-v" })
 | |
| 	local git_remote = ""
 | |
| 	for remote in remotes:gmatch("[^\r\n]+") do
 | |
| 		local words = {}
 | |
| 		for word in remote:gmatch("%w+") do
 | |
| 			table.insert(words, word)
 | |
| 		end
 | |
| 		local match = words[1]:match("/github.com[:/]neovim/neovim/")
 | |
| 		if fork == "fork" then
 | |
| 			match = not match
 | |
| 		end
 | |
| 		if match and words[3] == "(fetch)" then
 | |
| 			git_remote = words[0]
 | |
| 			break
 | |
| 		end
 | |
| 	end
 | |
| 	if git_remote == "" then
 | |
| 		git_remote = "origin"
 | |
| 	end
 | |
| 	return git_remote
 | |
| end
 | |
| 
 | |
| local function create_pr(pr_title, pr_body)
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	local push_first = true
 | |
| 
 | |
| 	local checked_out_branch = run({ "git", "rev-parse", "--abbrev-ref", "HEAD" })
 | |
| 	if push_first then
 | |
| 		local push_remote = run({ "git", "config", "--get", "branch." .. checked_out_branch .. ".pushRemote" })
 | |
| 		if push_remote == nil then
 | |
| 			push_remote = run({ "git", "config", "--get", "remote.pushDefault" })
 | |
| 			if push_remote == nil then
 | |
| 				push_remote = run({ "git", "config", "--get", "branch." .. checked_out_branch .. ".remote" })
 | |
| 				if push_remote == nil or push_remote == find_git_remote(nil) then
 | |
| 					push_remote = find_git_remote("fork")
 | |
| 				end
 | |
| 			end
 | |
| 		end
 | |
| 
 | |
| 		p("Pushing to " .. push_remote .. "/" .. checked_out_branch)
 | |
| 		run_die({ "git", "push", push_remote, checked_out_branch }, "Git failed to push")
 | |
| 	end
 | |
| 
 | |
| 	local pr_url = gh_pr(pr_title, pr_body)
 | |
| 	p("\nCreated PR: " .. pr_url .. "\n")
 | |
| end
 | |
| 
 | |
| function M.commit(dependency_name, commit)
 | |
| 	local dependency = get_dependency(dependency_name)
 | |
| 	verify_cmakelists_committed()
 | |
| 	local commit_sha = get_gh_commit_sha(dependency.repo, commit)
 | |
| 	if commit_sha ~= commit then
 | |
| 		p("Not a commit: " .. commit .. ". Did you mean version?")
 | |
| 		die()
 | |
| 	end
 | |
| 	local archive = get_archive_info(dependency.repo, commit)
 | |
| 	if dependency_name == "Luv" then
 | |
| 		warn_luv_symbol()
 | |
| 	end
 | |
| 	update_cmakelists(dependency, archive, short_commit(commit))
 | |
| end
 | |
| 
 | |
| function M.version(dependency_name, version)
 | |
| 	local dependency = get_dependency(dependency_name)
 | |
| 	verify_cmakelists_committed()
 | |
| 	local commit_sha = get_gh_commit_sha(dependency.repo, version)
 | |
| 	if commit_sha == version then
 | |
| 		p("Not a version: " .. version .. ". Did you mean commit?")
 | |
| 		die()
 | |
| 	end
 | |
| 	local archive = get_archive_info(dependency.repo, version)
 | |
| 	if dependency_name == "Luv" then
 | |
| 		write_cmakelists_line(dependency.symbol, "VERSION", version)
 | |
| 	end
 | |
| 	update_cmakelists(dependency, archive, version)
 | |
| end
 | |
| 
 | |
| function M.head(dependency_name)
 | |
| 	local dependency = get_dependency(dependency_name)
 | |
| 	verify_cmakelists_committed()
 | |
| 	local commit_sha = get_gh_commit_sha(dependency.repo, "HEAD")
 | |
| 	local archive = get_archive_info(dependency.repo, commit_sha)
 | |
| 	if dependency_name == "Luv" then
 | |
| 		warn_luv_symbol()
 | |
| 	end
 | |
| 	update_cmakelists(dependency, archive, "HEAD - " .. short_commit(commit_sha))
 | |
| end
 | |
| 
 | |
| function M.create_branch(dep)
 | |
| 	explicit_create_branch(dep)
 | |
| end
 | |
| 
 | |
| function M.submit_pr()
 | |
| 	require_executable("git")
 | |
| 
 | |
| 	verify_branch("deps")
 | |
| 
 | |
| 	local nvim_remote = find_git_remote(nil)
 | |
| 	local relevant_commit = run_die({
 | |
| 		"git",
 | |
| 		"log",
 | |
| 		"--grep=" .. commit_prefix,
 | |
| 		"--reverse",
 | |
| 		"--format='%s'",
 | |
| 		nvim_remote .. "/master..HEAD",
 | |
| 		"-1",
 | |
| 	}, "Failed to fetch commits")
 | |
| 
 | |
| 	local pr_title
 | |
| 	local pr_body
 | |
| 
 | |
| 	if relevant_commit == "" then
 | |
| 		pr_title = commit_prefix .. "bump some dependencies"
 | |
| 		pr_body = "bump some dependencies"
 | |
| 	else
 | |
| 		relevant_commit = relevant_commit:gsub("'", "")
 | |
| 		pr_title = relevant_commit
 | |
| 		pr_body = relevant_commit:gsub(commit_prefix:gsub("%(", "%%("):gsub("%)", "%%)"), "")
 | |
| 	end
 | |
| 	pr_body = pr_body .. "\n\n(add explanations if needed)"
 | |
| 	p(pr_title .. "\n" .. pr_body .. "\n")
 | |
| 	create_pr(pr_title, pr_body)
 | |
| end
 | |
| 
 | |
| return M
 |