aboutsummaryrefslogtreecommitdiffstats
path: root/lua/mason-core/installer/managers/common.lua
blob: c730a3aa9d1c5fe0ab6f4d79c8df819bfde8a597 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
local Result = require "mason-core.result"
local _ = require "mason-core.functional"
local a = require "mason-core.async"
local async_uv = require "mason-core.async.uv"
local installer = require "mason-core.installer"
local log = require "mason-core.log"
local platform = require "mason-core.platform"
local powershell = require "mason-core.managers.powershell"
local std = require "mason-core.installer.managers.std"

local M = {}

---@class DownloadItem
---@field download_url string
---@field out_file string

---@class FileDownloadSpec
---@field file string | string[]

local get_source_file = _.compose(_.head, _.split ":")
local get_outfile = _.compose(_.last, _.split ":")

---Normalizes file paths from e.g. "file:out-dir/" to "out-dir/file".
---@param file string
local function normalize_file_path(file)
    local source_file = get_source_file(file)
    local new_path = get_outfile(file)

    -- a dir expression (e.g. "libexec/")
    if _.matches("/$", new_path) then
        return new_path .. source_file
    end
    return new_path
end

---@generic T : FileDownloadSpec
---@type fun(download: T): T
M.normalize_files = _.evolve {
    file = _.cond {
        { _.is "string", normalize_file_path },
        { _.T, _.map(normalize_file_path) },
    },
}

---@param download FileDownloadSpec
---@param url_generator fun(file: string): string
---@return DownloadItem[]
function M.parse_downloads(download, url_generator)
    local files = download.file
    if type(files) == "string" then
        files = { files }
    end

    return _.map(function(file)
        local source_file = get_source_file(file)
        local out_file = normalize_file_path(file)
        return {
            download_url = url_generator(source_file),
            out_file = out_file,
        }
    end, files)
end

---@async
---@param ctx InstallContext
---@param downloads DownloadItem[]
---@nodiscard
function M.download_files(ctx, downloads)
    return Result.try(function(try)
        for __, download in ipairs(downloads) do
            a.scheduler()
            local out_dir = vim.fn.fnamemodify(download.out_file, ":h")
            local out_file = vim.fn.fnamemodify(download.out_file, ":t")
            if out_dir ~= "." then
                try(Result.pcall(function()
                    ctx.fs:mkdirp(out_dir)
                end))
            end
            try(ctx:chdir(out_dir, function()
                return Result.try(function(try)
                    try(std.download_file(download.download_url, out_file))
                    try(std.unpack(out_file))
                end)
            end))
        end
    end)
end

---@class BuildInstruction
---@field target? Platform | Platform[]
---@field run string
---@field staged? boolean
---@field env? table<string, string>

---@async
---@param build BuildInstruction
---@return Result
---@nodiscard
function M.run_build_instruction(build)
    log.fmt_debug("build: run %s", build)
    local ctx = installer.context()
    if build.staged == false then
        ctx:promote_cwd()
    end
    return platform.when {
        unix = function()
            return ctx.spawn.bash {
                on_spawn = a.scope(function(_, stdio)
                    local stdin = stdio[1]
                    async_uv.write(stdin, "set -euxo pipefail;\n")
                    async_uv.write(stdin, build.run)
                    async_uv.shutdown(stdin)
                    async_uv.close(stdin)
                end),
                env = build.env,
            }
        end,
        win = function()
            return powershell.command(build.run, {
                env = build.env,
            }, ctx.spawn)
        end,
    }
end

return M