feat: initial commit

This commit is contained in:
2025-09-19 02:07:23 +03:00
commit 1600756a59
12 changed files with 589 additions and 0 deletions

13
.github/FUNDING.yml vendored Normal file
View File

@@ -0,0 +1,13 @@
# These are supported funding model platforms
github: chenasraf
patreon: # Replace with a single Patreon username
open_collective: # Replace with a single Open Collective username
ko_fi: casraf
tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel
community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry
liberapay: # Replace with a single Liberapay username
issuehunt: # Replace with a single IssueHunt username
otechie: # Replace with a single Otechie username
custom:
- "https://www.paypal.com/cgi-bin/webscr?cmd=_donations&business=TSH3C3ABGQM22&currency_code=ILS&source=url"

View File

@@ -0,0 +1,42 @@
name: Manual Homebrew Release
on:
workflow_dispatch:
permissions:
contents: read
jobs:
release-homebrew:
name: Trigger Homebrew Formula Update
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Get latest tag
id: latest
run: |
tag=$(gh release view --json tagName -q .tagName)
echo "Latest release tag: $tag"
echo "tag=$tag" >> "$GITHUB_OUTPUT"
env:
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
- name: Send dispatch to homebrew-tap
env:
GH_TOKEN: ${{ secrets.REPO_DISPATCH_PAT }}
run: |
tag="${{ steps.latest.outputs.tag }}"
repo="${{ github.event.repository.name }}"
data="{\"event_type\":\"trigger-from-release\",\"client_payload\":{\"tag\":\"$tag\",\"repo\":\"$repo\"}}"
echo "Dispatching tag $tag from $repo"
echo "Data: $data"
curl -X POST \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer $GH_TOKEN" \
https://api.github.com/repos/chenasraf/homebrew-tap/dispatches \
-d "$data"
echo "Dispatched tag $tag from $repo"
echo "Created job on https://github.com/chenasraf/homebrew-tap/actions"

124
.github/workflows/release.yml vendored Normal file
View File

@@ -0,0 +1,124 @@
name: Release
on:
push:
branches: ["**"]
pull_request:
branches: ["**"]
permissions:
contents: write
pull-requests: write
jobs:
test:
name: Test
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Set up Go
uses: actions/setup-go@v4
with:
go-version: '1.23'
- name: Test
run: go test -v ./...
generate:
name: Build for ${{ matrix.platform }}
runs-on: ubuntu-latest
strategy:
matrix:
include:
- platform: linux/amd64
label: linux-amd64
- platform: darwin/amd64
label: darwin-amd64
- platform: darwin/arm64
label: darwin-arm64
- platform: windows/amd64
label: windows-amd64
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Build for ${{ matrix.label }}
uses: chenasraf/go-cross-build@v1
with:
platforms: ${{ matrix.platform }}
package: ''
name: 'vstask'
compress: 'true'
dest: dist
- name: Upload build
uses: actions/upload-artifact@v4
with:
name: "dist-${{ matrix.label }}"
path: dist
release-please:
name: Release
if: github.ref == 'refs/heads/master'
runs-on: ubuntu-latest
outputs:
release_created: ${{ steps.release.outputs.release_created }}
tag_name: ${{ steps.release.outputs.tag_name }}
needs:
- test
- generate
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Download all builds
uses: actions/download-artifact@v4
with:
path: dist
- name: Verify Release Artifacts
run: |
ls -la dist
for i in "linux-amd64" "darwin-amd64" "windows-amd64" "darwin-arm64"; do
if [[ ! -f ./dist/dist-$i/vstask-$i.tar.gz ]]; then
echo "File not found: ./dist/dist-$i/vstask-$i.tar.gz"
exit 1
fi
done
- name: Run Release Please
uses: googleapis/release-please-action@v4
id: release
with:
release-type: simple
- name: Upload Release Artifacts
if: ${{ steps.release.outputs.release_created }}
env:
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
run: |
for i in "linux-amd64" "darwin-amd64" "darwin-arm64" "windows-amd64"; do
gh release upload "${{ steps.release.outputs.tag_name }}" "./dist/dist-$i/vstask-$i.tar.gz"
done
release-homebrew:
name: Homebrew Release
needs: [release-please]
if: ${{ needs.release-please.outputs.release_created }}
runs-on: ubuntu-latest
steps:
- name: Send dispatch to homebrew-tap
env:
GH_TOKEN: ${{ secrets.REPO_DISPATCH_PAT }}
run: |
repo="${{ github.event.repository.name }}"
tag="${{ needs.release-please.outputs.tag_name }}"
data="{\"event_type\":\"trigger-from-release\",\"client_payload\":{\"tag\":\"$tag\",\"repo\":\"$repo\"}}"
echo "Dispatching tag $tag from $repo"
echo "Data: $data"
curl -X POST \
-H "Accept: application/vnd.github+json" \
-H "Authorization: Bearer $GH_TOKEN" \
https://api.github.com/repos/chenasraf/homebrew-tap/dispatches \
-d "$data"
echo "Dispatched tag $tag from $repo"
echo "Created job on https://github.com/chenasraf/homebrew-tap/actions"

44
.github/workflows/test.yml vendored Normal file
View File

@@ -0,0 +1,44 @@
name: Test
on:
push:
branches:
- develop
pull_request:
branches:
- master
jobs:
build:
name: Build & Test
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Set up Go
uses: actions/setup-go@v4
with:
go-version: '1.23'
- name: Build
run: go build -v
- name: Test
run: go test -v ./...
- name: Create dist/ dir
run: mkdir dist
- name: Generate build files
uses: chenasraf/go-cross-build@v1
with:
platforms: 'linux/amd64, darwin/amd64, windows/amd64' # , darwin/arm64' # '
package: ''
name: 'vstask'
compress: 'true'
dest: 'dist'
- name: Upload builds
uses: actions/upload-artifact@v4
with:
name: dist
path: dist

24
go.mod Normal file
View File

@@ -0,0 +1,24 @@
module github.com/chenasraf/vstask
go 1.24
toolchain go1.24.7
require (
github.com/ktr0731/go-fuzzyfinder v0.9.0
github.com/samber/lo v1.51.0
)
require (
github.com/gdamore/encoding v1.0.1 // indirect
github.com/gdamore/tcell/v2 v2.6.0 // indirect
github.com/ktr0731/go-ansisgr v0.1.0 // indirect
github.com/lucasb-eyer/go-colorful v1.2.0 // indirect
github.com/mattn/go-runewidth v0.0.16 // indirect
github.com/nsf/termbox-go v1.1.1 // indirect
github.com/pkg/errors v0.9.1 // indirect
github.com/rivo/uniseg v0.4.7 // indirect
golang.org/x/sys v0.32.0 // indirect
golang.org/x/term v0.31.0 // indirect
golang.org/x/text v0.24.0 // indirect
)

66
go.sum Normal file
View File

@@ -0,0 +1,66 @@
github.com/gdamore/encoding v1.0.0/go.mod h1:alR0ol34c49FCSBLjhosxzcPHQbf2trDkoo5dl+VrEg=
github.com/gdamore/encoding v1.0.1 h1:YzKZckdBL6jVt2Gc+5p82qhrGiqMdG/eNs6Wy0u3Uhw=
github.com/gdamore/encoding v1.0.1/go.mod h1:0Z0cMFinngz9kS1QfMjCP8TY7em3bZYeeklsSDPivEo=
github.com/gdamore/tcell/v2 v2.6.0 h1:OKbluoP9VYmJwZwq/iLb4BxwKcwGthaa1YNBJIyCySg=
github.com/gdamore/tcell/v2 v2.6.0/go.mod h1:be9omFATkdr0D9qewWW3d+MEvl5dha+Etb5y65J2H8Y=
github.com/google/go-cmp v0.7.0 h1:wk8382ETsv4JYUZwIsn6YpYiWiBsYLSJiTsyBybVuN8=
github.com/google/go-cmp v0.7.0/go.mod h1:pXiqmnSA92OHEEa9HXL2W4E7lf9JzCmGVUdgjX3N/iU=
github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0=
github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
github.com/ktr0731/go-ansisgr v0.1.0 h1:fbuupput8739hQbEmZn1cEKjqQFwtCCZNznnF6ANo5w=
github.com/ktr0731/go-ansisgr v0.1.0/go.mod h1:G9lxwgBwH0iey0Dw5YQd7n6PmQTwTuTM/X5Sgm/UrzE=
github.com/ktr0731/go-fuzzyfinder v0.9.0 h1:JV8S118RABzRl3Lh/RsPhXReJWc2q0rbuipzXQH7L4c=
github.com/ktr0731/go-fuzzyfinder v0.9.0/go.mod h1:uybx+5PZFCgMCSDHJDQ9M3nNKx/vccPmGffsXPn2ad8=
github.com/lucasb-eyer/go-colorful v1.2.0 h1:1nnpGOrhyZZuNyfu1QjKiUICQ74+3FNCN69Aj6K7nkY=
github.com/lucasb-eyer/go-colorful v1.2.0/go.mod h1:R4dSotOR9KMtayYi1e77YzuveK+i7ruzyGqttikkLy0=
github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI=
github.com/mattn/go-runewidth v0.0.14/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
github.com/mattn/go-runewidth v0.0.16 h1:E5ScNMtiwvlvB5paMFdw9p4kSQzbXFikJ5SQO6TULQc=
github.com/mattn/go-runewidth v0.0.16/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
github.com/nsf/termbox-go v1.1.1 h1:nksUPLCb73Q++DwbYUBEglYBRPZyoXJdrj5L+TkjyZY=
github.com/nsf/termbox-go v1.1.1/go.mod h1:T0cTdVuOwf7pHQNtfhnEbzHbcNyCEcVU4YPpouCbVxo=
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
github.com/rivo/uniseg v0.4.3/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
github.com/rivo/uniseg v0.4.7 h1:WUdvkW8uEhrYfLC4ZzdpI2ztxP1I582+49Oc5Mq64VQ=
github.com/rivo/uniseg v0.4.7/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
github.com/samber/lo v1.51.0 h1:kysRYLbHy/MB7kQZf5DSN50JHmMsNEdeY24VzJFu7wI=
github.com/samber/lo v1.51.0/go.mod h1:4+MXEGsJzbKGaUEQFKBq2xtfuznW9oz/WrgyzMzRoM0=
github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc=
golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4=
golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs=
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg=
golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c=
golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs=
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
golang.org/x/sys v0.32.0 h1:s77OFDvIQeibCmezSnk/q6iAfkdiQaJi4VzroCFrN20=
golang.org/x/sys v0.32.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo=
golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8=
golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k=
golang.org/x/term v0.31.0 h1:erwDkOK1Msy6offm1mOgvspSkslFnIGsFnxOKoufg3o=
golang.org/x/term v0.31.0/go.mod h1:R4BeIy7D95HzImkxGkTW1UQTtP54tio2RyHz7PwK0aw=
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
golang.org/x/text v0.14.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU=
golang.org/x/text v0.24.0 h1:dd5Bzh4yt5KYA8f9CJHCP4FB4D51c2c6JvN37xJJkJ0=
golang.org/x/text v0.24.0/go.mod h1:L8rBsPeo2pSS+xqN0d5u2ikmjtmoJbDBT1b7nHvFCdU=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc=
golang.org/x/tools v0.6.0/go.mod h1:Xwgl3UAJ/d3gWutnCtw505GrjyAbvKui8lOU390QaIU=
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=

37
main.go Normal file
View File

@@ -0,0 +1,37 @@
package main
import (
"errors"
"os"
"github.com/chenasraf/vstask/runner"
"github.com/chenasraf/vstask/tasks"
"github.com/samber/lo"
)
func main() {
args := os.Args[1:]
if len(args) > 0 {
if args[0] == "--help" || args[0] == "-h" {
// PrintHelp()
os.Exit(0)
}
taskList, err := tasks.GetTasks()
if err != nil {
panic(err)
}
task, found := lo.Find(taskList, func(t tasks.Task) bool {
return t.Label == args[0]
})
if !found {
panic(errors.New("task not found: " + args[0]))
}
runner.RunTask(task)
os.Exit(0)
}
selected, err := tasks.PromptForTask()
if err != nil {
panic(err)
}
runner.RunTask(selected)
}

12
runner/run_task.go Normal file
View File

@@ -0,0 +1,12 @@
package runner
import (
"fmt"
"github.com/chenasraf/vstask/tasks"
)
func RunTask(task tasks.Task) error {
fmt.Println("Running task:", task.Label)
return nil
}

88
tasks/task.go Normal file
View File

@@ -0,0 +1,88 @@
package tasks
import "encoding/json"
// File is the root of .vscode/tasks.json
type File struct {
Version string `json:"version,omitempty"`
Tasks []Task `json:"tasks,omitempty"`
Inputs []any `json:"inputs,omitempty"` // keep flexible; inputs can vary
}
// Task represents a single VS Code task (2.0.0 schema).
type Task struct {
// Required
Label string `json:"label,omitempty"`
Type string `json:"type,omitempty"` // e.g. "shell" | "process" | extension task type
// Command & args
Command string `json:"command,omitempty"`
Args []string `json:"args,omitempty"`
Windows *PlatformTask `json:"windows,omitempty"`
Osx *PlatformTask `json:"osx,omitempty"`
Linux *PlatformTask `json:"linux,omitempty"`
Options *Options `json:"options,omitempty"`
Presentation *Presentation `json:"presentation,omitempty"` // aka presentationOptions in older docs
RunOptions *RunOptions `json:"runOptions,omitempty"`
// Dependencies & grouping
DependsOn json.RawMessage `json:"dependsOn,omitempty"` // string | string[] | { tasks: string[] } via extension
DependsOrder string `json:"dependsOrder,omitempty"` // "sequence" | "parallel"
Group *Group `json:"group,omitempty"` // "build" | "test" | { "kind": "...", "isDefault": bool }
// Problem matchers (string | string[] | object | object[])
ProblemMatcher json.RawMessage `json:"problemMatcher,omitempty"`
// Misc
Detail string `json:"detail,omitempty"` // shown in the UI
}
// PlatformTask lets you override per-OS parts of the task.
// VS Code allows most of the same fields as the base task here.
type PlatformTask struct {
Command string `json:"command,omitempty"`
Args []string `json:"args,omitempty"`
Options *Options `json:"options,omitempty"`
Presentation *Presentation `json:"presentation,omitempty"`
}
// Options corresponds to "options" in tasks.json.
type Options struct {
Cwd string `json:"cwd,omitempty"`
Env map[string]string `json:"env,omitempty"`
Shell *ShellOptions `json:"shell,omitempty"`
// Windows/Osx/Linux sub-options also exist - TODO add if needed
}
// ShellOptions controls the shell used by "type": "shell" tasks.
type ShellOptions struct {
Executable string `json:"executable,omitempty"`
Args []string `json:"args,omitempty"`
// Quote settings exist too; add if needed (e.g. "quoting": "escape")
}
// Presentation controls terminal/UI behavior.
type Presentation struct {
Reveal string `json:"reveal,omitempty"` // "always" | "silent" | "never"
Panel string `json:"panel,omitempty"` // "shared" | "dedicated" | "new"
Focus bool `json:"focus,omitempty"`
Echo bool `json:"echo,omitempty"`
ShowReuseMessage bool `json:"showReuseMessage,omitempty"`
Clear bool `json:"clear,omitempty"`
// "RevealProblems": "onProblem"|"onProblemDependingOnSeverity" may exist in newer versions
}
// Group can be a simple string ("build"/"test") or an object.
type Group struct {
// If Kind is empty but you need simple groups, you can instead store "build" or "test"
// directly in JSON by using Raw below.
Kind string `json:"kind,omitempty"` // e.g. "build", "test"
IsDefault bool `json:"isDefault,omitempty"` // marks default task for the group
}
// RunOptions adds scheduling behavior (VS Code 1.59+).
type RunOptions struct {
RunOn string `json:"runOn,omitempty"` // "default" | "folderOpen"
ReevaluateOnRun bool `json:"reevaluateOnRun,omitempty"` // true: re-resolve variables each run
InstanceLimit int `json:"instanceLimit,omitempty"` // max parallel instances
}

42
tasks/task_list.go Normal file
View File

@@ -0,0 +1,42 @@
package tasks
import (
"encoding/json"
"errors"
"os"
"path"
"github.com/chenasraf/vstask/utils"
)
func GetTasks() ([]Task, error) {
projectRoot, err := utils.FindProjectRoot()
if err != nil {
return []Task{}, err
}
tasksPath := path.Join(projectRoot, utils.VSCODE_DIR, utils.TASKS_JSON)
if !utils.FileExists(tasksPath) {
return []Task{}, errors.New("tasks.json not found")
}
return LoadTasksFile(tasksPath)
}
func LoadTasksFile(tasksPath string) ([]Task, error) {
data, err := os.ReadFile(tasksPath)
if err != nil {
return nil, err
}
var file struct {
Version string `json:"version"`
Tasks []Task `json:"tasks"`
}
if err := json.Unmarshal(data, &file); err != nil {
return nil, err
}
return file.Tasks, nil
}

28
tasks/task_prompt.go Normal file
View File

@@ -0,0 +1,28 @@
package tasks
import "github.com/ktr0731/go-fuzzyfinder"
func PromptForTask() (Task, error) {
taskList, err := GetTasks()
if err != nil {
return Task{}, err
}
idx, err := fuzzyfinder.Find(
taskList,
func(i int) string {
return taskList[i].Label
},
fuzzyfinder.WithPreviewWindow(func(i, w, h int) string {
if i == -1 {
return "No task selected"
}
return taskList[i].Detail
}))
if err != nil {
return Task{}, err
}
return taskList[idx], nil
}

69
utils/directory.go Normal file
View File

@@ -0,0 +1,69 @@
package utils
import (
"errors"
"os"
"path"
)
const (
VSCODE_DIR = ".vscode"
TASKS_JSON = "tasks.json"
)
func FindProjectRoot() (string, error) {
cwd, err := os.Getwd()
if err != nil {
return "", err
}
if cwd == "/" || cwd == "\\" || len(cwd) <= 2 {
return "", errors.New("no project root found")
}
return FindProjectRootFrom(cwd)
}
func FindProjectRootFrom(p string) (string, error) {
vscodePath := path.Join(p, VSCODE_DIR)
if DirExists(vscodePath) {
return ".", nil
}
parent, err := getParentDir(p)
if err != nil {
return "", err
}
return FindProjectRootFrom(parent)
}
func getParentDir(p string) (string, error) {
if p == "/" || p == "\\" || len(p) <= 2 {
return "", errors.New("no parent directory")
}
return path.Dir(p), nil
}
// FileExists reports whether path exists and is a regular file.
func FileExists(p string) bool {
info, err := os.Stat(p)
if err != nil {
return false
// return !errors.Is(err, fs.ErrNotExist)
}
return info.Mode().IsRegular()
}
// DirExists reports whether path exists and is a directory.
func DirExists(p string) bool {
info, err := os.Stat(p)
if err != nil {
return false
// return !errors.Is(err, fs.ErrNotExist)
}
return info.IsDir()
}
// PathExists reports whether any filesystem object exists at path (file, dir, symlink target, etc.).
func PathExists(p string) bool {
_, err := os.Stat(p)
return err == nil
// return err == nil || !errors.Is(err, fs.ErrNotExist)
}