Skip to content
/ yiigo Public
forked from shenghui0779/yiigo

๐Ÿ”ฅ ไธ€ไธชๅฅฝ็”จ็š„่ฝป้‡็บง Go ๅผ€ๅ‘้€š็”จๅบ“ ๐Ÿš€๐Ÿš€๐Ÿš€

License

Notifications You must be signed in to change notification settings

kemiboy/yiigo

ย 
ย 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 
ย 

Repository files navigation

yiigo

golang GitHub release pkg.go.dev Apache 2.0 license

ไธ€ไธชๅฅฝ็”จ็š„่ฝป้‡็บง Go ๅผ€ๅ‘้€š็”จๅบ“ใ€‚ๅฆ‚ๆžœไฝ ไธๅ–œๆฌข่ฟ‡ๅบฆๅฐ่ฃ…็š„้‡้‡็บงๆก†ๆžถ๏ผŒ่ฟ™ไธชๅบ“ๅฏ่ƒฝๆ˜ฏไธชไธ้”™็š„้€‰ๆ‹ฉ ๐Ÿ˜Š

Features

  • ๆ”ฏๆŒ MySQL
  • ๆ”ฏๆŒ PostgreSQL
  • ๆ”ฏๆŒ SQLite3
  • ๆ”ฏๆŒ MongoDB
  • ๆ”ฏๆŒ Redis
  • ๆ”ฏๆŒ NSQ
  • SQLไฝฟ็”จ sqlx
  • ORMๆŽจ่ ent
  • ๆ—ฅๅฟ—ไฝฟ็”จ zap
  • ้…็ฝฎไฝฟ็”จ dotenv๏ผŒๆ”ฏๆŒ๏ผˆๅŒ…ๆ‹ฌ k8s configmap๏ผ‰็ƒญๅŠ ่ฝฝ
  • ๅ…ถไป–
    • ่ฝป้‡็š„ SQL Builder
    • ๅŸบไบŽ Redis ็š„็ฎ€ๅ•ๅˆ†ๅธƒๅผ้”
    • Websocket ็ฎ€ๅ•ไฝฟ็”จๅฐ่ฃ…๏ผˆๆ”ฏๆŒๆŽˆๆƒๆ ก้ชŒ๏ผ‰
    • ็ฎ€ๆ˜“็š„ๅ•ๆ—ถ้—ด่ฝฎ๏ผˆๆ”ฏๆŒไธ€ๆฌกๆ€งๅ’Œๅคšๆฌก้‡่ฏ•ไปปๅŠก๏ผ‰
    • ๅฎž็”จ็š„่พ…ๅŠฉๆ–นๆณ•๏ผŒๅŒ…ๅซ๏ผšhttpใ€cyptoใ€dateใ€IPใ€validatorใ€version compare ็ญ‰

Installation

go get -u github.com/shenghui0779/yiigo

Usage

ENV

  • load
// ้ป˜่ฎคๅŠ ่ฝฝๅฝ“ๅ‰็›ฎๅฝ•ไธ‹็š„`.env`ๆ–‡ไปถ
yiigo.LoadEnv()

// ๅŠ ่ฝฝๆŒ‡ๅฎš้…็ฝฎๆ–‡ไปถ
yiigo.LoadEnv(yiigo.WithEnvFile("mycfg.env"))

// ็ƒญๅŠ ่ฝฝ
yiigo.LoadEnv(yiigo.WithEnvWatcher(func(e fsnotify.Event) {
    fmt.Println(e.String())
}))
  • .env
ENV=dev
  • usage
fmt.Println(os.Getenv("ENV"))
// output: dev

DB

  • register
yiigo.Init(
    yiigo.WithMySQL(yiigo.Default, &yiigo.DBConfig{
        DSN: "dsn",
        Options: &yiigo.DBOptions{
            MaxOpenConns:    20,
            MaxIdleConns:    10,
            ConnMaxLifetime: 10 * time.Minute,
            ConnMaxIdleTime: 5 * time.Minute,
        },
    }),

    yiigo.WithMySQL("other", &yiigo.DBConfig{
        DSN: "dsn",
        Options: &yiigo.DBOptions{
            MaxOpenConns:    20,
            MaxIdleConns:    10,
            ConnMaxLifetime: 10 * time.Minute,
            ConnMaxIdleTime: 5 * time.Minute,
        },
    }),
)
  • sqlx
// default db
yiigo.DB().Get(&User{}, "SELECT * FROM user WHERE id = ?", 1)

// other db
yiigo.DB("other").Get(&User{}, "SELECT * FROM user WHERE id = ?", 1)
  • ent
import "<your_project>/ent"

// default driver
client := ent.NewClient(ent.Driver(yiigo.EntDriver()))

// other driver
client := ent.NewClient(ent.Driver(yiigo.EntDriver("other")))

MongoDB

// register
yiigo.Init(
    yiigo.WithMongo(yiigo.Default, "dsn"),
    yiigo.WithMongo("other", "dsn"),
)

// default mongodb
yiigo.Mongo().Database("test").Collection("numbers").InsertOne(context.Background(), bson.M{"name": "pi", "value": 3.14159})

// other mongodb
yiigo.Mongo("other").Database("test").Collection("numbers").InsertOne(context.Background(), bson.M{"name": "pi", "value": 3.14159})

Redis

// register
yiigo.Init(
    yiigo.WithRedis(yiigo.Default, &yiigo.RedisConfig{
        Addr: "addr",
        Options: &yiigo.RedisOptions{
            ConnTimeout:  10 * time.Second,
            ReadTimeout:  10 * time.Second,
            WriteTimeout: 10 * time.Second,
            PoolSize:     10,
            IdleTimeout:  5 * time.Minute,
        },
    }),

    yiigo.WithRedis("other", &yiigo.RedisConfig{
        Addr: "addr",
        Options: &yiigo.RedisOptions{
            ConnTimeout:  10 * time.Second,
            ReadTimeout:  10 * time.Second,
            WriteTimeout: 10 * time.Second,
            PoolSize:     10,
            IdleTimeout:  5 * time.Minute,
        },
    }),
)

// default redis
yiigo.Redis().Do(context.Background(), "SET", "test_key", "hello world")

yiigo.Redis().DoFunc(context.Background(), func(ctx context.Context, conn *RedisConn) error {
    if _, err := conn.Do("SET", "key1", "hello"); err != nil {
        return err
    }

    if _, err := conn.Do("SET", "key2", "world"); err != nil {
        return err
    }

    return nil
})

// other redis
yiigo.Redis("other").Do(context.Background(), "SET", "test_key", "hello world")

yiigo.Redis("other").DoFunc(context.Background(), func(ctx context.Context, conn *RedisConn) error {
    if _, err := conn.Do("SET", "key1", "hello"); err != nil {
        return err
    }

    if _, err := conn.Do("SET", "key2", "world"); err != nil {
        return err
    }

    return nil
})

Logger

// register
yiigo.Init(
    yiigo.WithLogger(yiigo.Default, yiigo.LoggerConfig{
        Filename: "filename",
        Options: &yiigo.LoggerOptions{
            Stderr: true,
        },
    }),

    yiigo.WithLogger("other", yiigo.LoggerConfig{
        Filename: "filename",
        Options: &yiigo.LoggerOptions{
            Stderr: true,
        },
    }),
)

// default logger
yiigo.Logger().Info("hello world")

// other logger
yiigo.Logger("other").Info("hello world")

HTTP

// default client
yiigo.HTTPGet(context.Background(), "URL")

// new client
client := yiigo.NewHTTPClient(*http.Client)
client.Do(context.Background(), http.MethodGet, "URL", nil)

// upload
form := yiigo.NewUploadForm(
    yiigo.WithFormField("title", "TITLE"),
    yiigo.WithFormField("description", "DESCRIPTION"),
    yiigo.WithFormFile("media", "demo.mp4", func(w io.Writer) error {
        f, err := os.Open("demo.mp4")

        if err != nil {
            return err
        }

        defer f.Close()

        if _, err = io.Copy(w, f); err != nil {
            return err
        }

        return nil
    }),
)

yiigo.HTTPUpload(context.Background(), "URL", form)

SQL Builder

๐Ÿ˜Š ไธบไธๆƒณๆ‰‹ๅ†™SQL็š„ไฝ ็”ŸๆˆSQL่ฏญๅฅ๏ผŒ็”จไบŽ sqlx ็š„็›ธๅ…ณๆ–นๆณ•๏ผ›

โš ๏ธ ไฝœไธบ่พ…ๅŠฉๆ–นๆณ•๏ผŒ็›ฎๅ‰ๆ”ฏๆŒ็š„็‰นๆ€งๆœ‰้™๏ผŒๅคๆ‚็š„SQL๏ผˆๅฆ‚๏ผšๅญๆŸฅ่ฏข็ญ‰๏ผ‰่ฟ˜้œ€่‡ชๅทฑๆ‰‹ๅ†™

builder := yiigo.NewMySQLBuilder()
// builder := yiigo.NewSQLBuilder(yiigo.MySQL)
  • Query
ctx := context.Background()

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Where("id = ?", 1),
).ToQuery(ctx)
// SELECT * FROM user WHERE id = ?
// [1]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Where("name = ? AND age > ?", "shenghui0779", 20),
).ToQuery(ctx)
// SELECT * FROM user WHERE name = ? AND age > ?
// [shenghui0779 20]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.WhereIn("age IN (?)", []int{20, 30}),
).ToQuery(ctx)
// SELECT * FROM user WHERE age IN (?, ?)
// [20 30]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Select("id", "name", "age"),
    yiigo.Where("id = ?", 1),
).ToQuery(ctx)
// SELECT id, name, age FROM user WHERE id = ?
// [1]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Distinct("name"),
    yiigo.Where("id = ?", 1),
).ToQuery(ctx)
// SELECT DISTINCT name FROM user WHERE id = ?
// [1]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.LeftJoin("address", "user.id = address.user_id"),
    yiigo.Where("user.id = ?", 1),
).ToQuery(ctx)
// SELECT * FROM user LEFT JOIN address ON user.id = address.user_id WHERE user.id = ?
// [1]

builder.Wrap(
    yiigo.Table("address"),
    yiigo.Select("user_id", "COUNT(*) AS total"),
    yiigo.GroupBy("user_id"),
    yiigo.Having("user_id = ?", 1),
).ToQuery(ctx)
// SELECT user_id, COUNT(*) AS total FROM address GROUP BY user_id HAVING user_id = ?
// [1]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Where("age > ?", 20),
    yiigo.OrderBy("age ASC", "id DESC"),
    yiigo.Offset(5),
    yiigo.Limit(10),
).ToQuery(ctx)
// SELECT * FROM user WHERE age > ? ORDER BY age ASC, id DESC LIMIT ? OFFSET ?
// [20, 10, 5]

wrap1 := builder.Wrap(
    Table("user_1"),
    Where("id = ?", 2),
)

builder.Wrap(
    Table("user_0"),
    Where("id = ?", 1),
    Union(wrap1),
).ToQuery(ctx)
// (SELECT * FROM user_0 WHERE id = ?) UNION (SELECT * FROM user_1 WHERE id = ?)
// [1, 2]

builder.Wrap(
    Table("user_0"),
    Where("id = ?", 1),
    UnionAll(wrap1),
).ToQuery(ctx)
// (SELECT * FROM user_0 WHERE id = ?) UNION ALL (SELECT * FROM user_1 WHERE id = ?)
// [1, 2]

builder.Wrap(
    Table("user_0"),
    WhereIn("age IN (?)", []int{10, 20}),
    Limit(5),
    Union(
        builder.Wrap(
            Table("user_1"),
            Where("age IN (?)", []int{30, 40}),
            Limit(5),
        ),
    ),
).ToQuery(ctx)
// (SELECT * FROM user_0 WHERE age IN (?, ?) LIMIT ?) UNION (SELECT * FROM user_1 WHERE age IN (?, ?) LIMIT ?)
// [10, 20, 5, 30, 40, 5]
  • Insert
ctx := context.Background()

type User struct {
    ID     int    `db:"-"`
    Name   string `db:"name"`
    Age    int    `db:"age"`
    Phone  string `db:"phone,omitempty"`
}

builder.Wrap(Table("user")).ToInsert(ctx, &User{
    Name: "yiigo",
    Age:  29,
})
// INSERT INTO user (name, age) VALUES (?, ?)
// [yiigo 29]

builder.Wrap(yiigo.Table("user")).ToInsert(ctx, yiigo.X{
    "name": "yiigo",
    "age":  29,
})
// INSERT INTO user (name, age) VALUES (?, ?)
// [yiigo 29]
  • Batch Insert
ctx := context.Background()

type User struct {
    ID     int    `db:"-"`
    Name   string `db:"name"`
    Age    int    `db:"age"`
    Phone  string `db:"phone,omitempty"`
}

builder.Wrap(Table("user")).ToBatchInsert(ctx, []*User{
    {
        Name: "shenghui0779",
        Age:  20,
    },
    {
        Name: "yiigo",
        Age:  29,
    },
})
// INSERT INTO user (name, age) VALUES (?, ?), (?, ?)
// [shenghui0779 20 yiigo 29]

builder.Wrap(yiigo.Table("user")).ToBatchInsert(ctx, []yiigo.X{
    {
        "name": "shenghui0779",
        "age":  20,
    },
    {
        "name": "yiigo",
        "age":  29,
    },
})
// INSERT INTO user (name, age) VALUES (?, ?), (?, ?)
// [shenghui0779 20 yiigo 29]
  • Update
ctx := context.Background()

type User struct {
    Name   string `db:"name"`
    Age    int    `db:"age"`
    Phone  string `db:"phone,omitempty"`
}

builder.Wrap(
    Table("user"),
    Where("id = ?", 1),
).ToUpdate(ctx, &User{
    Name: "yiigo",
    Age:  29,
})
// UPDATE user SET name = ?, age = ? WHERE id = ?
// [yiigo 29 1]

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Where("id = ?", 1),
).ToUpdate(ctx, yiigo.X{
    "name": "yiigo",
    "age":  29,
})
// UPDATE user SET name = ?, age = ? WHERE id = ?
// [yiigo 29 1]

builder.Wrap(
    yiigo.Table("product"),
    yiigo.Where("id = ?", 1),
).ToUpdate(ctx, yiigo.X{
    "price": yiigo.Clause("price * ? + ?", 2, 100),
})
// UPDATE product SET price = price * ? + ? WHERE id = ?
// [2 100 1]
  • Delete
ctx := context.Background()

builder.Wrap(
    yiigo.Table("user"),
    yiigo.Where("id = ?", 1),
).ToDelete(ctx)
// DELETE FROM user WHERE id = ?
// [1]

builder.Wrap(Table("user")).ToTruncate(ctx)
// TRUNCATE user

Documentation

Enjoy ๐Ÿ˜Š

About

๐Ÿ”ฅ ไธ€ไธชๅฅฝ็”จ็š„่ฝป้‡็บง Go ๅผ€ๅ‘้€š็”จๅบ“ ๐Ÿš€๐Ÿš€๐Ÿš€

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • Go 100.0%