KEINOS' Blog

Official blog of KEINOS but mostly for my own reference. A Japanese made in Mexico with Mexican quality. Who monkey around the jungle of codes. ;-)


Project maintained by KEINOS Hosted on GitHub Pages — Theme by mattgraham

INDEX


Golang CheetSheet

This is a memorandum for KEINOS.

Generate rowID and tableID for Key-Value SQLite3

Example of creating the keys (rowid and table name) from the contens to use SQLite3 as a CAS (Content Addressable Storage).

package main

import (
	"fmt"
	"math/big"

	"github.com/zeebo/blake3"
)

func Example() {
	for _, sample := range []string{
		"Hello World",
		"Hello World!",
		"Hello, world",
	} {
		rowID, tableID := GetIDs(sample)

		fmt.Println("RowID:", rowID)
		fmt.Println("TableID:", tableID)
	}
	// RowID: 4753612358325858618
	// TableID: table039
	// RowID: 6676447199849907433
	// TableID: table179
	// RowID: -124719410497300881
	// TableID: table218
}

// GetIDs returns a unique rowID and tableID combination from the given data.
//
// "rowID" is the first 8 bytes of BLAKE3-256 hash as signed decimal string.
// "tableID" is the 1 byte XOR checksum of the full hash as decimal string with
// "table" prefix (`table<xor sum>`).
func GetIDs(data string) (rowID, tableID string) {
	digest := blake3.Sum256([]byte(data))

	chksum := byte(0)
	for _, b := range digest {
		chksum ^= b
	}

	tableID = fmt.Sprintf("table%03d", chksum)

	trimmed := digest[:8]
	hashedInt := big.NewInt(0).SetBytes(trimmed).Int64()
	rowID = fmt.Sprintf("%d", hashedInt)

	return rowID, tableID
}

How to convert bytes to int ([]byte --> int)

import "math/big"

b := []byte{0xFF, 0xFF, 0xFF} // 0d16777215 = 0xffffff
fmt.Printf("%X\n", b)

e := int(big.NewInt(0).SetBytes(b).Uint64())
fmt.Printf("%v\n", e)
// Output:
// FFFFFF
// 16777215

Random Numbers

Cryptographically Insecure

Cryptographically Secure

// Cryptographically secure random number but slower than math/rand.
import "crypto/rand"

// Ger random number between 0 and 99.
num := int64(100)

n, err := rand.Int(rand.Reader, big.NewInt(num))
if err != nil {
    panic(err)
}

fmt.Println(n)

Enum (Enumerable)

type Season int

const (
	Summer Season = iota
	Autumn
	Winter
	Spring
)

func (s Season) String() string {
	switch s {
	case Summer:
		return "summer"
	case Autumn:
		return "autumn"
	case Winter:
		return "winter"
	case Spring:
		return "spring"
	}

    return "unknown"
}

func printSeason(s Season) {
	fmt.Println("season: ", s)
}

func main() {
	i := Summer
	printSeason(i)
}

[Back to top]

Get imported module’s version from the code

mods := []*debug.Module{}

if buildInfo, ok := debug.ReadBuildInfo(); ok {
  mods = buildInfo.Deps
}

if len(mods) == 0 {
  dummyMod := &debug.Module{
    Path:    "n/a",
    Version: "n/a",
    Sum:     "n/a",
  }

  mods = []*debug.Module{
    dummyMod,
  }
}

getModName := func(modDep *debug.Module) string {
  // module name without leading version in a path
  noVer := strings.ReplaceAll(modDep.Path, "/"+modDep.Version, "")

  return filepath.Base(noVer)
}

modsFound := map[string]debug.Module{}

for _, modDep := range mods {
  name := getModName(modDep)
  modsFound[name] = *modDep
}

return modsFound

[Back to top]

How to get content like cURL

response, err := http.Get(c.EndpointURL)
if err != nil {
  return nil, errors.Wrap(err, "failed to GET HTTP request")
}

defer response.Body.Close()

// Read responce body
resBody, err := io.ReadAll(response.Body)
if err != nil {
  return nil, errors.Wrap(err, "fail to read response")
}

if response.StatusCode != http.StatusOK {
  return nil, errors.Errorf(
    "fail to GET response from: %v\nStatus: %v\nResponse body: %v",
    c.EndpointURL,
    response.Status,
    string(resBody),
  )
}

fmt.Println(string(resBody))

[Back to top]

How to return error response in httptest.NewServer during test

dummySrv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
    w.WriteHeader(http.StatusBadRequest)
    fmt.Fprintf(w, "invalid request")

    // return is not needed. it will be redundant
}))
defer dummySrv.Close()

[Back to top]

How to update all the packages(go.mod and go.sum to the latest)

go get -u ./...
go mod tidy

[Back to top]

How to sleep a second

time.Sleep(time.Second)

How to sleep n seconds

import "time"

sleepTime := time.Duration(5)
time.Sleep(time.time.Second * time.Duration(sec))
func randSleep(secMax int) {
	if secMax == 0 {
		secMax = 1
	}
	// In case of secMax = 1, we get a random number between 0 and 999
	sec := randInt(secMax * 1000)

	time.Sleep(time.Millisecond * time.Duration(sec))
}

[Back to top]

Field Names/Variables of GoReleaser

List of available field names in GoReleaser’s config file. Such as , for example.

# =============================================================================
#  Configuration of goreleaser for go-multihash
# =============================================================================
#  For local-test run:
#    $ goreleaser release --snapshot --skip-publish --rm-dist
#    $ # *Note: Check the ./dist/ dir after ran.
#
#  Make sure to check the documentation as well at:
#    https://goreleaser.com/customization/
# =============================================================================
before:
  hooks:
    - go mod download
# Name to use on test release with --snapshot option.
snapshot:
  name_template: ''

# Settings to build the binaries.
builds:
  -
    # Target directory of main.go
    main: ./cmd/multihash
    # Output binary name
    binary: multihash
    env:
      - CGO_ENABLED=0
    # Target OS
    goos:
      - linux
      - darwin
    # Target architectures
    goarch:
      - amd64
      - arm
      - arm64
    # Variant for ARM32
    goarm:
      - "5"
      - "6"
      - "7"
    # Ignore ARM32/ARM64 build for both macOS and Windows
    ignore:
      - goos: darwin
        goarch: arm
    # Build the app as static binary and embed version and commit info
    ldflags:
      - -s -w -extldflags '-static' -X 'main.version=' -X 'main.tag='

# macOS universal binaries for both arm64 and amd64
universal_binaries:
  -
    name_template: 'multihash'
    # Combine arm64 and amd64 as a single binary and remove each
    replace: true

# Archiving the built binaries
archives:
  -
    replacements:
      darwin: macOS
      linux: Linux
    format_overrides:
      - goos: darwin
        format: zip

# Create checksum file of archived files
checksum:
  name_template: 'checksums.txt'

# Release/update Homebrew tap repository
brews:
  -
    # Name of the package: multihash.rb
    name: multihash
    # Target repo to tap: KEINOS/homebrew-apps
    tap:
      owner: KEINOS
      name: homebrew-apps
    # Target directory: KEINOS/homebrew-apps/Formula
    folder: Formula
    # URL of the archive in releases page
    url_template: "https://github.com/KEINOS/go-multihash/releases/download//"
    # Author info to commit to the tap repo
    commit_author:
      name: goreleaserbot
      email: [email protected]
    # Message to display on `brew search` or `brew info`
    description: "Multihash is a command that returns a self-explanatory hash value."
    homepage: "https://github.com/KEINOS/go-multihash/"
    # Let brew command pull the archive via cURL
    download_strategy: CurlDownloadStrategy
    # Let brew command instll the binary as `go-pallet`
    install: |
      bin.install "multihash"
    # Smoke test to run after install
    test: |
      system "#{bin}/multihash -h"

[Back to top]

How to install benchstat

go install "golang.org/x/perf/cmd/benchstat@latest"

[Back to top]

How to benchmark

func BenchmarkAppend_AllocateEveryTime(b *testing.B) {
    base := []string{}

    b.ResetTimer()
    // b.N is the number of iterations given from the benchmarking tool.
    for i := 0; i < b.N; i++ {
        base = append(base, fmt.Sprintf("no%d", i))
    }
}
$ go test -bench . -benchmem > bench.txt
...
$ benchstat ./bench.txt
# Options
-benchmem ............ Print memory allocations
-benchtime t ......... Iterate for t seconds. Default 1s.
-cpuprofile=*.prof ... Detaild CPU profiling information. Viewable with `go tool pprof`.
-count ............... Number of test iterations to run.
-cpu ................. Number of CPUs to use.
-memprofile=*.mem .... Detailed memory profiling information. Viewable with `go tool pprof`.

[Back to top]

How to generate 1MBytes of consistent data for testing

// inputData holds 1MB(1e6) size of data created by testData() function.
var inputData []byte

// The testData creates 1,000,000 bytes= 1MB (1e6) size of data.
// The returned values are consistent and not random.
func testData(b *testing.B) []byte {
	b.Helper()

  // use initialized data
	if len(inputData) != 0 {
    return inputData
  }

	// Initialize data
  inputData = make([]byte, 1e6)

  for i := range inputData {
    // Custom this line to generate different data
    inputData[i] = byte(i % 251)
  }

	return inputData
}

[Back to top]

How to write gigantic data to a file

For huge amount of data, instead of using directly os.File.Write() method, use bufio.Writer in-between to write data in chunks to speed up the process.

	fileP, err := os.Create(pathFile)
	if err != nil {
		return errors.Wrap(err, "failed to open/create file")
	}

	defer fileP.Close()

	bufP := bufio.NewWriter(fileP)
	defer bufP.Flush()

	totalSize := int64(0)
	countLine := 0

	for {
		countLine++

		written, err := bufP.WriteString(fmt.Sprintf("line: %d\n", countLine))
		if err != nil {
			return errors.Wrap(err, "failed to write line")
		}

		totalSize += int64(written)
	}

How to check if file exists

import (
  "os"
  "io/fs"
)

// PathExists returns true if the given path exists. Whether it is a file or a
// directory.
func PathExists(path string) bool {
	_, err := os.Stat(path)

	return err == nil
}

// IsFile returns true if the given file path exists and is not a directory.
func IsFile(pathFile string) bool {
	info, err := os.Stat(pathFile)
	if err == nil {
		return !info.IsDir()
	}

	return false
}

// FileExists returns true if the given file path exists and is not a directory.
func FileExists(path string) bool {
	fileInfo, err := os.Stat(path)

	return !errors.Is(err, fs.ErrNotExist) && !fileInfo.IsDir()
}

If you want to check if a file exists before opening it, you don’t need to check the path before opening it.

The below opens a file if it exists, otherwise it creates a new one.

f, err := os.OpenFile(pathFile, os.O_RDWR|os.O_CREATE|os.O_EXCL, 0666)
if errors.Is(err, os.ErrNotExist) {
  ...
}

[Back to top]

How to deal with/mimic io.Reader

package main

import (
	"errors"
	"fmt"
	"io"
	"strings"
)

func doSomething(rd io.Reader) error {
	if rd == nil {
		return errors.New("nil pointer for input given")
	}

	const bufferSize = 256 // Chunk size to read from rd
	var content []byte     // Read data to store

	// Create buffer to read
	buffer := make([]byte, bufferSize)

	for {
		n, err := rd.Read(buffer)
		if 0 < n {
			// Read
			content = append(content, buffer...)
		}
		if err == io.EOF {
			break // End of file
		}
		if err != nil {
			return err // error
		}
	}

	fmt.Println(string(content))

	return nil
}

func main() {
	// String
	input := "some string"

	r := strings.NewReader(input)

	doSomething(r)
}

[Back to top]

How to deal with/mimic io.Writer

package main

import (
	"bytes"
	"fmt"
	"io"
)

func writeSomething(w io.Writer) (n int, err error) {
	return w.Write([]byte("foo bar"))
}

func main() {
	var b bytes.Buffer

	writeSomething(&b)

	fmt.Println(b.String())
}

How to capture os.Stdout (dealing with/mock/mimic os.Stdout)

Use os.Pipe() to capture os.Stdout during the function execution.

package main

import (
	"fmt"
	"io"
	"log"
	"os"
)

func main() {
	output, err := captureOutput(sayHello)
	if err != nil {
		log.Fatal(err)
	}

	fmt.Println("Captured output:", output)
}

func sayHello() {
	fmt.Println("Hello, 世界")
}

func captureOutput(runFn func()) (string, error) {
	orig := os.Stdout

	r, w, err := os.Pipe()
	if err != nil {
		return "", err
	}

	os.Stdout = w

	// Run the function and capture
	runFn()

	os.Stdout = orig
	w.Close()

	out, err := io.ReadAll(r)
	if err != nil {
		return "", err
	}

	return string(out), err
}

[Back to top]

How to deal with/mock/mimic os.Stdin

// mockStdin is a helper function that lets the test pretend dummyInput as os.Stdin.
// It will return a function to `defer` clean up after the test.
//
// Note: This function is not thread-safe. It should not use in parallel tests.
func mockStdin(t *testing.T, dummyInput string) (funcDefer func(), err error) {
	t.Helper()

	oldOsStdin := os.Stdin

	tmpfile, err := os.CreateTemp(t.TempDir(), t.Name())
	if err != nil {
		return nil, errors.Wrap(err, "failed to create temp file during mocking os.Stdin")
	}

	content := []byte(dummyInput)

	if _, err := tmpfile.Write(content); err != nil {
		return nil, errors.Wrap(err, "failed to write to temp file during mocking os.Stdin")
	}

	if _, err := tmpfile.Seek(0, 0); err != nil {
		return nil, errors.Wrap(err, "failed to seek the temp file during mocking os.Stdin")
	}

	// Set stdin to the temp file
	os.Stdin = tmpfile

	return func() {
		// clean up
		os.Stdin = oldOsStdin
		os.Remove(tmpfile.Name())
	}, nil
}

[Back to top]

How to trim/remove comments

package main

import (
	"fmt"
	"strings"
	"testing"
	"unicode"
)

func StripComment(delimiter, source string) string {
	if strings.Contains(source, "\n") {
		result := []string{}

		lines := strings.Split(source, "\n")
		for _, line := range lines {
			if strings.TrimSpace(line) == "" {
				result = append(result, "")
			}
			stripped := StripComment(delimiter, line)
			if strings.TrimSpace(stripped) != "" {
				result = append(result, stripped)
			}
		}

		return strings.Join(result, "\n")
	}

	if cut := strings.IndexAny(source, delimiter); cut >= 0 {
		return strings.TrimRightFunc(source[:cut], unicode.IsSpace)
	}

	return source
}

func TestStripComment(t *testing.T) {
	for i, test := range []struct {
		input  string
		expect string
	}{
		{input: "# foo bar", expect: ""},
		{input: "foo # bar", expect: "foo"},
		{input: "foo bar # buzz", expect: "foo bar"},
		{input: "foo\n#bar\n#buz\nhoge", expect: "foo\nhoge"},
		{input: "foo\n#bar\n#buz\n\nhoge", expect: "foo\n\nhoge"},
		{input: "foo\n#bar\n#buz\n   \nhoge", expect: "foo\n\nhoge"},
		{input: "foo\n#bar\n#buz\n   hoge\nfuga", expect: "foo\n   hoge\nfuga"},
		{input: "foo\nbar #buz\n   hoge #fuga\npiyo", expect: "foo\nbar\n   hoge\npiyo"},
	} {
		expect := test.expect
		actual := StripComment("#", test.input)

		if expect != actual {
			fmt.Printf("test #%d failed. got: %s, want: %s\n", i+1, actual, expect)
			t.Fail()
		}
	}
}

[Back to top]

How to shuffle a slice

// https://golang.org/pkg/math/rand/

package main

import (
	"fmt"
	"math/rand"
	"time"
)

func main() {
	cards := []string{"2", "3", "4", "5", "6", "7", "8", "9", "10", "J", "Q", "K", "A"}
	cards4 := []string{}

	// Prepare 4 stacks of cards
	for i := 0; i < 4; i++ {
		cards4 = append(cards4, cards...)
	}

	rand.Seed(time.Now().UnixNano())
	rand.Shuffle(len(cards4), func(i, j int) { cards4[i], cards4[j] = cards4[j], cards4[i] })

	fmt.Println(cards4)
	// [7 3 10 7 10 9 A 8 K 9 J 10 10 A K Q K 5 K Q 6 J 3 8 2 6 2 7 Q J 8 3 J 6 2 8 4 9 A 9 2 4 4 3 5 A 4 6 5 Q 7 5]
	// [5 2 K K K 10 4 J 9 8 A 3 3 5 4 Q 9 3 4 J 7 6 5 Q 10 J A A 2 7 9 6 7 8 8 7 9 K Q 2 J 5 10 2 4 Q 8 A 6 6 10 3]
}

[Back to top]

How to remove a repeated spaces in a string

func TrimWordGaps(s string) string {
	return strings.Join(strings.Fields(s), " ")
}

[Back to top]

How to get an element randomly from a slice

rand.Seed(time.Now().UnixNano())
choises := []string{
	"One",
	"Two",
	"Three",
	"Four",
}
fmt.Println("Random pick:", choises[rand.Intn(len(choises))])

[Back to top]

How to build static binary

go build \
      -ldflags="-s -w -extldflags \"-static\"" \
      -o /go/bin/myapp \
      ./cmd/myapp/main.go
# Smoke test
/go/bin/myapp

[Back to top]

Get maximum int value available

// Xor the uint zero value to get the max value of uint then bit shift to get
// the max positive value of int.
const MaxInt = int(^uint(0) >> 1)
const MaxUint = ^uint(0)
const MinUint = 0
const MaxInt = int(MaxUint >> 1)
const MinInt = -MaxInt - 1

[Back to top]