Skip to content

Commit

Permalink
add remote mixins (experimental)
Browse files Browse the repository at this point in the history
  • Loading branch information
kindermax committed Jul 30, 2022
1 parent 3dc4bc5 commit 6016851
Show file tree
Hide file tree
Showing 10 changed files with 236 additions and 41 deletions.
6 changes: 5 additions & 1 deletion docker/Dockerfile → Dockerfile
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
FROM golang:1.18.3-bullseye
FROM golang:1.18.3-bullseye as builder

ENV GOPROXY https://proxy.golang.org
WORKDIR /app
Expand All @@ -21,3 +21,7 @@ COPY go.mod .
COPY go.sum .

RUN go mod download

FROM golangci/golangci-lint:v1.45-alpine as linter

RUN mkdir -p /.cache && chmod -R 777 /.cache
22 changes: 8 additions & 14 deletions checksum/checksum.go
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,6 @@ import (
const (
DefaultChecksumKey = "__default_checksum__"
DefaultChecksumFileName = "lets_default_checksum"
checksumsDir = "checksums"
)

var checksumCache = make(map[string][]byte)
Expand Down Expand Up @@ -155,32 +154,27 @@ func ReadChecksumFromDisk(dotLetsDir, cmdName, checksumName string) (string, err
return string(fileData), nil
}

func getCmdChecksumPath(dotLetsDir string, cmdName string) string {
return filepath.Join(dotLetsDir, checksumsDir, cmdName)
func getCmdChecksumPath(checksumsDir string, cmdName string) string {
return filepath.Join(checksumsDir, cmdName)
}

// returns dir path and full file path to checksum
// (.lets/checksums/[command_name]/, .lets/checksums/[command_name]/[checksum_name]).
func getChecksumPath(dotLetsDir string, cmdName string, checksumName string) (string, string) {
dirPath := getCmdChecksumPath(dotLetsDir, cmdName)
func getChecksumPath(checksumsDir string, cmdName string, checksumName string) (string, string) {
dirPath := getCmdChecksumPath(checksumsDir, cmdName)

return dirPath, filepath.Join(dirPath, checksumName)
}

// TODO maybe checksumMap has to be separate struct ?
func PersistCommandsChecksumToDisk(dotLetsDir string, checksumMap map[string]string, cmdName string) error {
checksumPath := filepath.Join(dotLetsDir, checksumsDir)
if err := util.SafeCreateDir(checksumPath); err != nil {
return fmt.Errorf("can not create %s: %w", checksumPath, err)
}

func PersistCommandsChecksumToDisk(checksumsDir string, checksumMap map[string]string, cmdName string) error {
// TODO if at least one write failed do we have to revert all writes ???
for checksumName, checksum := range checksumMap {
filename := checksumName
if checksumName == DefaultChecksumKey {
filename = DefaultChecksumFileName
}
err := persistOneChecksum(dotLetsDir, cmdName, filename, checksum)
err := persistOneChecksum(checksumsDir, cmdName, filename, checksum)
if err != nil {
return err
}
Expand All @@ -189,8 +183,8 @@ func PersistCommandsChecksumToDisk(dotLetsDir string, checksumMap map[string]str
return nil
}

func persistOneChecksum(dotLetsDir string, cmdName string, checksumName string, checksum string) error {
checksumDirPath, checksumFilePath := getChecksumPath(dotLetsDir, cmdName, checksumName)
func persistOneChecksum(checksumsDir string, cmdName string, checksumName string, checksum string) error {
checksumDirPath, checksumFilePath := getChecksumPath(checksumsDir, cmdName, checksumName)
if err := util.SafeCreateDir(checksumDirPath); err != nil {
return fmt.Errorf("can not create checksum dir at %s: %w", checksumDirPath, err)
}
Expand Down
48 changes: 38 additions & 10 deletions config/config/config.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,12 @@
package config

import "github.com/lets-cli/lets/set"
import (
"fmt"
"path/filepath"

"github.com/lets-cli/lets/set"
"github.com/lets-cli/lets/util"
)

var (
// COMMANDS is a top-level directive. Includes all commands to run.
Expand Down Expand Up @@ -36,21 +42,43 @@ type Config struct {
isMixin bool // if true, we consider config as mixin and apply different parsing and validation
// absolute path to .lets
DotLetsDir string
// absolute path to .lets/checksums
ChecksumsDir string
// absolute path to .lets/mixins
MixinsDir string
}

func NewConfig(workDir string, configAbsPath string, dotLetsDir string) *Config {
return &Config{
Commands: make(map[string]Command),
Env: make(map[string]string),
WorkDir: workDir,
FilePath: configAbsPath,
DotLetsDir: dotLetsDir,
Commands: make(map[string]Command),
Env: make(map[string]string),
WorkDir: workDir,
FilePath: configAbsPath,
DotLetsDir: dotLetsDir,
ChecksumsDir: filepath.Join(dotLetsDir, "checksums"),
MixinsDir: filepath.Join(dotLetsDir, "mixins"),
}
}

func NewMixinConfig(cfg *Config, configAbsPath string) *Config {
mixin := NewConfig(cfg.WorkDir, configAbsPath, cfg.DotLetsDir)
mixin.isMixin = true

return mixin
}

func (c *Config) CreateChecksumsDir() error {
if err := util.SafeCreateDir(c.ChecksumsDir); err != nil {
return fmt.Errorf("can not create %s: %w", c.ChecksumsDir, err)
}

return nil
}

func NewMixinConfig(workDir string, configAbsPath string, dotLetsDir string) *Config {
cfg := NewConfig(workDir, configAbsPath, dotLetsDir)
cfg.isMixin = true
func (c *Config) CreateMixinsDir() error {
if err := util.SafeCreateDir(c.MixinsDir); err != nil {
return fmt.Errorf("can not create %s: %w", c.MixinsDir, err)
}

return cfg
return nil
}
150 changes: 146 additions & 4 deletions config/parser/parser.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,16 @@ package parser

import (
"bytes"
"context"
"crypto/sha256"
"errors"
"fmt"
"io"
"net/http"
"os"
"path/filepath"
"strings"
"time"

"github.com/lets-cli/lets/config/config"
"github.com/lets-cli/lets/config/path"
Expand Down Expand Up @@ -193,15 +199,113 @@ func isIgnoredMixin(filename string) bool {
return strings.HasPrefix(filename, "-")
}

type RemoteMixin struct {
URL string
Version string

mixinsDir string
}

// Filename is name of mixin file (hash from url).
func (rm *RemoteMixin) Filename() string {
hasher := sha256.New()
hasher.Write([]byte(rm.URL))

if rm.Version != "" {
hasher.Write([]byte(rm.Version))
}

return fmt.Sprintf("%x", hasher.Sum(nil))
}

// Path is abs path to mixin file (.lets/mixins/<filename>).
func (rm *RemoteMixin) Path() string {
return filepath.Join(rm.mixinsDir, rm.Filename())
}

func (rm *RemoteMixin) persist(data []byte) error {
f, err := os.OpenFile(rm.Path(), os.O_CREATE|os.O_WRONLY, 0o755)
if err != nil {
return fmt.Errorf("can not open file %s to persist mixin: %w", rm.Path(), err)
}

_, err = f.Write(data)
if err != nil {
return fmt.Errorf("can not write mixin to file %s: %w", rm.Path(), err)
}

return nil
}

func (rm *RemoteMixin) exists() bool {
return util.FileExists(rm.Path())
}

func (rm *RemoteMixin) tryRead() ([]byte, error) {
if !rm.exists() {
return nil, nil
}
data, err := os.ReadFile(rm.Path())
if err != nil {
return nil, fmt.Errorf("can not read mixin config file at %s: %w", rm.Path(), err)
}

return data, nil
}

func (rm *RemoteMixin) download() ([]byte, error) {
// TODO: maybe create a client for this?
ctx, cancel := context.WithTimeout(context.Background(), 60*5*time.Second)
defer cancel()

req, err := http.NewRequestWithContext(
ctx,
"GET",
rm.URL,
nil,
)
if err != nil {
return nil, err
}

client := &http.Client{
Timeout: 15 * 60 * time.Second, // TODO: move to client struct
}

resp, err := client.Do(req)
if err != nil {
return nil, fmt.Errorf("failed to make request: %w", err)
}

defer resp.Body.Close()

if resp.StatusCode == http.StatusNotFound {
return nil, fmt.Errorf("no such file at: %s", rm.URL)
} else if resp.StatusCode < 200 || resp.StatusCode > 299 {
return nil, fmt.Errorf("network error: %s", resp.Status)
}

data, err := io.ReadAll(resp.Body)
if err != nil {
return nil, fmt.Errorf("failed to read response: %w", err)
}

return data, nil
}

func readAndValidateMixins(mixins []interface{}, cfg *config.Config) error {
for _, filename := range mixins {
if filename, ok := filename.(string); ok { //nolint:nestif
if err := cfg.CreateMixinsDir(); err != nil {
return err
}

for _, mixin := range mixins {
if filename, ok := mixin.(string); ok { //nolint:nestif
configAbsPath, err := path.GetFullConfigPath(normalizeMixinFilename(filename), cfg.WorkDir)
if err != nil {
if isIgnoredMixin(filename) && errors.Is(err, path.ErrFileNotExists) {
continue
} else {
// complain non-existed mixin only if its filename does not starts with dash `-`
// complain non-existed mixin only if its filename does not start with dash `-`
return fmt.Errorf("failed to read mixin config: %w", err)
}
}
Expand All @@ -210,14 +314,52 @@ func readAndValidateMixins(mixins []interface{}, cfg *config.Config) error {
return fmt.Errorf("can not read mixin config file: %w", err)
}

mixinCfg := config.NewMixinConfig(cfg.WorkDir, filename, cfg.DotLetsDir)
mixinCfg := config.NewMixinConfig(cfg, filename)
if err := parseMixinConfig(fileData, mixinCfg); err != nil {
return fmt.Errorf("failed to load mixin config '%s': %w", filename, err)
}

if err := mergeConfigs(cfg, mixinCfg); err != nil {
return fmt.Errorf("failed to merge mixin config %s with main config: %w", filename, err)
}
} else if mixinMapping, ok := mixin.(map[string]interface{}); ok {
rm := &RemoteMixin{mixinsDir: cfg.MixinsDir}
if url, ok := mixinMapping["url"]; ok {
// TODO check if url is valid
rm.URL, _ = url.(string)
}

if version, ok := mixinMapping["version"]; ok {
rm.Version, _ = version.(string)
}

data, err := rm.tryRead()
if err != nil {
return err
}

if data == nil {
data, err = rm.download()
if err != nil {
return err
}
}

// TODO: what if multiple mixins have same commands
// 1 option - fail and suggest use to namespace all commands in remote mixin
// 2 option - namespace it (this may require specifying namespace in mixin config or in main config mixin section)
mixinCfg := config.NewMixinConfig(cfg, rm.Filename())
if err := parseMixinConfig(data, mixinCfg); err != nil {
return fmt.Errorf("failed to load remote mixin config '%s': %w", rm.URL, err)
}

if err := mergeConfigs(cfg, mixinCfg); err != nil {
return fmt.Errorf("failed to merge remote mixin config %s with main config: %w", rm.URL, err)
}

if err := rm.persist(data); err != nil {
return fmt.Errorf("failed to persist remote mixin config %s: %w", rm.URL, err)
}
} else {
return newConfigParseError(
"must be a string",
Expand Down
2 changes: 1 addition & 1 deletion docker-compose.yml
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ services:
image: lets
build:
context: .
dockerfile: docker/Dockerfile
dockerfile: Dockerfile
working_dir: /app
volumes:
- ./:/app
Expand Down
3 changes: 0 additions & 3 deletions docker/lint.Dockerfile

This file was deleted.

32 changes: 32 additions & 0 deletions docs/docs/best_practices.md
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,38 @@ id: best_practices
title: Best practices
---

### Naming conventions

Prefer single word over plural.

It is better to leverage semantics of `lets` as an intention to do something. For example it is natural saying `lets test` or `lets build` something.

`bad`

```
lets runs
```

`good`

```
lets run
```

---

`bad`

```
lets tests
```

`good`

```
lets test
```

### Default commands

If you have many projects (lets say - microservices) - it would be great to have one way to run and operate them when developing
Expand Down
4 changes: 2 additions & 2 deletions lets.build.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,8 @@ env:
commands:
build-lets-image:
description: Build lets docker image
cmd: docker build -t lets -f docker/Dockerfile .
cmd: docker build -t lets -f Dockerfile --target builder .

build-lint-image:
description: Build lets lint docker image
cmd: docker build -t lets-lint -f docker/lint.Dockerfile .
cmd: docker build -t lets-lint -f Dockerfile --target linter .
Loading

0 comments on commit 6016851

Please sign in to comment.