Prepare Release 2.0.6 (#5104)
* add: generic cache random time offset expired. * bugfix: Csrf token should be Secure and httpOnly, but not now * fix: expose the Offset property to allow external modifications * improving the concurrency performance of random value calculation * add WithOffsetFunc to define private RandomExpireCache.offset field * fix: add seconds definition * build(deps): bump github.com/stretchr/testify from 1.7.1 to 1.8.0 Bumps [github.com/stretchr/testify](https://github.com/stretchr/testify) from 1.7.1 to 1.8.0. - [Release notes](https://github.com/stretchr/testify/releases) - [Commits](https://github.com/stretchr/testify/compare/v1.7.1...v1.8.0) --- updated-dependencies: - dependency-name: github.com/stretchr/testify dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * fix 4907: force admin service http only * Feat: add get all tasks function (#4999) * feat: add get all tasks function * Refine Comments : admin/profile.go,bean/mock.go,config/global.go... (#5009) * Refine Comments * refine comments for cache.go * refine comments for log.go * Update orm.go * refine comments for orm_log.go,types.go * Update utils.go * Update doc.go * refine comments for for four files (#5011) * refine comments for cache.go * refine comments for log.go * Update orm.go * refine comments for orm_log.go,types.go * Update utils.go * Update doc.go * Update db.go * fix pass []any as any in variadic function by asasalint (#5012) * fix pass []any as any in variadic function * add change log * build(deps): bump go.opentelemetry.io/otel/trace from 1.7.0 to 1.8.0 (#5019) Bumps [go.opentelemetry.io/otel/trace](https://github.com/open-telemetry/opentelemetry-go) from 1.7.0 to 1.8.0. - [Release notes](https://github.com/open-telemetry/opentelemetry-go/releases) - [Changelog](https://github.com/open-telemetry/opentelemetry-go/blob/main/CHANGELOG.md) - [Commits](https://github.com/open-telemetry/opentelemetry-go/compare/v1.7.0...v1.8.0) --- updated-dependencies: - dependency-name: go.opentelemetry.io/otel/trace dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * refine comments for package core (#5014) * Refine Comments * refine comments for cache.go * refine comments for log.go * Update orm.go * refine comments for orm_log.go,types.go * Update utils.go * Update doc.go * refine comments * refine comments * Update db.go * refine comments for core * build(deps): bump go.opentelemetry.io/otel/exporters/stdout/stdouttrace (#5018) Bumps [go.opentelemetry.io/otel/exporters/stdout/stdouttrace](https://github.com/open-telemetry/opentelemetry-go) from 1.7.0 to 1.8.0. - [Release notes](https://github.com/open-telemetry/opentelemetry-go/releases) - [Changelog](https://github.com/open-telemetry/opentelemetry-go/blob/main/CHANGELOG.md) - [Commits](https://github.com/open-telemetry/opentelemetry-go/compare/v1.7.0...v1.8.0) --- updated-dependencies: - dependency-name: go.opentelemetry.io/otel/exporters/stdout/stdouttrace dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> * fix 5022: Miss assiging ln to graceful Server (#5028) * prepare for releasing v2.0.5 * prepare for releasing v2.0.5 (#5032) * feat: make commands and docker compose for ORM unit tests (#5031) * feat: make commands and docker compose for ORM unit tests Signed-off-by: mango <xu.weiKyrie@foxmail.com> * add changelog Signed-off-by: mango <xu.weiKyrie@foxmail.com> Signed-off-by: mango <xu.weiKyrie@foxmail.com> * Modify comment syntax error (#5094) * fix: revise the body wrapper to handle empty body case (#5102) Fix the router.go serverHttp method, wrap the body if the request body is empty, which can avoid panic when calling the CopyBody method. Signed-off-by: chlins <chenyuzh@vmware.com> Signed-off-by: chlins <chenyuzh@vmware.com> Signed-off-by: dependabot[bot] <support@github.com> Signed-off-by: mango <xu.weiKyrie@foxmail.com> Signed-off-by: chlins <chenyuzh@vmware.com> Co-authored-by: auual <ding@ibyte.me> Co-authored-by: Leon Ding <deen.job@qq.com> Co-authored-by: dada0z <zhang.guangda@qq.com> Co-authored-by: kevinzeng <kevinzeng@zego.im> Co-authored-by: Kevin Tsang <39397413+ktalg@users.noreply.github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: 日暮颂歌1991 <448081525@qq.com> Co-authored-by: Regan Yue <1131625869@qq.com> Co-authored-by: alingse <alingse@foxmail.com> Co-authored-by: mango <35127166+mangoGoForward@users.noreply.github.com> Co-authored-by: 王哈哈 <31426858+wanghaha-dev@users.noreply.github.com> Co-authored-by: Chlins Zhang <chlins.zhang@gmail.com>
This commit is contained in:
		
							parent
							
								
									f81689dfb1
								
							
						
					
					
						commit
						76343e4422
					
				| @ -9,6 +9,7 @@ Note: now we force the web admin service serving HTTP only. | |||||||
| - [Feat 4999: add get all tasks function](https://github.com/beego/beego/pull/4999) | - [Feat 4999: add get all tasks function](https://github.com/beego/beego/pull/4999) | ||||||
| - [Fix 5012: fix some bug, pass []any as any in variadic function](https://github.com/beego/beego/pull/5012) | - [Fix 5012: fix some bug, pass []any as any in variadic function](https://github.com/beego/beego/pull/5012) | ||||||
| - [Fix 5022: Miss assigning listener to graceful Server](https://github.com/beego/beego/pull/5028) | - [Fix 5022: Miss assigning listener to graceful Server](https://github.com/beego/beego/pull/5028) | ||||||
|  | - [Fix 4955: Make commands and Docker compose for ORM unit tests](https://github.com/beego/beego/pull/5031) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| # v2.0.4 | # v2.0.4 | ||||||
|  | |||||||
							
								
								
									
										64
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,64 @@ | |||||||
|  | # Copyright 2020
 | ||||||
|  | #
 | ||||||
|  | # Licensed under the Apache License, Version 2.0 (the "License");
 | ||||||
|  | # you may not use this file except in compliance with the License.
 | ||||||
|  | # You may obtain a copy of the License at
 | ||||||
|  | #
 | ||||||
|  | # http://www.apache.org/licenses/LICENSE-2.0
 | ||||||
|  | #
 | ||||||
|  | # Unless required by applicable law or agreed to in writing, software
 | ||||||
|  | # distributed under the License is distributed on an "AS IS" BASIS,
 | ||||||
|  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | ||||||
|  | # See the License for the specific language governing permissions and
 | ||||||
|  | # limitations under the License.
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ##@ General
 | ||||||
|  | 
 | ||||||
|  | # The help target prints out all targets with their descriptions organized
 | ||||||
|  | # beneath their categories. The categories are represented by '##@' and the
 | ||||||
|  | # target descriptions by '##'. The awk commands is responsible for reading the
 | ||||||
|  | # entire set of makefiles included in this invocation, looking for lines of the
 | ||||||
|  | # file as xyz: ## something, and then pretty-format the target and help. Then,
 | ||||||
|  | # if there's a line with ##@ something, that gets pretty-printed as a category.
 | ||||||
|  | # More info on the usage of ANSI control characters for terminal formatting:
 | ||||||
|  | # https://en.wikipedia.org/wiki/ANSI_escape_code#SGR_parameters
 | ||||||
|  | # More info on the awk command:
 | ||||||
|  | # http://linuxcommand.org/lc3_adv_awk.php
 | ||||||
|  | 
 | ||||||
|  | help: ## Display this help.
 | ||||||
|  | 	@awk 'BEGIN {FS = ":.*##"; printf "\nUsage:\n  make \033[36m<target>\033[0m\n"} /^[a-zA-Z_0-9-]+:.*?##/ { printf "  \033[36m%-15s\033[0m %s\n", $$1, $$2 } /^##@/ { printf "\n\033[1m%s\033[0m\n", substr($$0, 5) } ' $(MAKEFILE_LIST) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | ##@ Test
 | ||||||
|  | 
 | ||||||
|  | test-orm-mysql5: ## Run ORM unit tests on mysql5.
 | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml up -d | ||||||
|  | 	export ORM_DRIVER=mysql | ||||||
|  | 	export ORM_SOURCE="beego:test@tcp(localhost:13306)/orm_test?charset=utf8" | ||||||
|  | 	go test -v github.com/beego/beego/v2/client/orm | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml down | ||||||
|  | 
 | ||||||
|  | test-orm-mysql8: ## Run ORM unit tests on mysql8.
 | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml up -d | ||||||
|  | 	export ORM_DRIVER=mysql | ||||||
|  | 	export ORM_SOURCE="beego:test@tcp(localhost:23306)/orm_test?charset=utf8" | ||||||
|  | 	go test -v github.com/beego/beego/v2/client/orm | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml down | ||||||
|  | 
 | ||||||
|  | test-orm-pgsql: ## Run ORM unit tests on postgresql.
 | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml up -d | ||||||
|  | 	export ORM_DRIVER=postgres | ||||||
|  | 	export ORM_SOURCE="user=postgres password=postgres dbname=orm_test sslmode=disable" | ||||||
|  | 	go test -v github.com/beego/beego/v2/client/orm | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml down | ||||||
|  | 
 | ||||||
|  | test-orm-tidb: ## Run ORM unit tests on tidb.
 | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml up -d | ||||||
|  | 	export ORM_DRIVER=tidb | ||||||
|  | 	export ORM_SOURCE="memory://test/test" | ||||||
|  | 	go test -v github.com/beego/beego/v2/client/orm | ||||||
|  | 	docker-compose -f scripts/orm_docker_compose.yaml down | ||||||
|  | 
 | ||||||
|  | .PHONY: test-orm-all | ||||||
|  | test-orm-all: test-orm-mysql5 test-orm-mysql8 test-orm-pgsql test-orm-tidb | ||||||
| @ -94,7 +94,7 @@ func (input *BeegoInput) IsHead() bool { | |||||||
| 	return (*context.BeegoInput)(input).IsHead() | 	return (*context.BeegoInput)(input).IsHead() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // IsOptions Is this a OPTIONS method request? | // IsOptions Is this an OPTIONS method request? | ||||||
| func (input *BeegoInput) IsOptions() bool { | func (input *BeegoInput) IsOptions() bool { | ||||||
| 	return (*context.BeegoInput)(input).IsOptions() | 	return (*context.BeegoInput)(input).IsOptions() | ||||||
| } | } | ||||||
|  | |||||||
| @ -52,7 +52,7 @@ func (c *Condition) AndCond(cond *Condition) *Condition { | |||||||
| 	return (*Condition)((*orm.Condition)(c).AndCond((*orm.Condition)(cond))) | 	return (*Condition)((*orm.Condition)(c).AndCond((*orm.Condition)(cond))) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // AndNotCond combine a AND NOT condition to current condition | // AndNotCond combine an AND NOT condition to current condition | ||||||
| func (c *Condition) AndNotCond(cond *Condition) *Condition { | func (c *Condition) AndNotCond(cond *Condition) *Condition { | ||||||
| 	return (*Condition)((*orm.Condition)(c).AndNotCond((*orm.Condition)(cond))) | 	return (*Condition)((*orm.Condition)(c).AndNotCond((*orm.Condition)(cond))) | ||||||
| } | } | ||||||
| @ -67,12 +67,12 @@ func (c Condition) OrNot(expr string, args ...interface{}) *Condition { | |||||||
| 	return (*Condition)((orm.Condition)(c).OrNot(expr, args...)) | 	return (*Condition)((orm.Condition)(c).OrNot(expr, args...)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // OrCond combine a OR condition to current condition | // OrCond combine an OR condition to current condition | ||||||
| func (c *Condition) OrCond(cond *Condition) *Condition { | func (c *Condition) OrCond(cond *Condition) *Condition { | ||||||
| 	return (*Condition)((*orm.Condition)(c).OrCond((*orm.Condition)(cond))) | 	return (*Condition)((*orm.Condition)(c).OrCond((*orm.Condition)(cond))) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // OrNotCond combine a OR NOT condition to current condition | // OrNotCond combine an OR NOT condition to current condition | ||||||
| func (c *Condition) OrNotCond(cond *Condition) *Condition { | func (c *Condition) OrNotCond(cond *Condition) *Condition { | ||||||
| 	return (*Condition)((*orm.Condition)(c).OrNotCond((*orm.Condition)(cond))) | 	return (*Condition)((*orm.Condition)(c).OrNotCond((*orm.Condition)(cond))) | ||||||
| } | } | ||||||
|  | |||||||
| @ -19,17 +19,17 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var InvalidUrl = berror.DefineCode(4001001, moduleName, "InvalidUrl", ` | var InvalidUrl = berror.DefineCode(4001001, moduleName, "InvalidUrl", ` | ||||||
| You pass a invalid url to httplib module. Please check your url, be careful about special character.  | You pass an invalid url to httplib module. Please check your url, be careful about special character.  | ||||||
| `) | `) | ||||||
| 
 | 
 | ||||||
| var InvalidUrlProtocolVersion = berror.DefineCode(4001002, moduleName, "InvalidUrlProtocolVersion", ` | var InvalidUrlProtocolVersion = berror.DefineCode(4001002, moduleName, "InvalidUrlProtocolVersion", ` | ||||||
| You pass a invalid protocol version. In practice, we use HTTP/1.0, HTTP/1.1, HTTP/1.2 | You pass an invalid protocol version. In practice, we use HTTP/1.0, HTTP/1.1, HTTP/1.2 | ||||||
| But something like HTTP/3.2 is valid for client, and the major version is 3, minor version is 2. | But something like HTTP/3.2 is valid for client, and the major version is 3, minor version is 2. | ||||||
| but you must confirm that server support those abnormal protocol version. | but you must confirm that server support those abnormal protocol version. | ||||||
| `) | `) | ||||||
| 
 | 
 | ||||||
| var UnsupportedBodyType = berror.DefineCode(4001003, moduleName, "UnsupportedBodyType", ` | var UnsupportedBodyType = berror.DefineCode(4001003, moduleName, "UnsupportedBodyType", ` | ||||||
| You use a invalid data as request body. | You use an invalid data as request body. | ||||||
| For now, we only support type string and byte[]. | For now, we only support type string and byte[]. | ||||||
| `) | `) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -112,7 +112,7 @@ func Head(url string) *BeegoHTTPRequest { | |||||||
| 	return NewBeegoRequest(url, "HEAD") | 	return NewBeegoRequest(url, "HEAD") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // BeegoHTTPRequest provides more useful methods than http.Request for requesting a url. | // BeegoHTTPRequest provides more useful methods than http.Request for requesting an url. | ||||||
| type BeegoHTTPRequest struct { | type BeegoHTTPRequest struct { | ||||||
| 	url     string | 	url     string | ||||||
| 	req     *http.Request | 	req     *http.Request | ||||||
|  | |||||||
| @ -1111,7 +1111,7 @@ func (d *dbBase) ReadBatch(ctx context.Context, q dbQuerier, qs *querySet, mi *m | |||||||
| 				ind.Set(mind) | 				ind.Set(mind) | ||||||
| 			} else { | 			} else { | ||||||
| 				if cnt == 0 { | 				if cnt == 0 { | ||||||
| 					// you can use a empty & caped container list | 					// you can use an empty & caped container list | ||||||
| 					// orm will not replace it | 					// orm will not replace it | ||||||
| 					if ind.Len() != 0 { | 					if ind.Len() != 0 { | ||||||
| 						// if container is not empty | 						// if container is not empty | ||||||
| @ -1135,7 +1135,7 @@ func (d *dbBase) ReadBatch(ctx context.Context, q dbQuerier, qs *querySet, mi *m | |||||||
| 			ind.Set(slice) | 			ind.Set(slice) | ||||||
| 		} else { | 		} else { | ||||||
| 			// when a result is empty and container is nil | 			// when a result is empty and container is nil | ||||||
| 			// to set a empty container | 			// to set an empty container | ||||||
| 			if ind.IsNil() { | 			if ind.IsNil() { | ||||||
| 				ind.Set(reflect.MakeSlice(ind.Type(), 0, 0)) | 				ind.Set(reflect.MakeSlice(ind.Type(), 0, 0)) | ||||||
| 			} | 			} | ||||||
|  | |||||||
| @ -155,7 +155,7 @@ func (d *dbBaseMysql) InsertOrUpdate(ctx context.Context, q dbQuerier, mi *model | |||||||
| 	if isMulti { | 	if isMulti { | ||||||
| 		qmarks = strings.Repeat(qmarks+"), (", multi-1) + qmarks | 		qmarks = strings.Repeat(qmarks+"), (", multi-1) + qmarks | ||||||
| 	} | 	} | ||||||
| 	// conflitValue maybe is a int,can`t use fmt.Sprintf | 	// conflitValue maybe is an int,can`t use fmt.Sprintf | ||||||
| 	query := fmt.Sprintf("INSERT INTO %s%s%s (%s%s%s) VALUES (%s) %s "+qupdates, Q, mi.table, Q, Q, columns, Q, qmarks, iouStr) | 	query := fmt.Sprintf("INSERT INTO %s%s%s (%s%s%s) VALUES (%s) %s "+qupdates, Q, mi.table, Q, Q, columns, Q, qmarks, iouStr) | ||||||
| 
 | 
 | ||||||
| 	d.ins.ReplaceMarks(&query) | 	d.ins.ReplaceMarks(&query) | ||||||
|  | |||||||
| @ -90,7 +90,7 @@ func (c *Condition) AndCond(cond *Condition) *Condition { | |||||||
| 	return c | 	return c | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // AndNotCond combine a AND NOT condition to current condition | // AndNotCond combine an AND NOT condition to current condition | ||||||
| func (c *Condition) AndNotCond(cond *Condition) *Condition { | func (c *Condition) AndNotCond(cond *Condition) *Condition { | ||||||
| 	c = c.clone() | 	c = c.clone() | ||||||
| 	if c == cond { | 	if c == cond { | ||||||
| @ -121,7 +121,7 @@ func (c Condition) OrNot(expr string, args ...interface{}) *Condition { | |||||||
| 	return &c | 	return &c | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // OrCond combine a OR condition to current condition | // OrCond combine an OR condition to current condition | ||||||
| func (c *Condition) OrCond(cond *Condition) *Condition { | func (c *Condition) OrCond(cond *Condition) *Condition { | ||||||
| 	c = c.clone() | 	c = c.clone() | ||||||
| 	if c == cond { | 	if c == cond { | ||||||
| @ -133,7 +133,7 @@ func (c *Condition) OrCond(cond *Condition) *Condition { | |||||||
| 	return c | 	return c | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // OrNotCond combine a OR NOT condition to current condition | // OrNotCond combine an OR NOT condition to current condition | ||||||
| func (c *Condition) OrNotCond(cond *Condition) *Condition { | func (c *Condition) OrNotCond(cond *Condition) *Condition { | ||||||
| 	c = c.clone() | 	c = c.clone() | ||||||
| 	if c == cond { | 	if c == cond { | ||||||
|  | |||||||
| @ -257,7 +257,7 @@ func (o *querySet) DeleteWithCtx(ctx context.Context) (int64, error) { | |||||||
| 	return o.orm.alias.DbBaser.DeleteBatch(ctx, o.orm.db, o, o.mi, o.cond, o.orm.alias.TZ) | 	return o.orm.alias.DbBaser.DeleteBatch(ctx, o.orm.db, o, o.mi, o.cond, o.orm.alias.TZ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // return a insert queryer. | // return an insert queryer. | ||||||
| // it can be used in times. | // it can be used in times. | ||||||
| // example: | // example: | ||||||
| // 	i,err := sq.PrepareInsert() | // 	i,err := sq.PrepareInsert() | ||||||
|  | |||||||
| @ -405,7 +405,7 @@ type QuerySeter interface { | |||||||
| 	// 	//delete two user  who's name is testing1 or testing2 | 	// 	//delete two user  who's name is testing1 or testing2 | ||||||
| 	Delete() (int64, error) | 	Delete() (int64, error) | ||||||
| 	DeleteWithCtx(context.Context) (int64, error) | 	DeleteWithCtx(context.Context) (int64, error) | ||||||
| 	// return a insert queryer. | 	// return an insert queryer. | ||||||
| 	// it can be used in times. | 	// it can be used in times. | ||||||
| 	// example: | 	// example: | ||||||
| 	// 	i,err := sq.PrepareInsert() | 	// 	i,err := sq.PrepareInsert() | ||||||
|  | |||||||
| @ -98,7 +98,7 @@ func (ini *IniConfig) parseData(dir string, data []byte) (*IniConfigContainer, e | |||||||
| 				break | 				break | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			// It might be a good idea to throw a error on all unknonw errors? | 			// It might be a good idea to throw an error on all unknonw errors? | ||||||
| 			if _, ok := err.(*os.PathError); ok { | 			if _, ok := err.(*os.PathError); ok { | ||||||
| 				return nil, err | 				return nil, err | ||||||
| 			} | 			} | ||||||
|  | |||||||
| @ -105,11 +105,11 @@ func (el *esLogger) WriteMsg(lm *logs.LogMsg) error { | |||||||
| 	return err | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Destroy is a empty method | // Destroy is an empty method | ||||||
| func (el *esLogger) Destroy() { | func (el *esLogger) Destroy() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Flush is a empty method | // Flush is an empty method | ||||||
| func (el *esLogger) Flush() { | func (el *esLogger) Flush() { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										124
									
								
								scripts/orm_docker_compose.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								scripts/orm_docker_compose.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,124 @@ | |||||||
|  | # Copyright 2022 | ||||||
|  | # | ||||||
|  | # Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | # you may not use this file except in compliance with the License. | ||||||
|  | # You may obtain a copy of the License at | ||||||
|  | # | ||||||
|  | # http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  | # | ||||||
|  | # Unless required by applicable law or agreed to in writing, software | ||||||
|  | # distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | # See the License for the specific language governing permissions and | ||||||
|  | # limitations under the License. | ||||||
|  | 
 | ||||||
|  | version: "3.8" | ||||||
|  | services: | ||||||
|  |   # mysql5.7 | ||||||
|  |   mysql5: | ||||||
|  |     container_name: "beego-mysql5" | ||||||
|  |     image: mysql:5.7.30 | ||||||
|  |     ports: | ||||||
|  |       - "13306:3306" | ||||||
|  |     environment: | ||||||
|  |       - MYSQL_ROOT_PASSWORD=1q2w3e | ||||||
|  |       - MYSQL_DATABASE=orm_test | ||||||
|  |       - MYSQL_USER=beego | ||||||
|  |       - MYSQL_PASSWORD=test | ||||||
|  | 
 | ||||||
|  |   # mysql8.0 | ||||||
|  |   mysql8: | ||||||
|  |     container_name: "beego-mysql8" | ||||||
|  |     image: mysql:8.0 | ||||||
|  |     ports: | ||||||
|  |       - "23306:3306" | ||||||
|  |     environment: | ||||||
|  |       - MYSQL_ROOT_PASSWORD=1q2w3e | ||||||
|  |       - MYSQL_DATABASE=orm_test | ||||||
|  |       - MYSQL_USER=beego | ||||||
|  |       - MYSQL_PASSWORD=test | ||||||
|  | 
 | ||||||
|  |   # postgresql | ||||||
|  |   postgresql: | ||||||
|  |     container_name: "beego-postgresql" | ||||||
|  |     image: bitnami/postgresql:latest | ||||||
|  |     ports: | ||||||
|  |     - "5432:5432" | ||||||
|  |     environment: | ||||||
|  |       - POSTGRES_USER=postgres | ||||||
|  |       - POSTGRES_PASSWORD=postgres | ||||||
|  |       - POSTGRES_DB=orm_test | ||||||
|  | 
 | ||||||
|  |   # tidb | ||||||
|  |   pd0: | ||||||
|  |     image: pingcap/pd:latest | ||||||
|  |     ports: | ||||||
|  |       - "2379:2379" | ||||||
|  |     volumes: | ||||||
|  |       - ./config/pd.toml:/pd.toml:ro | ||||||
|  |       - ./data:/data | ||||||
|  |       - ./logs:/logs | ||||||
|  |     command: | ||||||
|  |       - --name=pd0 | ||||||
|  |       - --client-urls=http://0.0.0.0:2379 | ||||||
|  |       - --peer-urls=http://0.0.0.0:2380 | ||||||
|  |       - --advertise-client-urls=http://pd0:2379 | ||||||
|  |       - --advertise-peer-urls=http://pd0:2380 | ||||||
|  |       - --initial-cluster=pd0=http://pd0:2380 | ||||||
|  |       - --data-dir=/data/pd0 | ||||||
|  |       - --config=/pd.toml | ||||||
|  |       - --log-file=/logs/pd0.log | ||||||
|  |     restart: on-failure | ||||||
|  | 
 | ||||||
|  |   tikv0: | ||||||
|  |     image: pingcap/tikv:latest | ||||||
|  |     volumes: | ||||||
|  |       - ./config/tikv.toml:/tikv.toml:ro | ||||||
|  |       - ./data:/data | ||||||
|  |       - ./logs:/logs | ||||||
|  |     command: | ||||||
|  |       - --addr=0.0.0.0:20160 | ||||||
|  |       - --advertise-addr=tikv0:20160 | ||||||
|  |       - --data-dir=/data/tikv0 | ||||||
|  |       - --pd=pd0:2379 | ||||||
|  |       - --config=/tikv.toml | ||||||
|  |       - --log-file=/logs/tikv0.log | ||||||
|  |     depends_on: | ||||||
|  |       - "pd0" | ||||||
|  |     restart: on-failure | ||||||
|  | 
 | ||||||
|  |   tikv1: | ||||||
|  |     image: pingcap/tikv:latest | ||||||
|  |     volumes: | ||||||
|  |       - ./config/tikv.toml:/tikv.toml:ro | ||||||
|  |       - ./data:/data | ||||||
|  |       - ./logs:/logs | ||||||
|  |     command: | ||||||
|  |       - --addr=0.0.0.0:20160 | ||||||
|  |       - --advertise-addr=tikv1:20160 | ||||||
|  |       - --data-dir=/data/tikv1 | ||||||
|  |       - --pd=pd0:2379 | ||||||
|  |       - --config=/tikv.toml | ||||||
|  |       - --log-file=/logs/tikv1.log | ||||||
|  |     depends_on: | ||||||
|  |       - "pd0" | ||||||
|  |     restart: on-failure | ||||||
|  | 
 | ||||||
|  |   tidb: | ||||||
|  |     image: pingcap/tidb:latest | ||||||
|  |     ports: | ||||||
|  |       - "4000:4000" | ||||||
|  |       - "10080:10080" | ||||||
|  |     volumes: | ||||||
|  |       - ./config/tidb.toml:/tidb.toml:ro | ||||||
|  |       - ./logs:/logs | ||||||
|  |     command: | ||||||
|  |       - --store=tikv | ||||||
|  |       - --path=pd0:2379 | ||||||
|  |       - --config=/tidb.toml | ||||||
|  |       - --log-file=/logs/tidb.log | ||||||
|  |       - --advertise-address=tidb | ||||||
|  |     depends_on: | ||||||
|  |       - "tikv0" | ||||||
|  |       - "tikv1" | ||||||
|  |     restart: on-failure | ||||||
| @ -27,7 +27,7 @@ type siprng struct { | |||||||
| 	k0, k1, ctr uint64 | 	k0, k1, ctr uint64 | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // siphash implements SipHash-2-4, accepting a uint64 as a message. | // siphash implements SipHash-2-4, accepting an uint64 as a message. | ||||||
| func siphash(k0, k1, m uint64) uint64 { | func siphash(k0, k1, m uint64) uint64 { | ||||||
| 	// Initialization. | 	// Initialization. | ||||||
| 	v0 := k0 ^ 0x736f6d6570736575 | 	v0 := k0 ^ 0x736f6d6570736575 | ||||||
|  | |||||||
| @ -153,7 +153,7 @@ func (input *BeegoInput) IsHead() bool { | |||||||
| 	return input.Is("HEAD") | 	return input.Is("HEAD") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // IsOptions Is this a OPTIONS method request? | // IsOptions Is this an OPTIONS method request? | ||||||
| func (input *BeegoInput) IsOptions() bool { | func (input *BeegoInput) IsOptions() bool { | ||||||
| 	return input.Is("OPTIONS") | 	return input.Is("OPTIONS") | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,8 +15,10 @@ | |||||||
| package web | package web | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"bytes" | ||||||
| 	"errors" | 	"errors" | ||||||
| 	"fmt" | 	"fmt" | ||||||
|  | 	"io" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"path" | 	"path" | ||||||
| 	"reflect" | 	"reflect" | ||||||
| @ -211,6 +213,7 @@ func (p *ControllerRegister) Init() { | |||||||
| 
 | 
 | ||||||
| // Add controller handler and pattern rules to ControllerRegister. | // Add controller handler and pattern rules to ControllerRegister. | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	default methods is the same name as method | //	default methods is the same name as method | ||||||
| //	Add("/user",&UserController{}) | //	Add("/user",&UserController{}) | ||||||
| //	Add("/api/list",&RestController{},"*:ListFood") | //	Add("/api/list",&RestController{},"*:ListFood") | ||||||
| @ -348,6 +351,7 @@ func (p *ControllerRegister) Include(cList ...ControllerInterface) { | |||||||
| // GetContext returns a context from pool, so usually you should remember to call Reset function to clean the context | // GetContext returns a context from pool, so usually you should remember to call Reset function to clean the context | ||||||
| // And don't forget to give back context to pool | // And don't forget to give back context to pool | ||||||
| // example: | // example: | ||||||
|  | // | ||||||
| //	ctx := p.GetContext() | //	ctx := p.GetContext() | ||||||
| //	ctx.Reset(w, q) | //	ctx.Reset(w, q) | ||||||
| //	defer p.GiveBackContext(ctx) | //	defer p.GiveBackContext(ctx) | ||||||
| @ -362,6 +366,7 @@ func (p *ControllerRegister) GiveBackContext(ctx *beecontext.Context) { | |||||||
| 
 | 
 | ||||||
| // CtrlGet add get method | // CtrlGet add get method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -370,6 +375,7 @@ func (p *ControllerRegister) GiveBackContext(ctx *beecontext.Context) { | |||||||
| //	   } | //	   } | ||||||
| // | // | ||||||
| //	   CtrlGet("/api/:id", MyController.Ping) | //	   CtrlGet("/api/:id", MyController.Ping) | ||||||
|  | // | ||||||
| // If the receiver of function Ping is pointer, you should use CtrlGet("/api/:id", (*MyController).Ping) | // If the receiver of function Ping is pointer, you should use CtrlGet("/api/:id", (*MyController).Ping) | ||||||
| func (p *ControllerRegister) CtrlGet(pattern string, f interface{}) { | func (p *ControllerRegister) CtrlGet(pattern string, f interface{}) { | ||||||
| 	p.AddRouterMethod(http.MethodGet, pattern, f) | 	p.AddRouterMethod(http.MethodGet, pattern, f) | ||||||
| @ -377,6 +383,7 @@ func (p *ControllerRegister) CtrlGet(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // CtrlPost add post method | // CtrlPost add post method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -385,6 +392,7 @@ func (p *ControllerRegister) CtrlGet(pattern string, f interface{}) { | |||||||
| //	   } | //	   } | ||||||
| // | // | ||||||
| //	   CtrlPost("/api/:id", MyController.Ping) | //	   CtrlPost("/api/:id", MyController.Ping) | ||||||
|  | // | ||||||
| // If the receiver of function Ping is pointer, you should use CtrlPost("/api/:id", (*MyController).Ping) | // If the receiver of function Ping is pointer, you should use CtrlPost("/api/:id", (*MyController).Ping) | ||||||
| func (p *ControllerRegister) CtrlPost(pattern string, f interface{}) { | func (p *ControllerRegister) CtrlPost(pattern string, f interface{}) { | ||||||
| 	p.AddRouterMethod(http.MethodPost, pattern, f) | 	p.AddRouterMethod(http.MethodPost, pattern, f) | ||||||
| @ -392,6 +400,7 @@ func (p *ControllerRegister) CtrlPost(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // CtrlHead add head method | // CtrlHead add head method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -400,6 +409,7 @@ func (p *ControllerRegister) CtrlPost(pattern string, f interface{}) { | |||||||
| //	   } | //	   } | ||||||
| // | // | ||||||
| //	   CtrlHead("/api/:id", MyController.Ping) | //	   CtrlHead("/api/:id", MyController.Ping) | ||||||
|  | // | ||||||
| // If the receiver of function Ping is pointer, you should use CtrlHead("/api/:id", (*MyController).Ping) | // If the receiver of function Ping is pointer, you should use CtrlHead("/api/:id", (*MyController).Ping) | ||||||
| func (p *ControllerRegister) CtrlHead(pattern string, f interface{}) { | func (p *ControllerRegister) CtrlHead(pattern string, f interface{}) { | ||||||
| 	p.AddRouterMethod(http.MethodHead, pattern, f) | 	p.AddRouterMethod(http.MethodHead, pattern, f) | ||||||
| @ -422,6 +432,7 @@ func (p *ControllerRegister) CtrlPut(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // CtrlPatch add patch method | // CtrlPatch add patch method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -436,6 +447,7 @@ func (p *ControllerRegister) CtrlPatch(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // CtrlDelete add delete method | // CtrlDelete add delete method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -450,6 +462,7 @@ func (p *ControllerRegister) CtrlDelete(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // CtrlOptions add options method | // CtrlOptions add options method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -464,6 +477,7 @@ func (p *ControllerRegister) CtrlOptions(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // CtrlAny add all method | // CtrlAny add all method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -478,6 +492,7 @@ func (p *ControllerRegister) CtrlAny(pattern string, f interface{}) { | |||||||
| 
 | 
 | ||||||
| // AddRouterMethod add http method router | // AddRouterMethod add http method router | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	   type MyController struct { | //	   type MyController struct { | ||||||
| //		     web.Controller | //		     web.Controller | ||||||
| //	   } | //	   } | ||||||
| @ -624,6 +639,7 @@ type HandleFunc func(ctx *beecontext.Context) | |||||||
| 
 | 
 | ||||||
| // Get add get method | // Get add get method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Get("/", func(ctx *context.Context){ | //	Get("/", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -633,6 +649,7 @@ func (p *ControllerRegister) Get(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Post add post method | // Post add post method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Post("/api", func(ctx *context.Context){ | //	Post("/api", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -642,6 +659,7 @@ func (p *ControllerRegister) Post(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Put add put method | // Put add put method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Put("/api/:id", func(ctx *context.Context){ | //	Put("/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -651,6 +669,7 @@ func (p *ControllerRegister) Put(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Delete add delete method | // Delete add delete method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Delete("/api/:id", func(ctx *context.Context){ | //	Delete("/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -660,6 +679,7 @@ func (p *ControllerRegister) Delete(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Head add head method | // Head add head method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Head("/api/:id", func(ctx *context.Context){ | //	Head("/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -669,6 +689,7 @@ func (p *ControllerRegister) Head(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Patch add patch method | // Patch add patch method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Patch("/api/:id", func(ctx *context.Context){ | //	Patch("/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -678,6 +699,7 @@ func (p *ControllerRegister) Patch(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Options add options method | // Options add options method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Options("/api/:id", func(ctx *context.Context){ | //	Options("/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -687,6 +709,7 @@ func (p *ControllerRegister) Options(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // Any add all method | // Any add all method | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	Any("/api/:id", func(ctx *context.Context){ | //	Any("/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -696,6 +719,7 @@ func (p *ControllerRegister) Any(pattern string, f HandleFunc) { | |||||||
| 
 | 
 | ||||||
| // AddMethod add http method router | // AddMethod add http method router | ||||||
| // usage: | // usage: | ||||||
|  | // | ||||||
| //	AddMethod("get","/api/:id", func(ctx *context.Context){ | //	AddMethod("get","/api/:id", func(ctx *context.Context){ | ||||||
| //	      ctx.Output.Body("hello world") | //	      ctx.Output.Body("hello world") | ||||||
| //	}) | //	}) | ||||||
| @ -784,6 +808,7 @@ func (p *ControllerRegister) InsertFilter(pattern string, pos int, filter Filter | |||||||
| // but it will using chainRoot.filterFunc as input to build a new filterFunc | // but it will using chainRoot.filterFunc as input to build a new filterFunc | ||||||
| // for example, assume that chainRoot is funcA | // for example, assume that chainRoot is funcA | ||||||
| // and we add new FilterChain | // and we add new FilterChain | ||||||
|  | // | ||||||
| //	fc := func(next) { | //	fc := func(next) { | ||||||
| //	    return func(ctx) { | //	    return func(ctx) { | ||||||
| //	          // do something | //	          // do something | ||||||
| @ -1025,10 +1050,14 @@ func (p *ControllerRegister) serveHttp(ctx *beecontext.Context) { | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if r.Method != http.MethodGet && r.Method != http.MethodHead { | 	if r.Method != http.MethodGet && r.Method != http.MethodHead { | ||||||
|  | 		body := ctx.Input.Context.Request.Body | ||||||
|  | 		if body == nil { | ||||||
|  | 			body = io.NopCloser(bytes.NewReader([]byte{})) | ||||||
|  | 		} | ||||||
| 
 | 
 | ||||||
| 		if ctx.Input.IsUpload() { | 		if ctx.Input.IsUpload() { | ||||||
| 			ctx.Input.Context.Request.Body = http.MaxBytesReader(ctx.Input.Context.ResponseWriter, | 			ctx.Input.Context.Request.Body = http.MaxBytesReader(ctx.Input.Context.ResponseWriter, | ||||||
| 				ctx.Input.Context.Request.Body, | 				body, | ||||||
| 				p.cfg.MaxUploadSize) | 				p.cfg.MaxUploadSize) | ||||||
| 		} else if p.cfg.CopyRequestBody { | 		} else if p.cfg.CopyRequestBody { | ||||||
| 			// connection will close if the incoming data are larger (RFC 7231, 6.5.11) | 			// connection will close if the incoming data are larger (RFC 7231, 6.5.11) | ||||||
| @ -1040,7 +1069,7 @@ func (p *ControllerRegister) serveHttp(ctx *beecontext.Context) { | |||||||
| 			ctx.Input.CopyBody(p.cfg.MaxMemory) | 			ctx.Input.CopyBody(p.cfg.MaxMemory) | ||||||
| 		} else { | 		} else { | ||||||
| 			ctx.Input.Context.Request.Body = http.MaxBytesReader(ctx.Input.Context.ResponseWriter, | 			ctx.Input.Context.Request.Body = http.MaxBytesReader(ctx.Input.Context.ResponseWriter, | ||||||
| 				ctx.Input.Context.Request.Body, | 				body, | ||||||
| 				p.cfg.MaxMemory) | 				p.cfg.MaxMemory) | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -98,7 +98,7 @@ func encrypt(block cipher.Block, value []byte) ([]byte, error) { | |||||||
| 
 | 
 | ||||||
| // decrypt decrypts a value using the given block in counter mode. | // decrypt decrypts a value using the given block in counter mode. | ||||||
| // | // | ||||||
| // The value to be decrypted must be prepended by a initialization vector | // The value to be decrypted must be prepended by an initialization vector | ||||||
| // (http://goo.gl/zF67k) with the length of the block size. | // (http://goo.gl/zF67k) with the length of the block size. | ||||||
| func decrypt(block cipher.Block, value []byte) ([]byte, error) { | func decrypt(block cipher.Block, value []byte) ([]byte, error) { | ||||||
| 	size := block.BlockSize() | 	size := block.BlockSize() | ||||||
|  | |||||||
| @ -103,7 +103,7 @@ func init() { | |||||||
| 	beegoTplFuncMap["lt"] = lt // < | 	beegoTplFuncMap["lt"] = lt // < | ||||||
| 	beegoTplFuncMap["ne"] = ne // != | 	beegoTplFuncMap["ne"] = ne // != | ||||||
| 
 | 
 | ||||||
| 	beegoTplFuncMap["urlfor"] = URLFor // build a URL to match a Controller and it's method | 	beegoTplFuncMap["urlfor"] = URLFor // build an URL to match a Controller and it's method | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // AddFuncMap let user to register a func in the template. | // AddFuncMap let user to register a func in the template. | ||||||
|  | |||||||
| @ -221,7 +221,7 @@ func (f optionFunc) apply(t *Task) { | |||||||
| 	f(t) | 	f(t) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // TimeoutOption return a option to set timeout duration for task | // TimeoutOption return an option to set timeout duration for task | ||||||
| func TimeoutOption(timeout time.Duration) Option { | func TimeoutOption(timeout time.Duration) Option { | ||||||
| 	return optionFunc(func(t *Task) { | 	return optionFunc(func(t *Task) { | ||||||
| 		t.Timeout = timeout | 		t.Timeout = timeout | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user