ไธไธชๅฅฝ็จ็่ฝป้็บง Go ๅผๅ้็จๅบใๅฆๆไฝ ไธๅๆฌข่ฟๅบฆๅฐ่ฃ ็้้็บงๆกๆถ๏ผ่ฟไธชๅบๅฏ่ฝๆฏไธชไธ้็้ๆฉ ๐
- ๆฏๆ 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 ็ญ
go get -u github.com/shenghui0779/yiigo
- 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
- 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")))
// 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})
// 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
})
// 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")
// 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็ไฝ ็ๆ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
Enjoy ๐