mypath.lua

Fri, 24 Jan 2025 13:27:45 +0100

author
Tuomo Valkonen <tuomov@iki.fi>
date
Fri, 24 Jan 2025 13:27:45 +0100
changeset 44
5700ebe9277b
parent 35
2f927eae429b
permissions
-rw-r--r--

Fix directory creation, as lfs doesn't do all leaves like the old approach.


--@module path
local path={}

local tuple=require("tuple")
local lfs=require("lfs")
local err=require("err")
local config=require("config")

local sep=config.dirsep

function path.join(p, f)
    if p=="" then
        return f
    else
        return p..sep..f
    end
end

function path.dirbasename(p)
    local d, b = string.match(p,
        "^(.*)"..sep.."+([^"..sep.."]+)"..sep.."*$")
    if not b then
        return "", p
    else
        return d, b
    end
end

function path.dirname(p)
    return tuple.fst(path.dirbasename(p))
end

function path.dirname_slash(p)
    local dn=dirname(p)
    if dn=="" then
        return dn
    else
        return dn..sep
    end
end

function path.basename(p)
    return tuple.snd(dirbasename(p))
end

function path.rmext(p)
    return string.gsub(p, "%.[^.]*$", "")
end

-- would rather do this as an iterator, but can't
-- coroutine.yield from gsub handler
function path.split(p)
    local t={}
    local head=""
    local s, p2 = string.match(p, "^("..sep.."+)(.*)$")
    if s then
        table.insert(t, "/.") -- root
        p=p2
    end
    string.gsub(p, "([^"..sep.."]+)", --..sep.."+", 
            function(d) table.insert(t, d); end)
    return t
end

--[[
function path.parts(p)
    local s={i=1, t=split(p)}
    local function g(s)
        local v=s.t[s.i]
        s.i=s.i+1
        return v
    end
    return g, s
end

function path.makepath(p, head)
    head=head or ""
    for d in parts(p) do
        head=head..d
        local a=lfs.attributes(head)
        if not a then
            local success, e = lfs.mkdir(head)
            if not success then
                err.file(head, e)
            end
        elseif a.mode~="directory" then
            err.file(head, "not a directory")
        end
        head=head..sep
    end
end
]]

function path.simplify(p)
    local capts={}
    local start=string.match(p, "^(/)")
    string.gsub(p, "([^/]+)",
    function(e)
        if e==".." and #capts > 1 then
            capts[#capts]=nil
        elseif e~="." then
            table.insert(capts, e)
        end  
    end)
    return (start or "")..table.concat(capts, "/")
end

function path.to_root(p)
    return string.rep("../",#(path.split(p))-1)
end

return path

mercurial