forked from Elara6331/itd
		
	Compare commits
	
		
			24 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| f33b3d2b56 | |||
| 03f3968fe1 | |||
| dea92c6404 | |||
| 006f245c10 | |||
| d232340edd | |||
| c6458720e9 | |||
| 1e072a3540 | |||
| f639fef992 | |||
| 2d0db1dcf1 | |||
| 4efa4380c4 | |||
| fca64afbf3 | |||
| cf24c5ace8 | |||
| a25b2e3e62 | |||
| 2d0b64d92f | |||
| 5efafe9be7 | |||
| 271510d528 | |||
| 4d72a063b2 | |||
| 643245f16c | |||
| 6f87980d4b | |||
| 851f1975d6 | |||
| 5e66fe82ac | |||
| 645541e079 | |||
| 1012be6e5b | |||
| 5973290d6c | 
							
								
								
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -2,3 +2,4 @@ | |||||||
| /itd | /itd | ||||||
| /itgui | /itgui | ||||||
| /version.txt | /version.txt | ||||||
|  | dist/ | ||||||
|   | |||||||
							
								
								
									
										108
									
								
								.goreleaser.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										108
									
								
								.goreleaser.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,108 @@ | |||||||
|  | before: | ||||||
|  |   hooks: | ||||||
|  |     - go mod tidy | ||||||
|  | builds: | ||||||
|  |   - id: itd | ||||||
|  |     env: | ||||||
|  |       - CGO_ENABLED=0 | ||||||
|  |     binary: itd | ||||||
|  |     goos: | ||||||
|  |       - linux | ||||||
|  |     goarch: | ||||||
|  |       - 386 | ||||||
|  |       - amd64 | ||||||
|  |       - arm | ||||||
|  |       - arm64 | ||||||
|  |   - id: itctl | ||||||
|  |     env: | ||||||
|  |       - CGO_ENABLED=0 | ||||||
|  |     main: ./cmd/itctl | ||||||
|  |     binary: itctl | ||||||
|  |     goos: | ||||||
|  |       - linux | ||||||
|  |     goarch: | ||||||
|  |       - 386 | ||||||
|  |       - amd64 | ||||||
|  |       - arm | ||||||
|  |       - arm64 | ||||||
|  | archives: | ||||||
|  |   - replacements: | ||||||
|  |       386: i386 | ||||||
|  |       amd64: x86_64 | ||||||
|  |       arm64: aarch64 | ||||||
|  |     files: | ||||||
|  |       - LICENSE | ||||||
|  |       - README.md | ||||||
|  |       - itd.toml | ||||||
|  |       - itd.service | ||||||
|  | nfpms: | ||||||
|  |   - id: itd | ||||||
|  |     file_name_template: '{{.PackageName}}-{{.Version}}-{{.Os}}-{{.Arch}}' | ||||||
|  |     description: "Companion daemon for the InfiniTime firmware on the PineTime smartwatch" | ||||||
|  |     replacements: | ||||||
|  |       386: i386 | ||||||
|  |       amd64: x86_64 | ||||||
|  |       arm64: aarch64 | ||||||
|  |     homepage: 'https://gitea.arsenm.dev/Arsen6331/itd' | ||||||
|  |     maintainer: 'Arsen Musyaelyan <arsen@arsenm.dev>' | ||||||
|  |     license: GPLv3 | ||||||
|  |     formats: | ||||||
|  |       - apk | ||||||
|  |       - deb | ||||||
|  |       - rpm | ||||||
|  |     dependencies: | ||||||
|  |       - dbus | ||||||
|  |       - bluez | ||||||
|  |       - pulseaudio-utils | ||||||
|  |     contents: | ||||||
|  |       - src: itd.toml | ||||||
|  |         dst: /etc/itd.toml | ||||||
|  |         type: "config|noreplace" | ||||||
|  |       - src: itd.service | ||||||
|  |         dst: /usr/lib/systemd/user/itd.service | ||||||
|  | aurs: | ||||||
|  |   - name: itd-bin | ||||||
|  |     homepage: 'https://gitea.arsenm.dev/Arsen6331/itd' | ||||||
|  |     description: "Companion daemon for the InfiniTime firmware on the PineTime smartwatch" | ||||||
|  |     maintainers: | ||||||
|  |       - 'Arsen Musyaelyan <arsen@arsenm.dev>' | ||||||
|  |     license: GPLv3 | ||||||
|  |     private_key: '{{ .Env.AUR_KEY }}' | ||||||
|  |     git_url: 'ssh://aur@aur.archlinux.org/itd-bin.git' | ||||||
|  |     provides: | ||||||
|  |       - itd | ||||||
|  |       - itctl | ||||||
|  |     conflicts: | ||||||
|  |       - itd | ||||||
|  |       - itctl | ||||||
|  |     depends: | ||||||
|  |       - dbus | ||||||
|  |       - bluez | ||||||
|  |       - libpulse | ||||||
|  |     package: |- | ||||||
|  |       # binaries | ||||||
|  |       install -Dm755 "./itd" "${pkgdir}/usr/bin/itd" | ||||||
|  |       install -Dm755 "./itctl" "${pkgdir}/usr/bin/itctl" | ||||||
|  |  | ||||||
|  |       # service | ||||||
|  |       install -Dm644 "./itd.service" ${pkgdir}/usr/lib/systemd/user/itd.service | ||||||
|  |  | ||||||
|  |       # config | ||||||
|  |       install -Dm644 "./itd.toml" ${pkgdir}/etc/itd.toml | ||||||
|  |        | ||||||
|  |       # license | ||||||
|  |       install -Dm644 "./LICENSE" "${pkgdir}/usr/share/licenses/itd/LICENSE" | ||||||
|  | release: | ||||||
|  |   gitea: | ||||||
|  |     owner: Arsen6331 | ||||||
|  |     name: itd | ||||||
|  | gitea_urls: | ||||||
|  |   api: 'https://gitea.arsenm.dev/api/v1/' | ||||||
|  |   download: 'https://gitea.arsenm.dev' | ||||||
|  |   skip_tls_verify: false | ||||||
|  | checksum: | ||||||
|  |   name_template: 'checksums.txt' | ||||||
|  | snapshot: | ||||||
|  |   name_template: "{{ incpatch .Version }}-next" | ||||||
|  | changelog: | ||||||
|  |   sort: asc | ||||||
							
								
								
									
										48
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								README.md
									
									
									
									
									
								
							| @@ -3,9 +3,9 @@ | |||||||
|  |  | ||||||
| `itd` is a daemon that uses my infinitime [library](https://go.arsenm.dev/infinitime) to interact with the [PineTime](https://www.pine64.org/pinetime/) running [InfiniTime](https://infinitime.io). | `itd` is a daemon that uses my infinitime [library](https://go.arsenm.dev/infinitime) to interact with the [PineTime](https://www.pine64.org/pinetime/) running [InfiniTime](https://infinitime.io). | ||||||
|  |  | ||||||
| [](https://ci.appveyor.com/project/moussaelianarsen/itd) | [](https://ci.appveyor.com/project/moussaelianarsen/itd-7t6ko) | ||||||
| [](https://minio.arsenm.dev/minio/itd/) | [](https://aur.archlinux.org/packages/itd-git/) | ||||||
| [](https://aur.archlinux.org/packages/itd-git/) | [](https://aur.archlinux.org/packages/itd-bin/) | ||||||
|  |  | ||||||
| --- | --- | ||||||
|  |  | ||||||
| @@ -24,6 +24,36 @@ | |||||||
|  |  | ||||||
| --- | --- | ||||||
|  |  | ||||||
|  | ### Installation | ||||||
|  |  | ||||||
|  | Since ITD 0.0.7, packages are built and uploaded whenever a new release is created. | ||||||
|  |  | ||||||
|  | #### Arch Linux | ||||||
|  |  | ||||||
|  | Use the `itd-bin` or `itd-git` AUR packages. | ||||||
|  |  | ||||||
|  | #### Debian/Ubuntu | ||||||
|  |  | ||||||
|  | - Go to the [latest release](https://gitea.arsenm.dev/Arsen6331/itd/releases/latest) and download the `.deb` package for your CPU architecture. You can find your architecture by running `uname -m` in the terminal. | ||||||
|  | - Run `sudo apt install <package>`, replacing `<package>` with the path to the downloaded file. Note: relative paths must begin with `./`. | ||||||
|  | - Example: `sudo apt install ~/Downloads/itd-0.0.7-linux-aarch64.deb` | ||||||
|  |  | ||||||
|  | #### Fedora | ||||||
|  |  | ||||||
|  | - Go to the [latest release](https://gitea.arsenm.dev/Arsen6331/itd/releases/latest) and download the `.rpm` package for your CPU architecture. You can find your architecture by running `uname -m` in the terminal. | ||||||
|  | - Run `sudo dnf install <package>`, replacing `<package>` with the path to the downloaded file. | ||||||
|  | - Example: `sudo dnf install ~/Downloads/itd-0.0.7-linux-aarch64.rpm` | ||||||
|  |  | ||||||
|  | #### Alpine (and postmarketOS) | ||||||
|  |  | ||||||
|  | - Go to the [latest release](https://gitea.arsenm.dev/Arsen6331/itd/releases/latest) and download the `.apk` package for your CPU architecture. You can find your architecture by running `uname -m` in the terminal. | ||||||
|  | - Run `sudo apk add --allow-untrusted <package>`, replacing `<package>` with the path to the downloaded file. | ||||||
|  | - Example: `sudo apk add --allow-untrusted ~/Downloads/itd-0.0.7-linux-aarch64.apk` | ||||||
|  |  | ||||||
|  | Note: `--allow-untrusted` is required because ITD isn't part of a repository, and therefore is not signed. | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
| ### Socket | ### Socket | ||||||
|  |  | ||||||
| This daemon creates a UNIX socket at `/tmp/itd/socket`. It allows you to directly control the daemon and, by extension, the connected watch. | This daemon creates a UNIX socket at `/tmp/itd/socket`. It allows you to directly control the daemon and, by extension, the connected watch. | ||||||
| @@ -105,12 +135,22 @@ Use "itctl [command] --help" for more information about a command. | |||||||
|  |  | ||||||
| ### `itgui` | ### `itgui` | ||||||
|  |  | ||||||
| In `cmd/itgui`, there is a gui frontend to the socket of `itd`. It uses the [fyne library](https://fyne.io/) for Go. It can be compiled by running: | In `cmd/itgui`, there is a gui frontend to the socket of `itd`. It uses the [Fyne library](https://fyne.io/) for Go. | ||||||
|  |  | ||||||
|  | #### Compilation | ||||||
|  |  | ||||||
|  | Before compiling, certain prerequisites must be installed. These are listed on the following page: https://developer.fyne.io/started/#prerequisites | ||||||
|  |  | ||||||
|  | It can be compiled by running: | ||||||
|  |  | ||||||
| ```shell | ```shell | ||||||
| go build ./cmd/itgui | go build ./cmd/itgui | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
|  | #### Cross-compilation | ||||||
|  |  | ||||||
|  | Due to the use of OpenGL, cross-compilation of `itgui` isn't as simple as that of `itd` and `itctl`. The following guide from the Fyne website should work for `itgui`: https://developer.fyne.io/started/cross-compiling. | ||||||
|  |  | ||||||
| #### Screenshots | #### Screenshots | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										20
									
								
								api/fs.go
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								api/fs.go
									
									
									
									
									
								
							| @@ -2,6 +2,16 @@ package api | |||||||
|  |  | ||||||
| import "context" | import "context" | ||||||
|  |  | ||||||
|  | func (c *Client) RemoveAll(ctx context.Context, paths ...string) error { | ||||||
|  | 	return c.client.Call( | ||||||
|  | 		ctx, | ||||||
|  | 		"FS", | ||||||
|  | 		"RemoveAll", | ||||||
|  | 		paths, | ||||||
|  | 		nil, | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (c *Client) Remove(ctx context.Context, paths ...string) error { | func (c *Client) Remove(ctx context.Context, paths ...string) error { | ||||||
| 	return c.client.Call( | 	return c.client.Call( | ||||||
| 		ctx, | 		ctx, | ||||||
| @@ -22,6 +32,16 @@ func (c *Client) Rename(ctx context.Context, old, new string) error { | |||||||
| 	) | 	) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (c *Client) MkdirAll(ctx context.Context, paths ...string) error { | ||||||
|  | 	return c.client.Call( | ||||||
|  | 		ctx, | ||||||
|  | 		"FS", | ||||||
|  | 		"MkdirAll", | ||||||
|  | 		paths, | ||||||
|  | 		nil, | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  |  | ||||||
| func (c *Client) Mkdir(ctx context.Context, paths ...string) error { | func (c *Client) Mkdir(ctx context.Context, paths ...string) error { | ||||||
| 	return c.client.Call( | 	return c.client.Call( | ||||||
| 		ctx, | 		ctx, | ||||||
|   | |||||||
							
								
								
									
										26
									
								
								api/resources.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								api/resources.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | package api | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  |  | ||||||
|  | 	"go.arsenm.dev/infinitime" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // LoadResources loads resources onto the watch from the given | ||||||
|  | // file path to the resources zip | ||||||
|  | func (c *Client) LoadResources(ctx context.Context, path string) (<-chan infinitime.ResourceLoadProgress, error) { | ||||||
|  | 	progCh := make(chan infinitime.ResourceLoadProgress) | ||||||
|  |  | ||||||
|  | 	err := c.client.Call( | ||||||
|  | 		ctx, | ||||||
|  | 		"FS", | ||||||
|  | 		"LoadResources", | ||||||
|  | 		path, | ||||||
|  | 		progCh, | ||||||
|  | 	) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return progCh, nil | ||||||
|  | } | ||||||
| @@ -6,11 +6,26 @@ import ( | |||||||
| 	"time" | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/cheggaaa/pb/v3" | 	"github.com/cheggaaa/pb/v3" | ||||||
|  | 	"github.com/rs/zerolog/log" | ||||||
| 	"github.com/urfave/cli/v2" | 	"github.com/urfave/cli/v2" | ||||||
| 	"go.arsenm.dev/itd/api" | 	"go.arsenm.dev/itd/api" | ||||||
| ) | ) | ||||||
|  |  | ||||||
| func fwUpgrade(c *cli.Context) error { | func fwUpgrade(c *cli.Context) error { | ||||||
|  | 	resources := c.String("resources") | ||||||
|  | 	if resources != "" { | ||||||
|  | 		absRes, err := filepath.Abs(resources) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		err = resLoad(c.Context, []string{absRes}) | ||||||
|  | 		if err != nil { | ||||||
|  | 			log.Error().Msg("Resource loading has returned an error. This can happen if your current version of InfiniTime doesn't support BLE FS. Try updating without resource loading, and then load them after using the `itctl res load` command.") | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 	start := time.Now() | 	start := time.Now() | ||||||
|  |  | ||||||
| 	var upgType api.UpgradeType | 	var upgType api.UpgradeType | ||||||
|   | |||||||
| @@ -34,7 +34,12 @@ func fsMkdir(c *cli.Context) error { | |||||||
| 		return cli.Exit("Command mkdir requires one or more arguments", 1) | 		return cli.Exit("Command mkdir requires one or more arguments", 1) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err := client.Mkdir(c.Context, c.Args().Slice()...) | 	var err error | ||||||
|  | 	if c.Bool("parents") { | ||||||
|  | 		err = client.MkdirAll(c.Context, c.Args().Slice()...) | ||||||
|  | 	} else { | ||||||
|  | 		err = client.Mkdir(c.Context, c.Args().Slice()...) | ||||||
|  | 	} | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| @@ -109,7 +114,12 @@ func fsRemove(c *cli.Context) error { | |||||||
| 		return cli.Exit("Command remove requires one or more arguments", 1) | 		return cli.Exit("Command remove requires one or more arguments", 1) | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	err := client.Remove(c.Context, c.Args().Slice()...) | 	var err error | ||||||
|  | 	if c.Bool("recursive") { | ||||||
|  | 		err = client.RemoveAll(c.Context, c.Args().Slice()...) | ||||||
|  | 	} else { | ||||||
|  | 		err = client.Remove(c.Context, c.Args().Slice()...) | ||||||
|  | 	} | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -1,11 +1,11 @@ | |||||||
| package main | package main | ||||||
|  |  | ||||||
| import ( | import ( | ||||||
| 	"time" |  | ||||||
| 	"context" | 	"context" | ||||||
| 	"os" | 	"os" | ||||||
| 	"os/signal" | 	"os/signal" | ||||||
| 	"syscall" | 	"syscall" | ||||||
|  | 	"time" | ||||||
|  |  | ||||||
| 	"github.com/rs/zerolog" | 	"github.com/rs/zerolog" | ||||||
| 	"github.com/rs/zerolog/log" | 	"github.com/rs/zerolog/log" | ||||||
| @@ -29,12 +29,12 @@ func main() { | |||||||
| 	// at most 200ms after the user sends SIGINT/SIGTERM. | 	// at most 200ms after the user sends SIGINT/SIGTERM. | ||||||
| 	go func() { | 	go func() { | ||||||
| 		<-ctx.Done() | 		<-ctx.Done() | ||||||
| 		time.Sleep(200*time.Millisecond) | 		time.Sleep(200 * time.Millisecond) | ||||||
| 		os.Exit(0) | 		os.Exit(0) | ||||||
| 	}() | 	}() | ||||||
|  |  | ||||||
| 	app := cli.App{ | 	app := cli.App{ | ||||||
| 		Name: "itctl", | 		Name:            "itctl", | ||||||
| 		HideHelpCommand: true, | 		HideHelpCommand: true, | ||||||
| 		Flags: []cli.Flag{ | 		Flags: []cli.Flag{ | ||||||
| 			&cli.StringFlag{ | 			&cli.StringFlag{ | ||||||
| @@ -46,10 +46,23 @@ func main() { | |||||||
| 		}, | 		}, | ||||||
| 		Commands: []*cli.Command{ | 		Commands: []*cli.Command{ | ||||||
| 			{ | 			{ | ||||||
| 				Name: "help", | 				Name:      "help", | ||||||
| 				ArgsUsage: "<command>", | 				ArgsUsage: "<command>", | ||||||
| 				Usage: "Display help screen for a command", | 				Usage:     "Display help screen for a command", | ||||||
| 				Action: helpCmd, | 				Action:    helpCmd, | ||||||
|  | 			}, | ||||||
|  | 			{ | ||||||
|  | 				Name:    "resources", | ||||||
|  | 				Aliases: []string{"res"}, | ||||||
|  | 				Usage:   "Handle InfiniTime resource loading", | ||||||
|  | 				Subcommands: []*cli.Command{ | ||||||
|  | 					{ | ||||||
|  | 						Name:      "load", | ||||||
|  | 						ArgsUsage: "<path>", | ||||||
|  | 						Usage:     "Load an InifiniTime resources package", | ||||||
|  | 						Action:    resourcesLoad, | ||||||
|  | 					}, | ||||||
|  | 				}, | ||||||
| 			}, | 			}, | ||||||
| 			{ | 			{ | ||||||
| 				Name:    "filesystem", | 				Name:    "filesystem", | ||||||
| @@ -64,6 +77,13 @@ func main() { | |||||||
| 						Action:    fsList, | 						Action:    fsList, | ||||||
| 					}, | 					}, | ||||||
| 					{ | 					{ | ||||||
|  | 						Flags: []cli.Flag{ | ||||||
|  | 							&cli.BoolFlag{ | ||||||
|  | 								Name:    "parents", | ||||||
|  | 								Aliases: []string{"p"}, | ||||||
|  | 								Usage:   "Make parent directories if needed, no error if already existing", | ||||||
|  | 							}, | ||||||
|  | 						}, | ||||||
| 						Name:      "mkdir", | 						Name:      "mkdir", | ||||||
| 						ArgsUsage: "<paths...>", | 						ArgsUsage: "<paths...>", | ||||||
| 						Usage:     "Create new directories", | 						Usage:     "Create new directories", | ||||||
| @@ -84,6 +104,13 @@ func main() { | |||||||
| 						Action:      fsRead, | 						Action:      fsRead, | ||||||
| 					}, | 					}, | ||||||
| 					{ | 					{ | ||||||
|  | 						Flags: []cli.Flag{ | ||||||
|  | 							&cli.BoolFlag{ | ||||||
|  | 								Name:    "recursive", | ||||||
|  | 								Aliases: []string{"r", "R"}, | ||||||
|  | 								Usage:   "Remove directories and their contents recursively", | ||||||
|  | 							}, | ||||||
|  | 						}, | ||||||
| 						Name:      "remove", | 						Name:      "remove", | ||||||
| 						ArgsUsage: "<paths...>", | 						ArgsUsage: "<paths...>", | ||||||
| 						Aliases:   []string{"rm"}, | 						Aliases:   []string{"rm"}, | ||||||
| @@ -116,6 +143,11 @@ func main() { | |||||||
| 								Aliases: []string{"f"}, | 								Aliases: []string{"f"}, | ||||||
| 								Usage:   "Path to firmware image (.bin file)", | 								Usage:   "Path to firmware image (.bin file)", | ||||||
| 							}, | 							}, | ||||||
|  | 							&cli.PathFlag{ | ||||||
|  | 								Name:    "resources", | ||||||
|  | 								Aliases: []string{"r"}, | ||||||
|  | 								Usage:   "Path to resources file (.zip file)", | ||||||
|  | 							}, | ||||||
| 							&cli.PathFlag{ | 							&cli.PathFlag{ | ||||||
| 								Name:    "archive", | 								Name:    "archive", | ||||||
| 								Aliases: []string{"a"}, | 								Aliases: []string{"a"}, | ||||||
|   | |||||||
							
								
								
									
										57
									
								
								cmd/itctl/resources.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								cmd/itctl/resources.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | package main | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"context" | ||||||
|  | 	"path/filepath" | ||||||
|  |  | ||||||
|  | 	"github.com/cheggaaa/pb/v3" | ||||||
|  | 	"github.com/urfave/cli/v2" | ||||||
|  | 	"go.arsenm.dev/infinitime" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func resourcesLoad(c *cli.Context) error { | ||||||
|  | 	return resLoad(c.Context, c.Args().Slice()) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func resLoad(ctx context.Context, args []string) error { | ||||||
|  | 	if len(args) == 0 { | ||||||
|  | 		return cli.Exit("Command load requires one argument.", 1) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Create progress bar templates | ||||||
|  | 	rmTmpl := `Removing {{string . "filename"}}` | ||||||
|  | 	upTmpl := `Uploading {{string . "filename"}} {{counters . }} B {{bar . "|" "-" (cycle .) " " "|"}} {{percent . }} {{rtime . "%s"}}` | ||||||
|  | 	// Start full bar at 0 total | ||||||
|  | 	bar := pb.ProgressBarTemplate(rmTmpl).Start(0) | ||||||
|  |  | ||||||
|  | 	path, err := filepath.Abs(args[0]) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	progCh, err := client.LoadResources(ctx, path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for evt := range progCh { | ||||||
|  | 		if evt.Err != nil { | ||||||
|  | 			return evt.Err | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		if evt.Operation == infinitime.ResourceOperationRemoveObsolete { | ||||||
|  | 			bar.SetTemplateString(rmTmpl) | ||||||
|  | 			bar.Set("filename", evt.Name) | ||||||
|  | 		} else { | ||||||
|  | 			bar.SetTemplateString(upTmpl) | ||||||
|  | 			bar.Set("filename", evt.Name) | ||||||
|  |  | ||||||
|  | 			bar.SetTotal(evt.Total) | ||||||
|  | 			bar.SetCurrent(evt.Sent) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	bar.Finish() | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								go.mod
									
									
									
									
									
								
							| @@ -15,7 +15,7 @@ require ( | |||||||
| 	github.com/mozillazg/go-pinyin v0.19.0 | 	github.com/mozillazg/go-pinyin v0.19.0 | ||||||
| 	github.com/rs/zerolog v1.26.1 | 	github.com/rs/zerolog v1.26.1 | ||||||
| 	github.com/urfave/cli/v2 v2.3.0 | 	github.com/urfave/cli/v2 v2.3.0 | ||||||
| 	go.arsenm.dev/infinitime v0.0.0-20220511202257-9ed74726c478 | 	go.arsenm.dev/infinitime v0.0.0-20221025193634-0ad671d3f550 | ||||||
| 	go.arsenm.dev/lrpc v0.0.0-20220513001344-3bcc01fdb6a0 | 	go.arsenm.dev/lrpc v0.0.0-20220513001344-3bcc01fdb6a0 | ||||||
| 	golang.org/x/text v0.3.7 | 	golang.org/x/text v0.3.7 | ||||||
| 	modernc.org/sqlite v1.17.2 | 	modernc.org/sqlite v1.17.2 | ||||||
| @@ -38,11 +38,11 @@ require ( | |||||||
| 	github.com/gopherjs/gopherjs v1.17.2 // indirect | 	github.com/gopherjs/gopherjs v1.17.2 // indirect | ||||||
| 	github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect | 	github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51 // indirect | ||||||
| 	github.com/mattn/go-colorable v0.1.8 // indirect | 	github.com/mattn/go-colorable v0.1.8 // indirect | ||||||
| 	github.com/mattn/go-runewidth v0.0.12 // indirect | 	github.com/mattn/go-runewidth v0.0.13 // indirect | ||||||
| 	github.com/mitchellh/copystructure v1.2.0 // indirect | 	github.com/mitchellh/copystructure v1.2.0 // indirect | ||||||
| 	github.com/mitchellh/mapstructure v1.5.0 // indirect | 	github.com/mitchellh/mapstructure v1.5.0 // indirect | ||||||
| 	github.com/mitchellh/reflectwalk v1.0.2 // indirect | 	github.com/mitchellh/reflectwalk v1.0.2 // indirect | ||||||
| 	github.com/muka/go-bluetooth v0.0.0-20220219050759-674a63b8741a // indirect | 	github.com/muka/go-bluetooth v0.0.0-20220819140550-1d8857e3b268 // indirect | ||||||
| 	github.com/pelletier/go-toml v1.9.3 // indirect | 	github.com/pelletier/go-toml v1.9.3 // indirect | ||||||
| 	github.com/pmezard/go-difflib v1.0.0 // indirect | 	github.com/pmezard/go-difflib v1.0.0 // indirect | ||||||
| 	github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 // indirect | 	github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0 // indirect | ||||||
|   | |||||||
							
								
								
									
										11
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								go.sum
									
									
									
									
									
								
							| @@ -270,8 +270,9 @@ github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNx | |||||||
| github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= | ||||||
| github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= | github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y= | ||||||
| github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= | github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= | ||||||
| github.com/mattn/go-runewidth v0.0.12 h1:Y41i/hVW3Pgwr8gV+J23B9YEY0zxjptBuCWEaxmAOow= |  | ||||||
| github.com/mattn/go-runewidth v0.0.12/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk= | github.com/mattn/go-runewidth v0.0.12/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk= | ||||||
|  | github.com/mattn/go-runewidth v0.0.13 h1:lTGmDsbAYt5DmK6OnoV7EuIF1wEIFAcxld6ypU4OSgU= | ||||||
|  | github.com/mattn/go-runewidth v0.0.13/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w= | ||||||
| github.com/mattn/go-sqlite3 v1.14.12 h1:TJ1bhYJPV44phC+IMu1u2K/i5RriLTPe+yc68XDJ1Z0= | github.com/mattn/go-sqlite3 v1.14.12 h1:TJ1bhYJPV44phC+IMu1u2K/i5RriLTPe+yc68XDJ1Z0= | ||||||
| github.com/mattn/go-sqlite3 v1.14.12/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU= | github.com/mattn/go-sqlite3 v1.14.12/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU= | ||||||
| github.com/metal3d/fyne-x v0.0.0-20220508095732-177117e583fb h1:+fP6ENsbd+BUOmD/kSjNtrOmi2vgJ/JfWDSWjTKmTVY= | github.com/metal3d/fyne-x v0.0.0-20220508095732-177117e583fb h1:+fP6ENsbd+BUOmD/kSjNtrOmi2vgJ/JfWDSWjTKmTVY= | ||||||
| @@ -301,8 +302,8 @@ github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lN | |||||||
| github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||||
| github.com/mozillazg/go-pinyin v0.19.0 h1:p+J8/kjJ558KPvVGYLvqBhxf8jbZA2exSLCs2uUVN8c= | github.com/mozillazg/go-pinyin v0.19.0 h1:p+J8/kjJ558KPvVGYLvqBhxf8jbZA2exSLCs2uUVN8c= | ||||||
| github.com/mozillazg/go-pinyin v0.19.0/go.mod h1:iR4EnMMRXkfpFVV5FMi4FNB6wGq9NV6uDWbUuPhP4Yc= | github.com/mozillazg/go-pinyin v0.19.0/go.mod h1:iR4EnMMRXkfpFVV5FMi4FNB6wGq9NV6uDWbUuPhP4Yc= | ||||||
| github.com/muka/go-bluetooth v0.0.0-20220219050759-674a63b8741a h1:fnzS9RRQW8B5AgNCxkN0vJ/AoX+Xfqk3sAYon3iVrzA= | github.com/muka/go-bluetooth v0.0.0-20220819140550-1d8857e3b268 h1:kOnq7TfaAO2Vc/MHxPqFIXe00y1qBxJAvhctXdko6vo= | ||||||
| github.com/muka/go-bluetooth v0.0.0-20220219050759-674a63b8741a/go.mod h1:dMCjicU6vRBk34dqOmIZm0aod6gUwZXOXzBROqGous0= | github.com/muka/go-bluetooth v0.0.0-20220819140550-1d8857e3b268/go.mod h1:dMCjicU6vRBk34dqOmIZm0aod6gUwZXOXzBROqGous0= | ||||||
| github.com/neelance/astrewrite v0.0.0-20160511093645-99348263ae86/go.mod h1:kHJEU3ofeGjhHklVoIGuVj85JJwZ6kWPaJwCIxgnFmo= | github.com/neelance/astrewrite v0.0.0-20160511093645-99348263ae86/go.mod h1:kHJEU3ofeGjhHklVoIGuVj85JJwZ6kWPaJwCIxgnFmo= | ||||||
| github.com/neelance/sourcemap v0.0.0-20200213170602-2833bce08e4c/go.mod h1:Qr6/a/Q4r9LP1IltGz7tA7iOK1WonHEYhu1HRBA7ZiM= | github.com/neelance/sourcemap v0.0.0-20200213170602-2833bce08e4c/go.mod h1:Qr6/a/Q4r9LP1IltGz7tA7iOK1WonHEYhu1HRBA7ZiM= | ||||||
| github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8= | github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646/go.mod h1:jpp1/29i3P1S/RLdc7JQKbRpFeM1dOBd8T9ki5s+AY8= | ||||||
| @@ -396,8 +397,8 @@ github.com/yuin/goldmark v1.3.8/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1 | |||||||
| github.com/yuin/goldmark v1.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | github.com/yuin/goldmark v1.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= | ||||||
| github.com/yuin/goldmark v1.4.4 h1:zNWRjYUW32G9KirMXYHQHVNFkXvMI7LpgNW2AgYAoIs= | github.com/yuin/goldmark v1.4.4 h1:zNWRjYUW32G9KirMXYHQHVNFkXvMI7LpgNW2AgYAoIs= | ||||||
| github.com/yuin/goldmark v1.4.4/go.mod h1:rmuwmfZ0+bvzB24eSC//bk1R1Zp3hM0OXYv/G2LIilg= | github.com/yuin/goldmark v1.4.4/go.mod h1:rmuwmfZ0+bvzB24eSC//bk1R1Zp3hM0OXYv/G2LIilg= | ||||||
| go.arsenm.dev/infinitime v0.0.0-20220511202257-9ed74726c478 h1:HO+fteXuSnnT7po1PhGJK6nk8qMAGN2RqDaVP4sRN8g= | go.arsenm.dev/infinitime v0.0.0-20221025193634-0ad671d3f550 h1:GScKSLy9KI83+au3mV9w6koiYRap/uYNarRoUleou4k= | ||||||
| go.arsenm.dev/infinitime v0.0.0-20220511202257-9ed74726c478/go.mod h1:1cBQ3fp6QlRbSqu9kEBAHsVThINj31FtqHIYVsQ7wgg= | go.arsenm.dev/infinitime v0.0.0-20221025193634-0ad671d3f550/go.mod h1:K3NJ6fyPv5qqHUedB3MccKOE0whJMJZ80l/yTzzTrgc= | ||||||
| go.arsenm.dev/lrpc v0.0.0-20220513001344-3bcc01fdb6a0 h1:1K96g1eww+77GeGchwMhd0NTrs7Mk/Hc3M3ItW5NbG4= | go.arsenm.dev/lrpc v0.0.0-20220513001344-3bcc01fdb6a0 h1:1K96g1eww+77GeGchwMhd0NTrs7Mk/Hc3M3ItW5NbG4= | ||||||
| go.arsenm.dev/lrpc v0.0.0-20220513001344-3bcc01fdb6a0/go.mod h1:goK9z735lfXmqlDxu9qN7FS8t0HJHN3PjyDtCToUY4w= | go.arsenm.dev/lrpc v0.0.0-20220513001344-3bcc01fdb6a0/go.mod h1:goK9z735lfXmqlDxu9qN7FS8t0HJHN3PjyDtCToUY4w= | ||||||
| go.etcd.io/etcd/api/v3 v3.5.0/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= | go.etcd.io/etcd/api/v3 v3.5.0/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= | ||||||
|   | |||||||
							
								
								
									
										48
									
								
								socket.go
									
									
									
									
									
								
							
							
						
						
									
										48
									
								
								socket.go
									
									
									
									
									
								
							| @@ -293,6 +293,17 @@ type FS struct { | |||||||
| 	fs  *blefs.FS | 	fs  *blefs.FS | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (fs *FS) RemoveAll(_ *server.Context, paths []string) error { | ||||||
|  | 	fs.updateFS() | ||||||
|  | 	for _, path := range paths { | ||||||
|  | 		err := fs.fs.RemoveAll(path) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| func (fs *FS) Remove(_ *server.Context, paths []string) error { | func (fs *FS) Remove(_ *server.Context, paths []string) error { | ||||||
| 	fs.updateFS() | 	fs.updateFS() | ||||||
| 	for _, path := range paths { | 	for _, path := range paths { | ||||||
| @@ -309,6 +320,17 @@ func (fs *FS) Rename(_ *server.Context, paths [2]string) error { | |||||||
| 	return fs.fs.Rename(paths[0], paths[1]) | 	return fs.fs.Rename(paths[0], paths[1]) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (fs *FS) MkdirAll(_ *server.Context, paths []string) error { | ||||||
|  | 	fs.updateFS() | ||||||
|  | 	for _, path := range paths { | ||||||
|  | 		err := fs.fs.MkdirAll(path) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| func (fs *FS) Mkdir(_ *server.Context, paths []string) error { | func (fs *FS) Mkdir(_ *server.Context, paths []string) error { | ||||||
| 	fs.updateFS() | 	fs.updateFS() | ||||||
| 	for _, path := range paths { | 	for _, path := range paths { | ||||||
| @@ -424,6 +446,32 @@ func (fs *FS) Download(ctx *server.Context, paths [2]string) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func (fs *FS) LoadResources(ctx *server.Context, path string) error { | ||||||
|  | 	resFl, err := os.Open(path) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	progCh, err := infinitime.LoadResources(resFl, fs.fs) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	ch, err := ctx.MakeChannel() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	go func() { | ||||||
|  | 		for evt := range progCh { | ||||||
|  | 			ch <- evt | ||||||
|  | 		} | ||||||
|  | 		close(ch) | ||||||
|  | 	}() | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
| func (fs *FS) updateFS() { | func (fs *FS) updateFS() { | ||||||
| 	if fs.fs == nil || updateFS { | 	if fs.fs == nil || updateFS { | ||||||
| 		// Get new FS | 		// Get new FS | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user