fix(lib): binary descriptor fallback not matching

* Fixes a bug which caused binary parsing fallbacks to not work as
  expected, due to a field unexpectedly being `None`.
* Fixed lint error to do with warn being potentially none by defining
  our own warn function instead. Also updated fallback binary parsing
  fallback warn message to be clearer.
* Fixed decompression error for non-supported artifacts by only
  decompressing those we support.
This commit is contained in:
Erica Marigold 2024-11-25 08:10:05 +00:00
parent 9dd820d804
commit b6f75cbda9
5 changed files with 69 additions and 37 deletions

View file

@ -64,7 +64,11 @@ local decompress: { [CompressionFormat]: (compressed: buffer) -> Result<pathfs.A
-- FIXME: remove unknown usage -- FIXME: remove unknown usage
:withStdioStrategy({ :withStdioStrategy({
stdout = Option.Some("pipe" :: CommandBuilder.StdioStrategy) :: Option<unknown>, stdout = Option.Some("pipe" :: CommandBuilder.StdioStrategy) :: Option<unknown>,
stderr = Option.Some(if process.env.PESDE_LOG == "debug" then "forward" else "pipe" :: CommandBuilder.StdioStrategy) :: Option<unknown>, stderr = Option.Some(
if process.env.PESDE_LOG == "debug"
then "forward"
else "pipe" :: CommandBuilder.StdioStrategy
) :: Option<unknown>,
} :: CommandBuilder.IoStrategyMapping) } :: CommandBuilder.IoStrategyMapping)
:intoChildProcess() :intoChildProcess()
@ -91,7 +95,4 @@ local decompress: { [CompressionFormat]: (compressed: buffer) -> Result<pathfs.A
-- TODO: Other formats -- TODO: Other formats
} }
-- local path = "pesde-0.5.0-rc.7-windows-x86_64.zip"
-- print(decompress[detectFormat(path):unwrap()](buffer.fromstring(pathfs.readFile(path))))
return { decompress = decompress, detectFormat = detectFormat } return { decompress = decompress, detectFormat = detectFormat }

View file

@ -47,22 +47,30 @@ export type GithubReleases = {
} }
} }
local WARN_PREFIX = `{stdio.color("yellow")}{stdio.style("bold")}warn{stdio.color("reset")}:`
local function warn(...)
stdio.ewrite(`{WARN_PREFIX} {stdio.format(...)}\n`)
end
local function downloadAndDecompress(asset: { local function downloadAndDecompress(asset: {
name: string, name: string,
browser_download_url: string, browser_download_url: string,
size: number, size: number,
content_type: string, content_type: string,
}): pathfs.Path }): Option<pathfs.Path>
-- TODO: Small optimization by first detecting that its a valid format we support
-- before downloading the file
local contentsResp = net.request(asset.browser_download_url) local contentsResp = net.request(asset.browser_download_url)
if not contentsResp.ok then if not contentsResp.ok then
return error(`Failed to download asset {asset.name}: HTTP Code {contentsResp.statusCode}`) return error(`Failed to download asset {asset.name}: HTTP Code {contentsResp.statusCode}`)
end end
local decompressedPath = compression.decompress return compression
[compression.detectFormat(asset.name):unwrap()](buffer.fromstring(contentsResp.body)) .detectFormat(asset.name)
:unwrap() :: pathfs.Path :map(function(format: compression.CompressionFormat)
return compression.decompress[format](buffer.fromstring(contentsResp.body)):unwrap() :: pathfs.Path
return decompressedPath end) :: Option<pathfs.Path>
end end
local function toolAliasOrDefault(tool: ToolId): string local function toolAliasOrDefault(tool: ToolId): string
@ -125,20 +133,28 @@ function installTool(tool: ToolId, installPath: pathfs.Path)
local aliasPath = pathfs.Path.from(toolAlias):withExtension(if currentDesc.os == "windows" then "exe" else "") local aliasPath = pathfs.Path.from(toolAlias):withExtension(if currentDesc.os == "windows" then "exe" else "")
for _, asset in assets do for _, asset in assets do
local desc = PlatformDescriptor.fromString(asset.name):unwrap() local desc = PlatformDescriptor.fromString(asset.name)
if eq(currentDesc, desc) then local descWithArch = desc:map(function(inner: PlatformDescriptor.PlatformDescriptor)
-- FIXME: unknown usage
return inner
end)
if descWithArch:isOk() and eq(currentDesc, descWithArch:unwrap()) then
matchingAsset = asset matchingAsset = asset
break
end end
end end
local binaryPath: pathfs.Path local binaryPath: pathfs.Path
if matchingAsset == nil then if matchingAsset == nil then
stdio.ewrite("No matching asset found, testing all binaries") warn("Pesde could not find a matching binary for your system")
warn("Will now attempt to download all binaries and find a matching one")
for _, asset in assets do for _, asset in assets do
local decompressedPath = downloadAndDecompress(asset) local decompressedPath = downloadAndDecompress(asset)
if decompressedPath:isSome() then
for _, file in pathfs.readDir(decompressedPath) do local path = decompressedPath:unwrap()
local filePath = decompressedPath:join(file) for _, file in pathfs.readDir(path) do
local filePath = path:join(file)
local nativeDesc = PlatformDescriptor.fromExecutable(filePath:toString()):unwrap() local nativeDesc = PlatformDescriptor.fromExecutable(filePath:toString()):unwrap()
if eq(currentDesc, nativeDesc) then if eq(currentDesc, nativeDesc) then
@ -147,8 +163,9 @@ function installTool(tool: ToolId, installPath: pathfs.Path)
end end
end end
end end
end
else else
local decompressedPath = downloadAndDecompress(matchingAsset) local decompressedPath = downloadAndDecompress(matchingAsset):unwrap()
binaryPath = decompressedPath:join(aliasPath) binaryPath = decompressedPath:join(aliasPath)
if not pathfs.isFile(binaryPath) then if not pathfs.isFile(binaryPath) then
error(`No matching binary found in {decompressedPath}`) error(`No matching binary found in {decompressedPath}`)

View file

@ -14,11 +14,15 @@ type Option<T> = types.Option<T>
type Result<T, E> = types.Result<T, E> type Result<T, E> = types.Result<T, E>
local PlatformDescriptor = {} local PlatformDescriptor = {}
type PlatformDescriptor = { export type PlatformDescriptor = {
os: process.OS, os: process.OS,
arch: Option<arch.Arch>, arch: Option<arch.Arch>,
toolchain: Option<toolchain.Toolchain>, toolchain: Option<toolchain.Toolchain>,
} }
type ExecutableDetectionResult = {
os: Option<process.OS>,
arch: Option<arch.Arch>,
}
function PlatformDescriptor.currentSystem(): PlatformDescriptor function PlatformDescriptor.currentSystem(): PlatformDescriptor
return { return {
@ -43,12 +47,20 @@ end
function PlatformDescriptor.fromExecutable(path: string): result.PlatformResult<PlatformDescriptor> function PlatformDescriptor.fromExecutable(path: string): result.PlatformResult<PlatformDescriptor>
local binaryContents = fs.readFile(path) local binaryContents = fs.readFile(path)
local detected = local detected = Option.from(detectFromExecutable(buffer.fromstring(binaryContents)))
Option.from(detectFromExecutable(buffer.fromstring(binaryContents))) :: Option<detectFromExecutable.ExecutableDetectionResult> :map(function(inner: detectFromExecutable.ExecutableDetectionResult): ExecutableDetectionResult
local platformDesc = detected:map(function(inner: detectFromExecutable.ExecutableDetectionResult) return {
local innerClone: PlatformDescriptor = table.clone(inner) :: any os = Option.from(inner.os) :: Option<process.OS>,
innerClone.toolchain = Option.None :: Option<toolchain.Toolchain> arch = Option.from(inner.arch) :: Option<arch.Arch>,
return innerClone }
end) :: Option<ExecutableDetectionResult>
local platformDesc = detected:map(function(inner: ExecutableDetectionResult): PlatformDescriptor
return {
os = inner.os:unwrap(),
arch = inner.arch :: Option<arch.Arch>,
toolchain = Option.None :: Option<toolchain.Toolchain>,
}
end) :: Option<PlatformDescriptor> end) :: Option<PlatformDescriptor>
return platformDesc:okOr( return platformDesc:okOr(

View file

@ -24,11 +24,13 @@ return function<T>(str: string, substrings: { [T]: { string } }, fullWords: { [T
local components = String.splitAtChar(lowercased, charWordSep) local components = String.splitAtChar(lowercased, charWordSep)
for _, component in components do for _, component in components do
for item, keywords in fullWords do for item, keywords in fullWords do
if table.find(keywords, component) then for _, keyword in keywords do
if string.find(component, keyword) then
return Option.Some(item) :: Option<T> return Option.Some(item) :: Option<T>
end end
end end
end end
end
return Option.None :: Option<T> return Option.None :: Option<T>
end end