updates #207
| @@ -1,24 +0,0 @@ | ||||
| name: lint | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|     - master | ||||
|     - v3 | ||||
| jobs: | ||||
|   lint: | ||||
|     name: lint | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: setup-go | ||||
|       uses: actions/setup-go@v3 | ||||
|       with: | ||||
|         go-version: 1.21 | ||||
|     - name: checkout | ||||
|       uses: actions/checkout@v3 | ||||
|     - name: deps | ||||
|       run: go get -v -d ./... | ||||
|     - name: lint | ||||
|       uses: https://github.com/golangci/golangci-lint-action@v3.4.0 | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         version: v1.52 | ||||
							
								
								
									
										40
									
								
								.gitea/workflows/pipeline.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								.gitea/workflows/pipeline.yml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | ||||
| name: pipeline | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|     - master | ||||
|     - v3 | ||||
|     - v4 | ||||
| jobs: | ||||
|   lint: | ||||
|     name: lint | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: setup-go | ||||
|       uses: actions/setup-go@v5 | ||||
|       with: | ||||
|         go-version: 'stable' | ||||
|     - name: checkout | ||||
|       uses: actions/checkout@v3 | ||||
|     - name: deps | ||||
|       run: go get -v -d ./... | ||||
|     - name: lint | ||||
|       uses: https://github.com/golangci/golangci-lint-action@v6 | ||||
|       with: | ||||
|         version: v1.62.2 | ||||
|   test: | ||||
|     name: test | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: setup-go | ||||
|       uses: actions/setup-go@v5 | ||||
|       with: | ||||
|         go-version: 'stable' | ||||
|     - name: checkout | ||||
|       uses: actions/checkout@v3 | ||||
|     - name: deps | ||||
|       run: go get -v -d ./... | ||||
|     - name: test | ||||
|       env: | ||||
|         INTEGRATION_TESTS: yes | ||||
|       run: go test -mod readonly -v ./... | ||||
| @@ -1,23 +0,0 @@ | ||||
| name: pr | ||||
| on: | ||||
|   pull_request: | ||||
|     branches: | ||||
|     - master | ||||
|     - v3 | ||||
| jobs: | ||||
|   test: | ||||
|     name: test | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - name: checkout | ||||
|       uses: actions/checkout@v3 | ||||
|     - name: setup-go | ||||
|       uses: actions/setup-go@v3 | ||||
|       with: | ||||
|         go-version: 1.21 | ||||
|     - name: deps | ||||
|       run: go get -v -t -d ./... | ||||
|     - name: test | ||||
|       env: | ||||
|         INTEGRATION_TESTS: yes | ||||
|       run: go test -mod readonly -v ./... | ||||
| @@ -1,44 +1,5 @@ | ||||
| run: | ||||
|   concurrency: 4 | ||||
|   concurrency: 8 | ||||
|   deadline: 5m | ||||
|   issues-exit-code: 1 | ||||
|   tests: true | ||||
|  | ||||
| linters-settings: | ||||
|   govet: | ||||
|     check-shadowing: true | ||||
|     enable: | ||||
|       - fieldalignment | ||||
|  | ||||
| linters: | ||||
|   enable: | ||||
|     - govet | ||||
|     - deadcode | ||||
|     - errcheck | ||||
|     - govet | ||||
|     - ineffassign | ||||
|     - staticcheck | ||||
|     - structcheck | ||||
|     - typecheck | ||||
|     - unused | ||||
|     - varcheck | ||||
|     - bodyclose | ||||
|     - gci | ||||
|     - goconst | ||||
|     - gocritic | ||||
|     - gosimple | ||||
|     - gofmt | ||||
|     - gofumpt | ||||
|     - goimports | ||||
|     - revive | ||||
|     - gosec | ||||
|     - makezero | ||||
|     - misspell | ||||
|     - nakedret | ||||
|     - nestif | ||||
|     - nilerr | ||||
|     - noctx | ||||
|     - prealloc | ||||
|     - unconvert | ||||
|     - unparam | ||||
|   disable-all: false | ||||
|   | ||||
| @@ -298,7 +298,7 @@ func (n *noopClient) fnCall(ctx context.Context, req Request, rsp interface{}, o | ||||
| 		// call backoff first. Someone may want an initial start delay | ||||
| 		t, err := callOpts.Backoff(ctx, req, i) | ||||
| 		if err != nil { | ||||
| 			return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 			return errors.InternalServerError("go.micro.client", "%s", err.Error()) | ||||
| 		} | ||||
|  | ||||
| 		// only sleep if greater than 0 | ||||
| @@ -312,7 +312,7 @@ func (n *noopClient) fnCall(ctx context.Context, req Request, rsp interface{}, o | ||||
| 			// TODO apply any filtering here | ||||
| 			routes, err = n.opts.Lookup(ctx, req, callOpts) | ||||
| 			if err != nil { | ||||
| 				return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 				return errors.InternalServerError("go.micro.client", "%s", err.Error()) | ||||
| 			} | ||||
|  | ||||
| 			// balance the list of nodes | ||||
| @@ -466,7 +466,7 @@ func (n *noopClient) fnStream(ctx context.Context, req Request, opts ...CallOpti | ||||
| 		// call backoff first. Someone may want an initial start delay | ||||
| 		t, cerr := callOpts.Backoff(ctx, req, i) | ||||
| 		if cerr != nil { | ||||
| 			return nil, errors.InternalServerError("go.micro.client", cerr.Error()) | ||||
| 			return nil, errors.InternalServerError("go.micro.client", "%s", cerr.Error()) | ||||
| 		} | ||||
|  | ||||
| 		// only sleep if greater than 0 | ||||
| @@ -480,7 +480,7 @@ func (n *noopClient) fnStream(ctx context.Context, req Request, opts ...CallOpti | ||||
| 			// TODO apply any filtering here | ||||
| 			routes, err = n.opts.Lookup(ctx, req, callOpts) | ||||
| 			if err != nil { | ||||
| 				return nil, errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 				return nil, errors.InternalServerError("go.micro.client", "%s", err.Error()) | ||||
| 			} | ||||
|  | ||||
| 			// balance the list of nodes | ||||
| @@ -609,13 +609,13 @@ func (n *noopClient) publish(ctx context.Context, ps []Message, opts ...PublishO | ||||
| 			// use codec for payload | ||||
| 			cf, err := n.newCodec(p.ContentType()) | ||||
| 			if err != nil { | ||||
| 				return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 				return errors.InternalServerError("go.micro.client", "%s", err.Error()) | ||||
| 			} | ||||
|  | ||||
| 			// set the body | ||||
| 			b, err := cf.Marshal(p.Payload()) | ||||
| 			if err != nil { | ||||
| 				return errors.InternalServerError("go.micro.client", err.Error()) | ||||
| 				return errors.InternalServerError("go.micro.client", "%s", err.Error()) | ||||
| 			} | ||||
| 			body = b | ||||
| 		} | ||||
|   | ||||
| @@ -2,6 +2,7 @@ package errors | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	er "errors" | ||||
| 	"fmt" | ||||
| 	"net/http" | ||||
| @@ -26,7 +27,7 @@ func TestMarshalJSON(t *testing.T) { | ||||
| func TestEmpty(t *testing.T) { | ||||
| 	msg := "test" | ||||
| 	var err *Error | ||||
| 	err = FromError(fmt.Errorf(msg)) | ||||
| 	err = FromError(errors.New(msg)) | ||||
| 	if err.Detail != msg { | ||||
| 		t.Fatalf("invalid error %v", err) | ||||
| 	} | ||||
|   | ||||
| @@ -46,11 +46,11 @@ func (h *wrapper) Handle(ctx context.Context, rec slog.Record) error { | ||||
| } | ||||
|  | ||||
| func (h *wrapper) WithAttrs(attrs []slog.Attr) slog.Handler { | ||||
| 	return h.WithAttrs(attrs) | ||||
| 	return h.h.WithAttrs(attrs) | ||||
| } | ||||
|  | ||||
| func (h *wrapper) WithGroup(name string) slog.Handler { | ||||
| 	return h.WithGroup(name) | ||||
| 	return h.h.WithGroup(name) | ||||
| } | ||||
|  | ||||
| func (s *slogLogger) renameAttr(_ []string, a slog.Attr) slog.Attr { | ||||
| @@ -89,7 +89,6 @@ func (s *slogLogger) renameAttr(_ []string, a slog.Attr) slog.Attr { | ||||
| } | ||||
|  | ||||
| type slogLogger struct { | ||||
| 	leveler *slog.LevelVar | ||||
| 	handler *wrapper | ||||
| 	opts    logger.Options | ||||
| 	mu      sync.RWMutex | ||||
|   | ||||
| @@ -36,8 +36,8 @@ var ( | ||||
| 	circularShortBytes = []byte("<shown>") | ||||
| 	invalidAngleBytes  = []byte("<invalid>") | ||||
| 	filteredBytes      = []byte("<filtered>") | ||||
| 	openBracketBytes   = []byte("[") | ||||
| 	closeBracketBytes  = []byte("]") | ||||
| 	// openBracketBytes   = []byte("[") | ||||
| 	// closeBracketBytes  = []byte("]") | ||||
| 	percentBytes    = []byte("%") | ||||
| 	precisionBytes  = []byte(".") | ||||
| 	openAngleBytes  = []byte("<") | ||||
|   | ||||
| @@ -82,11 +82,11 @@ func TestTagged(t *testing.T) { | ||||
| func TestTaggedNested(t *testing.T) { | ||||
| 	type val struct { | ||||
| 		key string `logger:"take"` | ||||
| 		val string `logger:"omit"` | ||||
| 		// val string `logger:"omit"` | ||||
| 		unk string | ||||
| 	} | ||||
| 	type str struct { | ||||
| 		key string `logger:"omit"` | ||||
| 		// key string `logger:"omit"` | ||||
| 		val *val `logger:"take"` | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -83,6 +83,7 @@ func TestPassing(t *testing.T) { | ||||
| 	if ok { | ||||
| 		t.Fatalf("create outgoing context") | ||||
| 	} | ||||
| 	_ = md | ||||
|  | ||||
| 	ctx = NewOutgoingContext(ctx, New(1)) | ||||
| 	testCtx(ctx) | ||||
|   | ||||
							
								
								
									
										36
									
								
								micro.go
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								micro.go
									
									
									
									
									
								
							| @@ -65,6 +65,8 @@ func As(b any, target any) bool { | ||||
| 			break | ||||
| 		case targetType.Implements(routerType): | ||||
| 			break | ||||
| 		case targetType.Implements(tracerType): | ||||
| 			break | ||||
| 		default: | ||||
| 			return false | ||||
| 		} | ||||
| @@ -76,19 +78,21 @@ func As(b any, target any) bool { | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| var brokerType = reflect.TypeOf((*broker.Broker)(nil)).Elem() | ||||
| var loggerType = reflect.TypeOf((*logger.Logger)(nil)).Elem() | ||||
| var clientType = reflect.TypeOf((*client.Client)(nil)).Elem() | ||||
| var serverType = reflect.TypeOf((*server.Server)(nil)).Elem() | ||||
| var codecType = reflect.TypeOf((*codec.Codec)(nil)).Elem() | ||||
| var flowType = reflect.TypeOf((*flow.Flow)(nil)).Elem() | ||||
| var fsmType = reflect.TypeOf((*fsm.FSM)(nil)).Elem() | ||||
| var meterType = reflect.TypeOf((*meter.Meter)(nil)).Elem() | ||||
| var registerType = reflect.TypeOf((*register.Register)(nil)).Elem() | ||||
| var resolverType = reflect.TypeOf((*resolver.Resolver)(nil)).Elem() | ||||
| var routerType = reflect.TypeOf((*router.Router)(nil)).Elem() | ||||
| var selectorType = reflect.TypeOf((*selector.Selector)(nil)).Elem() | ||||
| var storeType = reflect.TypeOf((*store.Store)(nil)).Elem() | ||||
| var syncType = reflect.TypeOf((*sync.Sync)(nil)).Elem() | ||||
| var tracerType = reflect.TypeOf((*tracer.Tracer)(nil)).Elem() | ||||
| var serviceType = reflect.TypeOf((*Service)(nil)).Elem() | ||||
| var ( | ||||
| 	brokerType   = reflect.TypeOf((*broker.Broker)(nil)).Elem() | ||||
| 	loggerType   = reflect.TypeOf((*logger.Logger)(nil)).Elem() | ||||
| 	clientType   = reflect.TypeOf((*client.Client)(nil)).Elem() | ||||
| 	serverType   = reflect.TypeOf((*server.Server)(nil)).Elem() | ||||
| 	codecType    = reflect.TypeOf((*codec.Codec)(nil)).Elem() | ||||
| 	flowType     = reflect.TypeOf((*flow.Flow)(nil)).Elem() | ||||
| 	fsmType      = reflect.TypeOf((*fsm.FSM)(nil)).Elem() | ||||
| 	meterType    = reflect.TypeOf((*meter.Meter)(nil)).Elem() | ||||
| 	registerType = reflect.TypeOf((*register.Register)(nil)).Elem() | ||||
| 	resolverType = reflect.TypeOf((*resolver.Resolver)(nil)).Elem() | ||||
| 	routerType   = reflect.TypeOf((*router.Router)(nil)).Elem() | ||||
| 	selectorType = reflect.TypeOf((*selector.Selector)(nil)).Elem() | ||||
| 	storeType    = reflect.TypeOf((*store.Store)(nil)).Elem() | ||||
| 	syncType     = reflect.TypeOf((*sync.Sync)(nil)).Elem() | ||||
| 	tracerType   = reflect.TypeOf((*tracer.Tracer)(nil)).Elem() | ||||
| 	serviceType  = reflect.TypeOf((*Service)(nil)).Elem() | ||||
| ) | ||||
|   | ||||
| @@ -294,7 +294,7 @@ type loggerOptions struct { | ||||
| 	brokers   []string | ||||
| 	registers []string | ||||
| 	stores    []string | ||||
| 	meters    []string | ||||
| 	// meters    []string | ||||
| 	tracers []string | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -469,9 +469,7 @@ func serviceToRecord(s *register.Service, ttl time.Duration) *record { | ||||
| 	} | ||||
|  | ||||
| 	endpoints := make([]*register.Endpoint, len(s.Endpoints)) | ||||
| 	for i, e := range s.Endpoints { | ||||
| 		endpoints[i] = e | ||||
| 	} | ||||
| 	copy(endpoints, s.Endpoints) | ||||
|  | ||||
| 	return &record{ | ||||
| 		Name:      s.Name, | ||||
|   | ||||
| @@ -290,27 +290,25 @@ func TestWatcher(t *testing.T) { | ||||
|  | ||||
| 	ctx := context.TODO() | ||||
| 	m := NewRegister() | ||||
| 	m.Init() | ||||
| 	m.Connect(ctx) | ||||
| 	_ = m.Init() | ||||
| 	_ = m.Connect(ctx) | ||||
| 	wc, err := m.Watch(ctx) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("cant watch: %v", err) | ||||
| 	} | ||||
| 	defer wc.Stop() | ||||
|  | ||||
| 	cherr := make(chan error, 10) | ||||
| 	var wg sync.WaitGroup | ||||
| 	wg.Add(1) | ||||
| 	go func() { | ||||
| 		for { | ||||
| 		_, err := wc.Next() | ||||
| 		if err != nil { | ||||
| 				t.Fatal("unexpected err", err) | ||||
| 			cherr <- fmt.Errorf("unexpected err %v", err) | ||||
| 		} | ||||
| 		// t.Logf("changes %#+v", ch.Service) | ||||
| 		wc.Stop() | ||||
| 		wg.Done() | ||||
| 			return | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	if err := m.Register(ctx, testSrv); err != nil { | ||||
|   | ||||
| @@ -171,7 +171,6 @@ type rpcMessage struct { | ||||
| 	header      metadata.Metadata | ||||
| 	topic       string | ||||
| 	contentType string | ||||
| 	body        []byte | ||||
| } | ||||
|  | ||||
| func (r *rpcMessage) ContentType() string { | ||||
|   | ||||
| @@ -38,7 +38,7 @@ func TestNoopSub(t *testing.T) { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	logger.DefaultLogger.Init(logger.WithLevel(logger.ErrorLevel)) | ||||
| 	_ = logger.DefaultLogger.Init(logger.WithLevel(logger.ErrorLevel)) | ||||
| 	s := server.NewServer( | ||||
| 		server.Broker(b), | ||||
| 		server.Codec("application/octet-stream", codec.NewCodec()), | ||||
|   | ||||
| @@ -23,8 +23,8 @@ import ( | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
| 	maxprocs.Set() | ||||
| 	memlimit.SetGoMemLimitWithOpts( | ||||
| 	_, _ = maxprocs.Set() | ||||
| 	_, _ = memlimit.SetGoMemLimitWithOpts( | ||||
| 		memlimit.WithRatio(0.9), | ||||
| 		memlimit.WithProvider( | ||||
| 			memlimit.ApplyFallback( | ||||
|   | ||||
| @@ -105,3 +105,7 @@ func NewWatchOptions(opts ...WatchOption) (WatchOptions, error) { | ||||
|  | ||||
| 	return options, err | ||||
| } | ||||
|  | ||||
| func Watch(context.Context) (Watcher, error) { | ||||
| 	return nil, nil | ||||
| } | ||||
|   | ||||
| @@ -66,8 +66,13 @@ func (c *dnsConn) RemoteAddr() net.Addr { | ||||
| } | ||||
|  | ||||
| func (c *dnsConn) SetDeadline(t time.Time) error { | ||||
| 	c.SetReadDeadline(t) | ||||
| 	c.SetWriteDeadline(t) | ||||
| 	var err error | ||||
| 	if err = c.SetReadDeadline(t); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	if err = c.SetWriteDeadline(t); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -16,7 +16,6 @@ type Ticker struct { | ||||
| 	C    chan time.Time | ||||
| 	min  int64 | ||||
| 	max  int64 | ||||
| 	exp  int64 | ||||
| 	exit bool | ||||
| 	rng  rand.Rand | ||||
| } | ||||
|   | ||||
| @@ -91,7 +91,7 @@ func Merge(dst interface{}, mp map[string]interface{}, opts ...Option) error { | ||||
| 	} | ||||
|  | ||||
| 	if mapper, ok := dst.(map[string]interface{}); ok { | ||||
| 		dst = mergeMap(mapper, mp, 0) | ||||
| 		mergeMap(mapper, mp, 0) | ||||
| 		return nil | ||||
| 	} | ||||
|  | ||||
|   | ||||
| @@ -1,9 +1,38 @@ | ||||
| package reflect | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestMergeMapStringInterface(t *testing.T) { | ||||
| 	var dst interface{} //nolint:gosimple | ||||
| 	dst = map[string]interface{}{ | ||||
| 		"xx": 11, | ||||
| 	} | ||||
|  | ||||
| 	src := map[string]interface{}{ | ||||
| 		"zz": "aa", | ||||
| 	} | ||||
|  | ||||
| 	if err := Merge(dst, src); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
|  | ||||
| 	mp, ok := dst.(map[string]interface{}) | ||||
| 	if !ok || mp == nil { | ||||
| 		t.Fatalf("xxx %#+v\n", dst) | ||||
| 	} | ||||
|  | ||||
| 	if fmt.Sprintf("%v", mp["xx"]) != "11" { | ||||
| 		t.Fatalf("xxx zzzz %#+v", mp) | ||||
| 	} | ||||
|  | ||||
| 	if fmt.Sprintf("%v", mp["zz"]) != "aa" { | ||||
| 		t.Fatalf("xxx zzzz %#+v", mp) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestMergeMap(t *testing.T) { | ||||
| 	src := map[string]interface{}{ | ||||
| 		"skey1": "sval1", | ||||
|   | ||||
| @@ -56,7 +56,7 @@ type DigitalOceanMetadata struct { | ||||
| func (stfs *DigitalOceanMetadata) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	switch r.URL.Path { | ||||
| 	case "/metadata/v1.json": | ||||
| 		json.NewEncoder(w).Encode(stfs.Metadata.V1) | ||||
| 		_ = json.NewEncoder(w).Encode(stfs.Metadata.V1) | ||||
| 	default: | ||||
| 		fs := FileServer(stfs, "json", time.Now()) | ||||
| 		idx := strings.Index(r.URL.Path[1:], "/") | ||||
|   | ||||
| @@ -12,7 +12,7 @@ type EC2Metadata struct { | ||||
| 			InstanceType       string   `json:"instance-type"` | ||||
| 			LocalHostname      string   `json:"local-hostname"` | ||||
| 			LocalIPv4          string   `json:"local-ipv4"` | ||||
| 			kernelID           int      `json:"kernel-id"` | ||||
| 			KernelID           int      `json:"kernel-id"` | ||||
| 			Placement          string   `json:"placement"` | ||||
| 			AvailabilityZone   string   `json:"availability-zone"` | ||||
| 			ProductCodes       string   `json:"product-codes"` | ||||
|   | ||||
| @@ -27,7 +27,7 @@ func (fs *fs) ServeHTTP(w http.ResponseWriter, r *http.Request) { | ||||
| 	f, err := fs.Open(r.URL.Path) | ||||
| 	if err != nil { | ||||
| 		w.WriteHeader(http.StatusInternalServerError) | ||||
| 		w.Write([]byte(err.Error())) | ||||
| 		_, _ = w.Write([]byte(err.Error())) | ||||
| 		return | ||||
| 	} | ||||
| 	w.Header().Set("Content-Type", "application/octet-stream") | ||||
| @@ -67,9 +67,9 @@ func (fi *fileInfo) Name() string { | ||||
|  | ||||
| func (fi *fileInfo) Mode() os.FileMode { | ||||
| 	if strings.HasSuffix(fi.name, "/") { | ||||
| 		return os.FileMode(0755) | os.ModeDir | ||||
| 		return os.FileMode(0o755) | os.ModeDir | ||||
| 	} | ||||
| 	return os.FileMode(0644) | ||||
| 	return os.FileMode(0o644) | ||||
| } | ||||
|  | ||||
| func (fi *fileInfo) IsDir() bool { | ||||
| @@ -112,15 +112,14 @@ func (f *file) Readdir(count int) ([]os.FileInfo, error) { | ||||
| func (f *file) Seek(offset int64, whence int) (int64, error) { | ||||
| 	//	log.Printf("seek %d %d %s\n", offset, whence, f.name) | ||||
| 	switch whence { | ||||
| 	case os.SEEK_SET: | ||||
| 	case io.SeekStart: | ||||
| 		f.offset = offset | ||||
| 	case os.SEEK_CUR: | ||||
| 	case io.SeekCurrent: | ||||
| 		f.offset += offset | ||||
| 	case os.SEEK_END: | ||||
| 	case io.SeekEnd: | ||||
| 		f.offset = int64(len(f.data)) + offset | ||||
| 	} | ||||
| 	return f.offset, nil | ||||
|  | ||||
| } | ||||
|  | ||||
| func (f *file) Stat() (os.FileInfo, error) { | ||||
| @@ -222,6 +221,7 @@ func getValue(name string, iface interface{}, tag string) ([]byte, error) { | ||||
| 	return nil, fmt.Errorf("failed to find %s in interface %T", name, iface) | ||||
| } | ||||
|  | ||||
| /* | ||||
| func hasValidType(obj interface{}, types []reflect.Kind) bool { | ||||
| 	for _, t := range types { | ||||
| 		if reflect.TypeOf(obj).Kind() == t { | ||||
| @@ -231,6 +231,7 @@ func hasValidType(obj interface{}, types []reflect.Kind) bool { | ||||
|  | ||||
| 	return false | ||||
| } | ||||
| */ | ||||
|  | ||||
| func reflectValue(obj interface{}) reflect.Value { | ||||
| 	var val reflect.Value | ||||
|   | ||||
| @@ -2,7 +2,7 @@ package structfs | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"io/ioutil" | ||||
| 	"io" | ||||
| 	"net/http" | ||||
| 	"reflect" | ||||
| 	"testing" | ||||
| @@ -61,7 +61,7 @@ var doOrig = []byte(`{ | ||||
| } | ||||
| `) | ||||
|  | ||||
| func server(t *testing.T) { | ||||
| func server(t *testing.T, ch chan error) { | ||||
| 	stfs := DigitalOceanMetadata{} | ||||
| 	err := json.Unmarshal(doOrig, &stfs.Metadata.V1) | ||||
| 	if err != nil { | ||||
| @@ -71,7 +71,7 @@ func server(t *testing.T) { | ||||
| 	http.Handle("/metadata/v1/", FileServer(&stfs, "json", time.Now())) | ||||
| 	http.Handle("/metadata/v1.json", &stfs) | ||||
| 	go func() { | ||||
| 		t.Fatal(http.ListenAndServe("127.0.0.1:8080", nil)) | ||||
| 		ch <- http.ListenAndServe("127.0.0.1:8080", nil) | ||||
| 	}() | ||||
| 	time.Sleep(2 * time.Second) | ||||
| } | ||||
| @@ -82,13 +82,14 @@ func get(path string) ([]byte, error) { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	defer res.Body.Close() | ||||
| 	return ioutil.ReadAll(res.Body) | ||||
| 	return io.ReadAll(res.Body) | ||||
| } | ||||
|  | ||||
| func TestAll(t *testing.T) { | ||||
| 	server(t) | ||||
| 	ch := make(chan error) | ||||
| 	server(t, ch) | ||||
|  | ||||
| 	var tests = []struct { | ||||
| 	tests := []struct { | ||||
| 		in  string | ||||
| 		out string | ||||
| 	}{ | ||||
| @@ -100,6 +101,10 @@ func TestAll(t *testing.T) { | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		select { | ||||
| 		case err := <-ch: | ||||
| 			t.Fatal(err) | ||||
| 		default: | ||||
| 			buf, err := get(tt.in) | ||||
| 			if err != nil { | ||||
| 				t.Fatal(err) | ||||
| @@ -108,7 +113,12 @@ func TestAll(t *testing.T) { | ||||
| 				t.Errorf("req %s output %s not match requested %s", tt.in, string(buf), tt.out) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	select { | ||||
| 	case err := <-ch: | ||||
| 		t.Fatal(err) | ||||
| 	default: | ||||
| 		doTest, err := get("http://127.0.0.1:8080/metadata/v1.json") | ||||
| 		if err != nil { | ||||
| 			t.Fatal(err) | ||||
| @@ -130,4 +140,5 @@ func TestAll(t *testing.T) { | ||||
| 		if !reflect.DeepEqual(oSt, nSt) { | ||||
| 			t.Fatalf("%v not match %v", oSt, nSt) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|   | ||||
| @@ -14,7 +14,7 @@ type Duration int64 | ||||
|  | ||||
| func ParseDuration(s string) (time.Duration, error) { | ||||
| 	if s == "" { | ||||
| 		return 0, fmt.Errorf(`time: invalid duration "` + s + `"`) | ||||
| 		return 0, errors.New(`time: invalid duration "` + s + `"`) | ||||
| 	} | ||||
|  | ||||
| 	var p int | ||||
|   | ||||
| @@ -39,9 +39,7 @@ func newStatsMeter() { | ||||
| 	ticker := time.NewTicker(meter.DefaultMeterStatsInterval) | ||||
| 	defer ticker.Stop() | ||||
|  | ||||
| 	for { | ||||
| 		select { | ||||
| 		case <-ticker.C: | ||||
| 	for range ticker.C { | ||||
| 		poolsMu.Lock() | ||||
| 		for _, st := range pools { | ||||
| 			stats := st.Stats() | ||||
| @@ -52,7 +50,6 @@ func newStatsMeter() { | ||||
| 		} | ||||
| 		poolsMu.Unlock() | ||||
| 	} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| var ( | ||||
|   | ||||
		Reference in New Issue
	
	Block a user