GOLANG SNIPPETS


SEND/RECEIVE DATA

SendResponse

// SendResponse...
func SendResponse(w http.ResponseWriter, d interface{}, s int) {
    w.WriteHeader(s)
    if d == nil {
        return
    }
    w.Header().Set("Content-Type", "application/json")
    dataJSON, err := json.MarshalIndent(d, "", " ")
    if err != nil {
        log.Printf("ERROR Marshaling %v\n", err)
        http.Error(w, "Internal Server Error", http.StatusInternalServerError)
        return
    }
    _, err = w.Write(dataJSON)
    if err != nil {
        log.Printf("ERROR writing JSON response: %v\n", err)
    }
}

// SendtResponseXML ...
func SendResponseXML(w http.ResponseWriter, d interface{}, s int) {
    w.WriteHeader(s)
    if d == nil {
        return
    }
    w.Header().Set("Content-Type", "application/xml")
    dataXML, err := xml.MarshalIndent(d, "", "  ")
    if err != nil {
        log.Printf("ERROR Marshaling XML: %v\n", err)
        http.Error(w, "Internal Server Error", http.StatusInternalServerError)
        return
    }
    _, err = w.Write(dataXML)
    if err != nil {
        log.Printf("ERROR writing XML response: %v\n", err)
    }
}

Fetch GET

// FetchGET ...
func FetchGET(url string, d interface{}) error {
    client := &http.Client{
        Timeout: 10 * time.Second,
    }

    // PARAMS IN URL
    u, err := url.Parse(BASE_URL)
    if err != nil {
        err := fmt.Sprintf("ERROR Parsing URL => %v", err)
    }
    params := url.Values{
        "key": {API_KEY},
        "aqi": {"no"},
        "q":   {city},
    }
    u.RawQuery = params.Encode()
    path := u.String()

    // HEADERS
    req, err := http.NewRequest("GET", path, nil)
    if err != nil {
        log.Printf("ERROR: creating request %s => %v", path, err)
        return err
    }
    req.Header.Set("Authorization", "Bearer ACCESS_TOKEN")
    req.Header.Set("Accept", "application/json")
    resp, err := client.Do(req) 
    if err != nil {
        log.Printf("ERROR: Request %s => %v", path, err)
        return err
    }
    /////

    // NO HEADERS
    resp, err := client.Get(path)
    if err != nil {
        log.Printf("ERROR: Request %s => %v", path, err)
        return err
    }
    /////

    // common
    defer resp.Body.Close()

    if resp.StatusCode < 200 || resp.StatusCode >= 300 {
        err := errors.New("HTTP status:" + http.StatusText(resp.StatusCode))
        log.Printf("ERROR: %v", err)
        return err
    }

    decoder := json.NewDecoder(resp.Body)
    err = decoder.Decode(&d)
    if err != nil {
        log.Printf("ERROR unnmarshalling => %v", err)
        return err
    }

    return nil
}

DoGetConcurrentRequest

func fillDefaultStocks(links []string) {
    ch := make(chan []byte)
    for _, link := range links {
        go doGetConcurrentRequest(link, ch)
    }
    for range links {
        json.Unmarshal(<-ch, &stocks)
    }
}

func doGetConcurrentRequest(url string, ch chan<- []byte) {
    resp, err := http.Get(url)
    if err != nil {
        msg := fmt.Sprintf("ERROR 1 HTTP Request %s", err)
        log.Printf(msg)
        ch <- []byte(msg)
        return
    }
    if resp.StatusCode != 200 {
        msg := fmt.Sprintf("ERROR 2 Status Code %d", resp.StatusCode)
        log.Printf(msg)
        ch <- []byte(msg)
        return
    }
    defer resp.Body.Close()
    body, err := ioutil.ReadAll(resp.Body)
    if err != nil {
        msg := fmt.Sprintf("ERROR 3 HTTP Request %s", err)
        log.Printf(msg)
        ch <- []byte(msg)
        return
    }
    ch <- body
}

fetchPOST

// FetchPOST ...
func FetchPOST(url string, d interface{}) error {
    client := &http.Client{
        Timeout: 10 * time.Second,
    }

    jsonData, err := json.Marshal(d)
    if err != nil {
        log.Printf("ERROR: Failed to marshal request body => %v", err)
        return err
    }

    req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonData))
    if err != nil {
        log.Printf("ERROR: Failed to create request for %s => %v", url, err)
        return err
    }

    req.Header.Set("Authorization", "Bearer ACCESS_TOKEN")
    req.Header.Set("Accept", "application/json")
    req.Header.Set("Content-Type", "application/json")

    resp, err := client.Do(req)
    if err != nil {
        log.Printf("ERROR: Request %s => %v", url, err)
        return err
    }
    defer resp.Body.Close()

    if resp.StatusCode < 200 || resp.StatusCode >= 300 {
        err := errors.New("HTTP status: " + http.StatusText(resp.StatusCode))
        log.Printf("ERROR: %v", err)
        return err
    }

    decoder := json.NewDecoder(resp.Body)
    err = decoder.Decode(&d)
    if err != nil {
        log.Printf("ERROR unmarshalling => %v", err)
        return err
    }

    return nil
}

fetchPOST with params

import (
    url "net/url"
)

// FetchPOSTparams ...
func FetchPOSTparams(path string, d map[string]string) error {
    client := &http.Client{
        Timeout: 10 * time.Second,
    }

    params := url.Values{}
    for key, value := range d {
        params.Add(key, value)
    }
    /*params := url.Values{
        "test": {myTest},
        "data": {myData},
    }*/

    body := bytes.NewBufferString(params.Encode())

    req, err := http.NewRequest("POST", path, body)
    if err != nil {
        log.Printf("ERROR: Failed to create request for %s => %v", path, err)
        return err
    }

    req.Header.Set("Authorization", "Bearer YOUR_ACCESS_TOKEN")
    req.Header.Set("Accept-Charset", "utf-8")
    req.Header.Set("Content-Type", "application/x-www-form-pathencoded")

    resp, err := client.Do(req)
    if err != nil {
        log.Printf("ERROR: Request %s => %v", path, err)
        return err
    }
    defer resp.Body.Close()

    if resp.StatusCode < 200 || resp.StatusCode >= 300 {
        err := errors.New("HTTP status: " + http.StatusText(resp.StatusCode))
        log.Printf("ERROR: %v", err)
        return err
    }

    return nil
}

GetInterfacesTypes

// GetInterfacesTypes ...
func GetInterfacesTypes(f interface{}) {
    switch vf := f.(type) {
    case map[string]interface{}:
        //fmt.Println("is a map:")
        for k, v := range vf {
            switch vv := v.(type) {
            case string:
                //fmt.Printf("%v: is string - %q\n", k, vv)
            case int:
                //fmt.Printf("%v: is int - %q\n", k, vv)
            case float64:
                //fmt.Printf("%v: is float64 - %g\n", k, vv)
            default:
                fmt.Sprintln(k, v, vv)
                //fmt.Printf("%v: ", k)
                GetInterfacesTypes(v)
            }
        }
    case []interface{}:
        //fmt.Println("is an array:")
        for k, v := range vf {
            switch vv := v.(type) {
            case string:
                //fmt.Printf("%v: is string - %q\n", k, vv)
            case int:
                //fmt.Printf("%v: is int - %q\n", k, vv)
            case float64:
                //fmt.Printf("%v: is float64 - %g\n", k, vv)
                if k == 4 {
                    fmt.Println(`ALELUYA==>`, vv)
                }
            default:
                fmt.Sprintln(k, v, vv)
                //fmt.Printf("%v: ", k)
                GetInterfacesTypes(v)
            }
        }
    }
}

IsJSON

// IsJSON ...
func IsJSON(str string) bool {
    var js json.RawMessage
    return json.Unmarshal([]byte(str), &js) == nil
}

IsPointer

func IsPointer(d interface{}) bool {
    return reflect.TypeOf(d).Kind() == reflect.Ptr
}

FILES

ReadFile

// ReadFile ...
func ReadFile(filePath string) (string, error) {
    data, err := ioutil.ReadFile(filePath)
    if err != nil {
        return "", err
    }
    return string(data), nil
}

// ReadFileLineByLine ...
func ReadFileLineByLine(filePath string) ([]string, error) {
    file, err := os.Open(filePath)
    if err != nil {
        return nil, err
    }
    defer file.Close()

    var lines []string
    scanner := bufio.NewScanner(file)
    for scanner.Scan() {
        lines = append(lines, scanner.Text())
    }

    if err := scanner.Err(); err != nil {
        return nil, err
    }

    return lines, nil
}

WriteFile

// WriteFile ...
func WriteFile(filePath string, content string) error {
    file, err := os.Create(filePath)
    if err != nil {
        return err
    }
    defer file.Close()

    _, err = file.WriteString(content)
    if err != nil {
        return err
    }

    return nil
}

LoadJSON from File

// LoadJSONFIleDecoder ... use streams
func LoadJSONFileDecoder(filePath string, data interface{}) error {
    file, err := os.Open(filePath)
    if err != nil {
        return err
    }
    defer file.Close()

    decoder := json.NewDecoder(file)
    err = decoder.Decode(data)
    if err != nil {
        return err
    }

    return nil
}

// LoadJSONFileMarschall ... loads all
func LoadJSONFileMarshall(filePath string, data interface{}) error {
    file, err := os.Open(filePath)
    if err != nil {
        return err
    }
    defer file.Close()

    body, err := ioutil.ReadAll(file)
    if err != nil {
        return err
    }

    err = json.Unmarshal(body, data)
    if err != nil {
        return err
    }

    return nil
}

WriteJSONtoFile

// WriteJSONtoFile ...
func WriteJSONtoFile(filePath string, d interface{}) error {
    f, err := os.Create(filePath)
    if err != nil {
        return err
    }
    defer f.Close()

    encoder := json.NewEncoder(f)
    err = encoder.Encode(d)
    if err != nil {
        return err
    }

    return nil
}

DownloadFile

// DownloadFile ...
func DownloadFile(filePath string, url string) error {
    out, err := os.Create(filePath)
    if err != nil {
        return err
    }
    defer out.Close()

    resp, err := http.Get(url)
    if err != nil {
        return err
    }
    defer resp.Body.Close()

    if resp.StatusCode != http.StatusOK {
        return fmt.Errorf("ERROR Downloading file: %d", resp.StatusCode)
    }

    _, err = io.Copy(out, resp.Body)
    if err != nil {
        return err
    }

    return nil
}

SendFileFromServerToClient

func Index(w http.ResponseWriter, r *http.Request) {
    url := "http://upload.wikimedia.org/wikipedia/en/b/bc/Wiki.png"

    timeout := time.Duration(5) * time.Second
    transport := &http.Transport{
        ResponseHeaderTimeout: timeout,
        Dial: func(network, addr string) (net.Conn, error) {
            return net.DialTimeout(network, addr, timeout)
        },
        DisableKeepAlives: true,
    }
    client := &http.Client{
        Transport: transport,
    }
    resp, err := client.Get(url)
    if err != nil {
        fmt.Println(err)
    }
    defer resp.Body.Close()

    //copy the relevant headers. If you want to preserve the downloaded 
    // file name, extract it with go's url parser.
    w.Header().Set("Content-Disposition", "attachment; filename=Wiki.png")
    w.Header().Set("Content-Type", r.Header.Get("Content-Type"))
    w.Header().Set("Content-Length", r.Header.Get("Content-Length"))

    //stream the body to the client without fully loading it into memory
    io.Copy(w, resp.Body)
}

func main() {
    http.HandleFunc("/", Index)
    err := http.ListenAndServe(":8000", nil)

    if err != nil {
        fmt.Println(err)
    }
}

ParseCSVFile

// file.csv
"AAPL","Apple Inc","4.10%"
"AMZN","Amazon.com Inc","3.49%"
"MSFT","Microsoft Corp","3.23%"
"GOOGL","Alphabet Inc","3.09%"
type stock struct {
    Symbol string `json:"symbol"`
}

func main() {
    csvFile, _ := os.Open("sp.csv")
    reader := csv.NewReader(bufio.NewReader(csvFile))
    var stocks []stock
    for {
        line, error := reader.Read()
        if error == io.EOF {
            break
        }
        aux := stock{
            Symbol: line[0],
        }
        stocks = append(stocks, aux)
    }
    //stocksJSON, _ := json.Marshal(stocks)
    //fmt.Println(string(stocksJSON))

    f, err := os.Create("spList.json")
    if err != nil {
        panic(err)
    }
    defer f.Close()
    e := json.NewEncoder(f)
    e.Encode(&stocks)
    fmt.Println(`END`)
}
// result
[
  {
    "symbol": "AAPL"
  },
  {
    "symbol": "AMZN"
  },
  {
    "symbol": "MSFT"
  },
  {
    "symbol": "GOOGL"
    }
]

HTTP SERVER

Lectura

Request Handling in Go

type Handler interface {
    ServeHttp( ResponseWriter, *Request )
}

Wrapper

Es muy sencillo pero luego complica para testearlo

mux.HandleFunc("/path", func(w http.ResponseWriter, r *http.Request) {
        nombreFuncion(w, r, loQueQueramosPasar)
})

Handle + HandleFunc

package main

import (
    "fmt"
    "net/http"
    "time"
)

func timeHandler1(w http.ResponseWriter, r *http.Request) {
    tm := time.Now().Format(time.RFC1123)
    fmt.Println("/time/" + tm)
    w.Write([]byte("The time is: " + tm))
}

func timeHandler2(format string) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request){
        tm := time.Now().Format(format)
        fmt.Println("/time/" + tm)
        w.Write([]byte("The time is: " + tm))
    })
}

/*  lo mismo pero con conversion implicita al tipo HandlerFunc
func timeHandler2(format string) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        tm := time.Now().Format(format)
        fmt.Println("/time/" + tm)
        w.Write([]byte("The time is: " + tm))
    }
}
*/

func hiHandler(w http.ResponseWriter, r *http.Request) {
    fmt.Println("/hello")
    w.Write([]byte("/hello"))
}

func main() {
    mux := http.NewServeMux()

    // Creamos un closure con las variables que queremos usar
    th2 := timeHandler2(time.RFC3339)

    mux.HandleFunc("/time/1", timeHandler1)
    mux.Handle("/time/2", th2)
    mux.HandleFunc("/hello", hiHandler)

    //http.HandleFunc("/time/1", timeHandler1)
    //http.Handle("/time/2", th2)
    //http.HandleFunc("/hello", hiHandler)

    http.ListenAndServe(":3000", mux /*nil*/)
}

Handler

type specificHandler struct {
    Thing string
}

func(h *specificHandler)ServeHTTP(w http.ResponseWriter,r *http.Request) {
    w.Write(h.Thing)
}

func main() {
  http.Handle("/something", &specificHandler{Thing: "Hello world!"})
  http.ListenAndServe(":8080", nil)
}
package main

import (
    "fmt"
    "net/http"
    "time"
)

type timeHandler struct {
    format string
}

func (th *timeHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    tm := time.Now().Format(th.format)
    fmt.Println("/time/" + tm)
    w.Write([]byte("The time is: " + tm))
}

type hiHandler struct{}

func (ti *hiHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    fmt.Println("/hello")
    w.Write([]byte("/hello"))
}

func main() {
    /mux := http.NewServeMux()

    th1 := &timeHandler{format: time.RFC1123}
    th2 := &timeHandler{format: time.RFC3339}
    hi := &hiHandler{}

    mux.Handle("/time/1", th1)
    mux.Handle("/time/2", th2)
    mux.Handle("/hello", hi)

    //http.Handle("/time/1", th1)
    //http.Handle("/time/2", th2)
    //http.Handle("/hello", hi)

    http.ListenAndServe(":3000", /*nil*/ mux)
}

Ejemplo Completo

package main

import (
    "errors"
    "fmt"
    "log"
    "net/http"
    "os"
    "time"

    _ "github.com/go-sql-driver/mysql"
)

type app struct {
    Config struct {
        Mode          string `json:"mode"`
        Port          int    `json:"port"`
        Valid         string `json:"valid"`
        ErrorsLogFile string `json:"errorsLogFile"`
        HitsLogFile   string `json:"hitsLogFile"`
    } `json:"config"`
    Mysql struct {
        User      string `json:"user"`
        Password  string `json:"password"`
        DB        string `json:"db"`
        Host      string `json:"host"`
        Port      int    `json:"port"`
        TableBW   string `json:"tableBw"`
        TableHits string `json:"tableHits"`
    }
}

type requestError struct {
    Error      error  `json:"-"`
    Message    string `json:"message"`
    StatusCode int    `json:"-"`
}

func (a *app) ServeHTTP(w http.ResponseWriter, r *http.Request) {
    r.ParseForm()
    valid := r.Form.Get("test")
    if valid != a.Config.Valid {
        http.Error(w, "Unauthorized", http.StatusUnauthorized)
        return
    }
    updateBw(w, r, a)
}

func main() {
    var a app
    loadConfigJSON(&a)
    checkMode(&a)

    // Custom Log File
    if a.Config.Mode == "production" {
        var f = a.Config.ErrorsLogFile
        mylog, err := os.OpenFile(f, os.O_WRONLY|os.O_CREATE|os.O_APPEND
                , 0644)
        if err != nil {
            log.Printf("ERROR opening log file %s\n", err)
        }
        defer mylog.Close() // defer must be in main
        log.SetOutput(mylog)
    }

    mux := http.NewServeMux()

    mux.Handle("/savebw/", &a)
    mux.Handle("/saveHits/", checkValid(
        func(w http.ResponseWriter, r *http.Request) {
            updateHits(w, r, &a)
        }, a.Config.Valid))
    mux.HandleFunc("/get/", checkValid(
        func(w http.ResponseWriter, r *http.Request) {
            getStats(w, r, &a)
        }, a.Config.Valid))
    mux.HandleFunc("/", badRequest)

    server := http.Server{
        Addr:           fmt.Sprintf("localhost:%d", a.Config.Port),
        Handler:        mux,
        ReadTimeout:    10 * time.Second,
        WriteTimeout:   30 * time.Second,
        MaxHeaderBytes: 1 << 20,
    }

    log.Printf("Server up listening %s in mode %s", server.Addr
            , a.Config.Mode)
    server.ListenAndServe()

}

func checkValid(next http.HandlerFunc, test string) http.HandlerFunc {
    return func(w http.ResponseWriter, r *http.Request) {
        r.ParseForm()
        valid := r.Form.Get("test")
        if valid != test {
            http.Error(w, "Unauthorized", http.StatusUnauthorized)
            return
        }
        next.ServeHTTP(w, r)
    }
}

func badRequest(w http.ResponseWriter, r *http.Request) {
    re := &requestError{
        Error:      errors.New("Unexistent Endpoint"),
        Message:    "Bad Request",
        StatusCode: 400,
    }
    sendErrorToClient(w, re)
}

NETWORK

GetIP

// GetIP ...
func GetIP(r *http.Request) string {
    headers := []string{
        "X-Forwarded-For",
        "X-Real-IP",
        "CF-Connecting-IP",
    }
    for _, header := range headers {
        ips := r.Header.Get(header)
        if ips != "" {
            return strings.TrimSpace(strings.Split(ips, ",")[0])
        }
    }
    ip := r.RemoteAddr
    colon := strings.LastIndex(ip, ":")
    if colon != -1 {
        ip = ip[:colon]
    }
    return strings.TrimSpace(ip)
}

GetRequestOrigin

func GetRequestOrigin(r *http.Request) string {
    switch {
    case r.Header.Get("Host") != "":
        return r.Header.Get("Host")
    case r.Header.Get("Origin") != "":
        return r.Header.Get("Origin")
    case r.Header.Get("Referer") != "":
        return r.Header.Get("Referer")
    default:
        return "?????"
    }
}

IsValidURL

// IsValidURL ...
func IsValidURL(rawurl string) bool {
    rawurl = strings.TrimSpace(rawurl) 
    parsedURL, err := url.Parse(rawurl)
    if err != nil {
        return false
    }   
    if parsedURL.Scheme != "http" && parsedURL.Scheme != "https" {
        return false
    }
    if parsedURL.Host == "" {
        return false
    }
    return true
}

ExistsURL

// ExistsURL ...
func ExistsURL(myUrl string) bool {
    client := http.Client{
        Timeout: 5 * time.Second,
        CheckRedirect: func(req *http.Request, via []*http.Request) error {
            return http.ErrUseLastResponse // Dont follow redirections
        },
    }

    resp, err := client.Head(myUrl)
    if err != nil {
        return false
    }
    defer resp.Body.Close()

    return resp.StatusCode >= 200 && resp.StatusCode < 400
}

GetLanguage

// GetLanguage ...
func GetLanguage(r *http.Request) string {
    lang := r.Header.Get("Accept-Language")
    if lang != "" {
        langs := strings.SplitN(lang, ",", 2)
        return strings.ToLower(strings.TrimSpace(langs[0]))
    }
    return "en"
}

RemoveProtocolFromURL

// RemoveProtocol ...
func RemoveProtocol(url string) string {
    if strings.HasPrefix(url, "https://") {
        return url[8:]
    }
    if strings.HasPrefix(url, "http://") {
        return url[7:]
    }
    return url
}

RemoveProtocolAndWWWFromURL

// RemoveProtocolAndWWW ...
func RemoveProtocolAndWWWL(url string) string {
    url = RemoveProtocol(url)
    if strings.HasPrefix(url, "www.") {
        return url[4:]
    }
    return url
}

Nginx return 444

func close(w http.ResponseWriter, r *http.Request) {
    hijacker, ok := w.(http.Hijacker)
    if !ok {
        http.Error(w, "Server does not support hijacking", http.StatusInternalServerError)
        return
    }
    conn, _, err := hijacker.Hijack()
    if err != nil {
        http.Error(w, err.Error(), http.StatusInternalServerError)
        return
    }
    conn.Close()
    return
}

Slow Response

func slowSend(w http.ResponseWriter, r *http.Request) {
    flusher, ok := w.(http.Flusher)
    if !ok {
        err := "Server does not support flusher"
        http.Error(w, err, http.StatusInternalServerError)
        return
    }

    w.Header().Set("Content-Type", "text/plain")
    fmt.Fprintln(w, "Initiating slow response...")

    for i := 0; i < 100; i++ {
        fmt.Fprintf(w, "Fragmento %d\n", i+1)
        flusher.Flush()
        time.Sleep(1 * time.Second)
    }

    fmt.Fprintln(w, "Task accomplished")
}

func holdConn(w http.ResponseWriter, r *http.Request) {
    fmt.Fprintf(w, "Begin.............")
    time.Sleep(60 * time.Second)
    fmt.Fprintf(w, "Thats all")
}

NUMBERS

GetRandomInt

import (
    "time"

    "golang.org/x/exp/rand"
)

// RandomInt ... min and max included
func RandomInt(min, max int) int {
    r := rand.New(rand.NewSource(time.Now().UnixNano()))
    return r.Intn(max-min+1) + min
}

RoundFloat64

// RoundFloat64 ... rounds float64 into integer
func RoundFloat64(num float64) int {
    if num < 0 {
        return int(num - 0.5)
    }
    return int(num + 0.5)
}

RoundFloat32

// RoundFloat32 ... rounds float32 into integer
func RoundFloat32(num float32) int {
    if num < 0 {
        return int(num - 0.5)
    }
    return int(num + 0.5)
}

ReverseSliceInt

// ReverseSliceInt ... [0,1,2,3,4,5] ==> [5,4,3,2,1,0]
func ReverseSliceInt(reverse []int) []int {
    for i, j := 0, len(reverse)-1; i < j; i, j = i+1, j-1 {
        reverse[i], reverse[j] = reverse[j], reverse[i]
    }
    return reverse
}

TransposeMatrixInt

// TransposeMatrixInt ... rows > cols or cols > rows
// but rows.elements >= cols.elements
func TransposeMatrixInt(matrix [][]int) [][]int {
    if len(matrix) == 0 {
        return [][]int{}
    }

    result := make([][]int, len(matrix[0]))
    for i := range result {
        result[i] = make([]int, len(matrix))
    }

    for y, row := range matrix {
        for x, value := range row {
            result[x][y] = value
        }
    }
    return result
}

SliceContainsInt

// SliceContainsInt ... returns true/false
func SliceContainsInt(num int, slice []int) bool {
    for _, v := range slice {
        if v == num {
            return true
        }
    }
    return false
}

STRINGS

RemoveAllWhitespaces

import "strings"

// RemoveAllWhitespaces ...
func RemoveAllWhitespaces(str string) string {
    return strings.ReplaceAll(str, " ", "")
}

ReplaceAllWhitespacesByChar

import "strings"

// ReplaceAllWhitespacesByChar ...
func ReplaceAllWhitespacesByChar(str, otherChar string) string {
    return strings.ReplaceAll(str, " ", otherChar)
}

ReverseSliceString

// ReverseSliceString ["H","O","L","A"] ==> ["A","L","O","H"]
func ReverseSliceString(reverse []string) []string {
    for i, j := 0, len(reverse)-1; i < j; i, j = i+1, j-1 {
        reverse[i], reverse[j] = reverse[j], reverse[i]
    }
    return reverse
}

TransposeMatrixString

// TransposeMatrixString rows > cols or cols > rows
// but rows.elements >= cols.elements
func TransposeMatrixString(matrix [][]string) [][]string {
    if len(matrix) == 0 || len(matrix[0]) == 0 {
        return [][]string{}
    }

    result := make([][]string, len(matrix[0]))
    for i := range result {
        result[i] = make([]string, len(matrix))
    }
    for y, row := range matrix {
        for x, value := range row {
            result[x][y] = value
        }
    }
    return result
}

SliceContainsString

// SliceContainsString ... returns true/false
func SliceContainsString(str string, slice []string) bool {
    for _, v := range slice {
        if v == str {
            return true
        }
    }
    return false
}

OS

execCommand

import (
    "fmt"
    "os/exec"
)

func main()  {
    command := []string{"vnstat", "-i", ifinterface, "--json"}
    ///fmt.Println("Command =>", command)
    chunk, err := execCommand(command)
    if err != nil {
        log.Fatal(err)
    }
    //fmt.Println(`CHUNK =>`, string(chunk))
}

func execCommand(args []string) (err error) {
    _, err = exec.Command(args[0], args[1:]...).CombinedOutput()
    if err != nil {
        fmt.Println(err)
        return err
    }
    return err
}

func execCommand(args []string) (c []byte, err error) {
    c, err = exec.Command(args[0], args[1:]...).CombinedOutput()
    if err != nil {
        return nil, err
    }
    return c, err
}

func execCommand(comm string) {
    _, err := exec.Command("sh", "-c", comm).CombinedOutput()
    if err != nil {
        log.Fatal(err)
    }
}

TIME

ParseStringToTime

import "time"

// ParseStringToTime ...
func ParseStringToTime(start string) (time.Time, error) {
    formats := []string{
        time.RFC3339,
        "2006-01-02T15:04:05",
        "2006-01-02",
    }

    for _, layout := range formats {
        t, err := time.Parse(layout, start)
        if err == nil {
            return t, nil
        }
    }

    fail := fmt.Errorf("Fail parsing time: %s", start)
    return time.Time{}, fail
}

OnceADayTask

func main() {
    go onceADayTask(3, 10, 10)
    select {}
}

func onceADayTask(h, m, s int) {
    t := time.Now()
    n := time.Date(
        t.Year(), t.Month(), t.Day(),
        h, m, s, 0,
        t.Location(),
    )
    if n.Before(t) {
        n = n.Add(24 * time.Hour)
    }

    for {
        time.Sleep(time.Until(n))
        doSomeTask()
        n = n.Add(24 * time.Hour)
    }
}

func doSomeTask() {
    fmt.Printf("Hi: %s\n", time.Now().Format("03:04:05 PM"))
}

SetInterval

func main() {
    ticker := time.NewTicker(1 * time.Second)
    defer ticker.Stop()
    done := make(chan bool)
    go func() {
        for {
            select {
            case <-ticker.C:
                fmt.Println("Ticker every X * time.Second")
            case <-done:
                return
            }
        }
    }()
    time.Sleep(5 * time.Second)
    done <- true
    fmt.Println("Ticker stopped. Exiting program.")
}
func main() {
    go interval()
    select {}
}
func interval() {
    ticker := time.NewTicker(2 * time.Second)
    defer ticker.Stop()

    for range ticker.C {
        fmt.Println("Hi Every 2 secs")
    }
}

LOGS

Custom Logs

// main.go
/////// Custom Error Log File + Custom Info Log File /////////
iLog := createCustomInfoLogFile2(a.Conf.InfoLogFile)
mylog := createCustomErrorLogFile(a.Conf.ErrorsLogFile)
defer mylog.Close()
//////////////////////////////////////////////////////////////

// ya por donde queramos
func createCustomErrorLogFile(f string) *os.File {
    mylog,err:=os.OpenFile(f,os.O_WRONLY|os.O_CREATE|os.O_APPEND,0644)
    if err != nil {
        log.Fatalf("ERROR opening Error log file %s\n", err)
    }
    log.SetOutput(mylog)
    return mylog
}

func createCustomInfoLogFile2(f string) *log.Logger {
    infoLog,err:=os.OpenFile(f,os.O_WRONLY|os.O_CREATE|os.O_APPEND,0644)
    if err != nil {
        log.Fatalf("ERROR opening Info log file %s\n", err)
    }
    var iLog *log.Logger
    iLog = log.New(infoLog, "INFO :\t", log.Ldate|log.Ltime)
    return iLog
}
const (
    errorLogFile = "error.log"
    infoLogFile  = "info.log"
)

func openLogFile(filename string) (*os.File, error) {
    return os.OpenFile(
        filename,
        os.O_CREATE|os.O_WRONLY|os.O_APPEND,
        0664,
    )
}

func main() {
    errorFile, err := openLogFile(errorLogFile)
    if err != nil {
        log.Fatalf("ERROR opening error log: %s: %v", errorLogFile, err)
    }
    defer errorFile.Close()

    infoFile, err := openLogFile(infoLogFile)
    if err != nil {
        log.Fatalf("ERRRO opening info log: %s: %v", infoLogFile, err)
    }
    defer infoFile.Close()

    eLog :=
        log.New(errorFile, "ERROR: ", log.Ldate|log.Ltime|log.Lshortfile)
    iLog :=
        log.New(infoFile, "INFO: ", log.Ldate|log.Ltime)

    eLog.Println("ERROR")
    iLog.Println("INFO")
}

PrettyPrint Structs

func prettyPrintStruct(s interface{}) {
    result, _ := json.MarshalIndent(s, "", "\t")
    fmt.Print(string(result), "\n")
}

FLAGS

Binarios con versiones

package main

import (
    "flag"
    "fmt"
    "os"
)

var version = "0.0.0"
var when = "undefined"

func main() {
    checkFlags()
    fmt.Println("Continue...")
}

func checkFlags() {
    versionFlag := flag.Bool("v", false, "Show current version and exit")
    flag.Parse()
    switch {
    case *versionFlag:
        fmt.Printf("Version:\t: %s\n", version)
        fmt.Printf("Date   :\t: %s\n", when)
        os.Exit(0)
    }
}

/*
go build  -ldflags="
-X 'main.version=v0.2.0' 
-X 'main.when=$(date -u +%F_%T)'"

go build  -ldflags="-X 'main.when=$(date -u +%F_%T)'"

luego podemos hacer ./binary -v
*/

Args + Flags + test

#go build -ldflags="-X 'main.when=$(date -u +%F_%T)'"
DATE=$(shell date -u +%F_%T)
LDFLAGS=-ldflags "-X main.when=$(DATE)"

all: build

build:
    go build $(LDFLAGS) 

clean:
    rm binary
// main.go
var version = "0.0.1"
var when = ""

func main() {
    tasks := checkFlags()

    fmt.Println("USER => ", getUserName())
    fmt.Println("TASKS=", tasks)
}

func checkFlags() []string {
    if len(os.Args) > 1 && os.Args[1] == "-v" {
        versionFlag := flag.Bool("v", false, "Show Version")
        flag.Parse()
        if *versionFlag {
            fmt.Printf("Version ->\t%s\n", version)
            fmt.Printf("Date    ->\t%s\n", when)
            os.Exit(0)
        }
    }
    var result []string
    validFlags := []string{"vnstat", "www", "cloud", "mysql", "rsync"}

    for _, arg := range os.Args[1:] {
        if strings.HasPrefix(arg, "-") {
            flag := strings.TrimPrefix(arg, "-")
            for _, validFlag := range validFlags {
                if flag == validFlag {
                    if !includes(result, validFlag) {
                        result = append(result, validFlag)
                    }
                }
            }
        }
    }
    return result
}

func getUserName() string {
    currentUser, err := user.Current()
    if err != nil {
        fmt.Println("Error:", err)
        os.Exit(0)
    }
    return currentUser.Username
}

func includes(slice []string, element string) bool {
    for _, v := range slice {
        if v == element {
            return true
        }
    }
    return false
}

// main_test.go
func TestCheckFlags(t *testing.T) {
    tests := []struct {
        name     string
        args     []string
        expected []string
    }{
        {"Test vnstat flag", []string{"cmd", "-vnstat"},
            []string{"vnstat"}},
        {"Test www flag", []string{"cmd", "-www"},
            []string{"www"}},
        {"Test cloud flag", []string{"cmd", "-cloud"},
            []string{"cloud"}},
        {"Test mysql flag", []string{"cmd", "-mysql"},
            []string{"mysql"}},
        {"Test rsync flag", []string{"cmd", "-rsync"},
            []string{"rsync"}},
        {"Test vnstat and mysql flags",
          []string{"cmd", "-vnstat", "-mysql"},
            []string{"vnstat", "mysql"}},
        {"Test no flags", []string{"cmd"}, []string{}},
        {"Test repeated flags", []string{"cmd", "-www", "-www"},
            []string{"www"}},
        {"Test unknown flag", []string{"cmd", "-unknown"},
            []string{}},
        {"Test multiple unknown flags", 
        []string{"cmd", "-unknown", "-anotherunknown"}, 
        []string{}},
        {"Test multiple unknown and known flags",
          []string{"cmd", "-www", "-unknown", "-rsync", "-anotherunk"},
            []string{"www", "rsync"}},
    }

    for _, tt := range tests {
        t.Run(tt.name, func(t *testing.T) {
            os.Args = tt.args
            flag.CommandLine = flag.NewFlagSet(os.Args[0], flag.ExitOnError)
            result := checkFlags()
            if len(result) != len(tt.expected) {
                t.Errorf("expected length %d, got %d",
                 len(tt.expected), len(result))
            }
            for i, v := range tt.expected {
                if result[i] != v {
                    t.Errorf("expected %s at index %d, got %s", 
                    v, i, result[i])
                }
            }
        })
    }
}



ORGANIZACION DE CODIGO

Compartir structs entre paquetes

// main.go
package main

import (
    s "pruebas/secondarypkg"
    "time"
)

func main() {
    p := &s.Placeholder{
        Name: "FooBar",
        Date: time.Now().String(),
    }
    s.Foo(p)
}

// secondarypkg/otro.go
package secondarypkg

import "fmt"

type Placeholder struct {
    Name string
    Date string
}

func Foo(p *Placeholder) {
    fmt.Println(p.Date, p.Name)
}
// main.go
package main

import (
    s "pruebas/paquete"
    "time"
)
func main() {
    p := s.NewPlaceHolder("FooBar", time.Now().String())
    p.Foo()
}

// secondarypkg/otro.go
package secondarypkg

import "fmt"

type Placeholder struct {
    Name string
    Date string
}

func (p *Placeholder) Foo() {
    fmt.Println(p.Date, p.Name)
}

func NewPlaceHolder(name string, date string) *Placeholder {
    return &Placeholder{
        Name: name,
        Date: date,
    }
}

Lo mismo usando interfaces

// main.go
package main

import (
    s "pruebas/paquete"
    "time"
)

func main() {
    p := s.NewPlaceHolder("FooBar", time.Now().String())
    p.Foo()
}

// secondarypkg/otro.go
package secondarypkg

import "fmt"

type PlaceHolder interface {
    Foo()
}

type placeholder struct {
    Name string
    Date string
}

func (p *placeholder) Foo() {
    fmt.Println(p.Date, p.Name)
}

func NewPlaceHolder(name string, date string) PlaceHolder {
    return &placeholder{
        Name: name,
        Date: date,
    }
}