Skip to content

Commit

Permalink
Merge pull request #11 from jcouture/feature/2.1.1-release
Browse files Browse the repository at this point in the history
Feature/2.1.1 release
  • Loading branch information
jcouture authored Jan 11, 2022
2 parents 4501b9f + 5500872 commit 9a9c97b
Show file tree
Hide file tree
Showing 9 changed files with 176 additions and 59 deletions.
6 changes: 3 additions & 3 deletions .goreleaser.yml
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ builds:
env:
- CGO_ENABLE=0
ldflags:
- -s -w -X github.com/jcouture/nv/internal/build.Version={{.Version}} -X github.com/jcouture/nv/internal/build.Date={{time "2006-01-02"}}
- -s -w -X github.com/jcouture/nv/internal/build.Version={{.Version}}
id: macos
goos: [darwin]
goarch: [amd64, arm64]
Expand All @@ -24,7 +24,7 @@ builds:
- <<: *build_defaults
id: windows
goos: [windows]
goarch: [386, amd64, arm]
goarch: [386, amd64, arm, arm64]

checksum:
name_template: 'checksums.txt'
Expand All @@ -45,4 +45,4 @@ brews:
homepage: https://github.com/jcouture/nv
description: Lightweight utility to load context specific environment variables
test: |
system "#{bin}/nv"
system "#{bin}/nv"
2 changes: 1 addition & 1 deletion LICENSE
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
The MIT License (MIT)

Copyright 2015-2018 Jean-Philippe Couture
Copyright 2015-2022 Jean-Philippe Couture

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
Expand Down
3 changes: 3 additions & 0 deletions Makefile
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
.PHONY: test run
test:
@go test -v -cover ./...
76 changes: 33 additions & 43 deletions cmd/nv/main.go
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
// Copyright 2015-2021 Jean-Philippe Couture
// Copyright 2015-2022 Jean-Philippe Couture
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
Expand Down Expand Up @@ -29,14 +29,25 @@ import (
"syscall"

"github.com/jcouture/nv/internal/build"
"github.com/jcouture/nv/internal/parser"
"github.com/jcouture/nv/internal/env"
"github.com/mitchellh/go-homedir"
)

func main() {
if len(os.Args) == 2 {
cmd := os.Args[1]
switch cmd {
case "-v", "version", "-version", "--version":
printVersion()
default:
printHelp()
}
os.Exit(0)
}

if len(os.Args) < 3 {
printUsage()
os.Exit(-1)
printHelp()
os.Exit(0)
}

fn := os.Args[1]
Expand All @@ -45,24 +56,23 @@ func main() {

filenames := strings.Split(fn, ",")

vars := make(map[string]string)
base := make(map[string]string)

for _, filename := range filenames {
// Parse file
parser := parser.NewParser(filename)
parsedVars, err := parser.Parse()
override, err := env.Load(filename)
if err != nil {
fmt.Printf("[Err] %s\n", err)
os.Exit(-1)
}
// Merge with possibly existing variables
mergeVars(vars, parsedVars)
base = env.Join(base, override)
}

loadAndMergeGlobalVars(vars)
globals := loadGlobals()
base = env.Join(base, globals)

clearEnv()
setEnvVars(vars)
env.Clear()
env.Set(base)

binary, lookErr := exec.LookPath(cmd)
if lookErr != nil {
Expand All @@ -76,40 +86,20 @@ func main() {
}
}

func printUsage() {
usage := `nv %s(%s) — context specific environment variables
Usage: nv <env files> <command> [arguments...]
`
fmt.Printf(usage, build.Version, build.Date)
}

func setEnvVars(vars map[string]string) {
for k, v := range vars {
os.Setenv(k, v)
}
func printHelp() {
fmt.Printf("usage: nv [--version] [--help]\n")
fmt.Printf(" <env files> <command> [arguments...]\n")
}

func mergeVars(vars1 map[string]string, vars2 map[string]string) {
for k, v := range vars2 {
vars1[k] = v
}
func printVersion() {
fmt.Printf("nv version %s\n", build.Version)
}

func loadAndMergeGlobalVars(vars map[string]string) {
dir, _ := homedir.Dir()
fn := filepath.Join(dir, ".nv")
parser := parser.NewParser(fn)
parsedVars, err := parser.Parse()
if err != nil {
// Return without breaking a sweat
return
}
mergeVars(vars, parsedVars)
}
func loadGlobals() map[string]string {
hdir, _ := homedir.Dir()
fn := filepath.Join(hdir, ".nv")
// Purposefuly ignoring any errors
globals, _ := env.Load(fn)

func clearEnv() {
// Clearing everything out the environment... but $PATH (we’re savages)!
path := os.Getenv("PATH")
os.Clearenv()
os.Setenv("PATH", path)
return globals
}
30 changes: 20 additions & 10 deletions internal/build/build.go
Original file line number Diff line number Diff line change
@@ -1,13 +1,23 @@
package build
// Copyright 2015-2022 Jean-Philippe Couture
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.

import "time"
package build

var Version = "DEV"

var Date = "" // YYYY-MM-DD

func init() {
if Version == "DEV" {
Date = time.Now().Format("2006-01-02")
}
}
56 changes: 56 additions & 0 deletions internal/env/env.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,56 @@
// Copyright 2015-2022 Jean-Philippe Couture
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.

package env

import (
"os"

"github.com/jcouture/nv/internal/parser"
)

func Set(vars map[string]string) {
for k, v := range vars {
os.Setenv(k, v)
}
}

func Join(base map[string]string, override map[string]string) map[string]string {
if len(base) == 0 {
return override
}
for k, v := range override {
base[k] = v
}

return base
}

func Load(fn string) (map[string]string, error) {
parser := parser.NewParser(fn)
return parser.Parse()
}

func Clear() {
// Clearing everything out the environment... except $PATH (we’re savages)!
path := os.Getenv("PATH")
os.Clearenv()
os.Setenv("PATH", path)
}
58 changes: 58 additions & 0 deletions internal/env/env_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
// Copyright 2015-2022 Jean-Philippe Couture
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.

package env

import (
"os"
"testing"
)

func TestSet(t *testing.T) {
name := "FOO"
expected := "BAR"

vars := make(map[string]string)
vars[name] = "BAR"
Set(vars)

result := os.Getenv(name)
if result != expected {
t.Errorf("Expected: %s, got: %s\n", expected, result)
}
}

func TestJoin(t *testing.T) {
base := make(map[string]string)
base["FOO"] = "BAR"

override := make(map[string]string)
override["COLOR"] = "RED"

result := Join(base, override)

if len(result) != 2 {
t.Errorf("Expected length: 2, got: %d\n", len(result))
}

if result["FOO"] != "BAR" {
t.Errorf("Expected FOO == BAR, got FOO == %s\n", result["FOO"])
}
}
2 changes: 1 addition & 1 deletion internal/parser/parser.go
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
// Copyright 2015-2021 Jean-Philippe Couture
// Copyright 2015-2022 Jean-Philippe Couture
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
Expand Down
2 changes: 1 addition & 1 deletion internal/parser/parser_test.go
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
// Copyright 2015-2021 Jean-Philippe Couture
// Copyright 2015-2022 Jean-Philippe Couture
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
Expand Down

0 comments on commit 9a9c97b

Please sign in to comment.