mirror of
				https://gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud
				synced 2025-10-25 17:11:43 +00:00 
			
		
		
		
	Compare commits
	
		
			13 Commits
		
	
	
		
			v1.1.0
			...
			7-move-to-
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 0551ffbca1 | |||
| a54ee4c2e6 | |||
| 2d66aa5161 | |||
| 3e33481fb4 | |||
| 4925b52a0e | |||
| 014ac7ff7e | |||
| 2953afd160 | |||
| dd062fe989 | |||
| 4f50c4817a | |||
| 014a0c6653 | |||
| 557d7213c3 | |||
| c793018d74 | |||
|  | 91d59033fb | 
| @@ -1,7 +1,7 @@ | ||||
|  | ||||
|  | ||||
| [](http://gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/commits/master) | ||||
| [](http://gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/commits/master) | ||||
| [](https://gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/commits/master) | ||||
| [](https://gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/commits/master) | ||||
| [](https://goreportcard.com/report/gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud) | ||||
| [](https://godoc.org/gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud) | ||||
| # GoNextcloud | ||||
| @@ -41,7 +41,7 @@ func main() { | ||||
|     } | ||||
|     defer c.Logout() | ||||
|  | ||||
|     users, err := c.Users.List() | ||||
|     users, err := c.Users().List() | ||||
|     if err != nil { | ||||
|         panic(err) | ||||
|     } | ||||
|   | ||||
| @@ -157,41 +157,6 @@ var ( | ||||
| 		//	}, | ||||
| 		//}, | ||||
| 		// | ||||
| 		//{ | ||||
| 		//	"TestUserUpdate", | ||||
| 		//	func(t *testing.T) { | ||||
| 		//		if err := initClient(); err != nil { | ||||
| 		//			return | ||||
| 		//		} | ||||
| 		//		username := fmt.Sprintf("%s-2", config.NotExistingUser) | ||||
| 		//		err := c.Users().Create(username, password, nil) | ||||
| 		//		assert.NoError(t, err) | ||||
| 		//		user := &types.Users{ | ||||
| 		//			ID:          username, | ||||
| 		//			Displayname: strings.ToUpper(username), | ||||
| 		//			Email:       "some@address.com", | ||||
| 		//			Address:     "Main Street, City", | ||||
| 		//			Twitter:     "@me", | ||||
| 		//			Phone:       "42 42 242 424", | ||||
| 		//			Website:     "my.site.com", | ||||
| 		//		} | ||||
| 		//		err = c.Users().Update(user) | ||||
| 		//		assert.NoError(t, err) | ||||
| 		//		u, err := c.Users().Get(username) | ||||
| 		//		assert.NoError(t, err) | ||||
| 		//		o := structs.Map(user) | ||||
| 		//		r := structs.Map(u) | ||||
| 		//		for k := range o { | ||||
| 		//			if ignoredUserField(k) { | ||||
| 		//				continue | ||||
| 		//			} | ||||
| 		//			assert.Equal(t, o[k], r[k]) | ||||
| 		//		} | ||||
| 		//		// Clean up | ||||
| 		//		err = c.Users().Delete(u.ID) | ||||
| 		//		assert.NoError(t, err) | ||||
| 		//	}, | ||||
| 		//}, | ||||
| 		{ | ||||
| 			"TestUserCreateExisting", | ||||
| 			func(t *testing.T) { | ||||
| @@ -314,7 +279,7 @@ var ( | ||||
| 		{ | ||||
| 			"TestUserUpdateQuota", | ||||
| 			func(t *testing.T) { | ||||
| 				quota := 1024 * 1024 * 1024 | ||||
| 				quota := int64(1024 * 1024 * 1024) | ||||
| 				err := c.Users().UpdateQuota(config.NotExistingUser, quota) | ||||
| 				assert.NoError(t, err) | ||||
| 				// TODO : Find better verification : A never connected Users does not have quota available | ||||
| @@ -539,7 +504,7 @@ func TestGroupListDetails(t *testing.T) { | ||||
| 	if err := initClient(); err != nil { | ||||
| 		t.Fatal(err) | ||||
| 	} | ||||
| 	gs, err := c.Groups().ListDetails() | ||||
| 	gs, err := c.Groups().ListDetails("") | ||||
| 	assert.NoError(t, err) | ||||
| 	assert.NotEmpty(t, gs) | ||||
| } | ||||
|   | ||||
| @@ -23,8 +23,13 @@ func (g *Groups) List() ([]string, error) { | ||||
| } | ||||
|  | ||||
| //ListDetails lists the Nextcloud groups | ||||
| func (g *Groups) ListDetails() ([]types.Group, error) { | ||||
| 	res, err := g.c.baseRequest(http.MethodGet, routes.groups, nil, "details") | ||||
| func (g *Groups) ListDetails(search string) ([]types.Group, error) { | ||||
| 	ro := &req.RequestOptions{ | ||||
| 		Params: map[string]string{ | ||||
| 			"search": search, | ||||
| 		}, | ||||
| 	} | ||||
| 	res, err := g.c.baseRequest(http.MethodGet, routes.groups, ro, "details") | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										15
									
								
								shares.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								shares.go
									
									
									
									
									
								
							| @@ -95,13 +95,13 @@ func (s *Shares) Delete(shareID int) error { | ||||
| // Update update share details | ||||
| // expireDate expireDate expects a well formatted date string, e.g. ‘YYYY-MM-DD’ | ||||
| func (s *Shares) Update(shareUpdate types.ShareUpdate) error { | ||||
| 	errs := make(chan types.UpdateError) | ||||
| 	errs := make(chan *types.UpdateError) | ||||
| 	var wg sync.WaitGroup | ||||
| 	wg.Add(4) | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		if err := s.UpdatePassword(shareUpdate.ShareID, shareUpdate.Password); err != nil { | ||||
| 			errs <- types.UpdateError{ | ||||
| 			errs <- &types.UpdateError{ | ||||
| 				Field: "password", | ||||
| 				Error: err, | ||||
| 			} | ||||
| @@ -110,7 +110,7 @@ func (s *Shares) Update(shareUpdate types.ShareUpdate) error { | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		if err := s.UpdateExpireDate(shareUpdate.ShareID, shareUpdate.ExpireDate); err != nil { | ||||
| 			errs <- types.UpdateError{ | ||||
| 			errs <- &types.UpdateError{ | ||||
| 				Field: "expireDate", | ||||
| 				Error: err, | ||||
| 			} | ||||
| @@ -119,7 +119,7 @@ func (s *Shares) Update(shareUpdate types.ShareUpdate) error { | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		if err := s.UpdatePermissions(shareUpdate.ShareID, shareUpdate.Permissions); err != nil { | ||||
| 			errs <- types.UpdateError{ | ||||
| 			errs <- &types.UpdateError{ | ||||
| 				Field: "permissions", | ||||
| 				Error: err, | ||||
| 			} | ||||
| @@ -128,7 +128,7 @@ func (s *Shares) Update(shareUpdate types.ShareUpdate) error { | ||||
| 	go func() { | ||||
| 		defer wg.Done() | ||||
| 		if err := s.UpdatePublicUpload(shareUpdate.ShareID, shareUpdate.PublicUpload); err != nil { | ||||
| 			errs <- types.UpdateError{ | ||||
| 			errs <- &types.UpdateError{ | ||||
| 				Field: "publicUpload", | ||||
| 				Error: err, | ||||
| 			} | ||||
| @@ -138,7 +138,10 @@ func (s *Shares) Update(shareUpdate types.ShareUpdate) error { | ||||
| 		wg.Wait() | ||||
| 		close(errs) | ||||
| 	}() | ||||
| 	return types.NewUpdateError(errs) | ||||
| 	if err := types.NewUpdateError(errs); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| //UpdateExpireDate updates the share's expire date | ||||
|   | ||||
| @@ -40,17 +40,16 @@ func (e *UserUpdateError) Error() string { | ||||
| 	for k, e := range e.Errors { | ||||
| 		errors = append(errors, fmt.Sprintf("%s: %v", k, e)) | ||||
| 	} | ||||
| 	return strings.Join(errors, ",") | ||||
| 	return strings.Join(errors, ", ") | ||||
| } | ||||
|  | ||||
| //NewUpdateError returns an UpdateError based on an UpdateError channel | ||||
| func NewUpdateError(errors chan UpdateError) *UserUpdateError { | ||||
| 	var ue UserUpdateError | ||||
| func NewUpdateError(errors chan *UpdateError) *UserUpdateError { | ||||
| 	ue := UserUpdateError{map[string]error{}} | ||||
| 	for e := range errors { | ||||
| 		if ue.Errors == nil { | ||||
| 			ue.Errors = map[string]error{e.Field: e.Error} | ||||
| 		if e != nil { | ||||
| 			ue.Errors[e.Field] = e.Error | ||||
| 		} | ||||
| 		ue.Errors[e.Field] = e.Error | ||||
| 	} | ||||
| 	if len(ue.Errors) > 0 { | ||||
| 		return &ue | ||||
|   | ||||
							
								
								
									
										46
									
								
								types/errors_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								types/errors_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | ||||
| package types | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"strconv" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestUserUpdateErrors(t *testing.T) { | ||||
| 	exp := map[string]error{} | ||||
| 	errs := make(chan *UpdateError) | ||||
| 	go func() { | ||||
| 		for i := 0; i < 10; i++ { | ||||
| 			f := strconv.Itoa(i) | ||||
| 			e := errors.New(f) | ||||
| 			err := UpdateError{ | ||||
| 				Field: f, | ||||
| 				Error: e, | ||||
| 			} | ||||
| 			exp[f] = e | ||||
| 			errs <- &err | ||||
| 		} | ||||
| 		close(errs) | ||||
| 	}() | ||||
| 	uerrs := NewUpdateError(errs) | ||||
| 	assert.Equal(t, exp, uerrs.Errors) | ||||
| 	assert.NotEmpty(t, uerrs.Error()) | ||||
| } | ||||
|  | ||||
| func TestUserUpdateErrorsNil(t *testing.T) { | ||||
| 	var wg sync.WaitGroup | ||||
| 	errs := make(chan *UpdateError) | ||||
| 	wg.Add(1) | ||||
| 	go func() { | ||||
| 		errs <- nil | ||||
| 		wg.Done() | ||||
| 	}() | ||||
| 	go func() { | ||||
| 		wg.Wait() | ||||
| 		close(errs) | ||||
| 	}() | ||||
| 	uerrs := NewUpdateError(errs) | ||||
| 	assert.Nil(t, uerrs) | ||||
| } | ||||
| @@ -9,6 +9,8 @@ type Client interface { | ||||
| 	Shares() Shares | ||||
| 	Users() Users | ||||
| 	Groups() Groups | ||||
| 	Login(username string, password string) error | ||||
| 	Logout() error | ||||
| } | ||||
|  | ||||
| type Auth interface { | ||||
| @@ -40,7 +42,7 @@ type AppsConfig interface { | ||||
| //Groups available methods | ||||
| type Groups interface { | ||||
| 	List() ([]string, error) | ||||
| 	ListDetails() ([]Group, error) | ||||
| 	ListDetails(search string) ([]Group, error) | ||||
| 	Users(name string) ([]string, error) | ||||
| 	Search(search string) ([]string, error) | ||||
| 	Create(name string) error | ||||
| @@ -113,7 +115,7 @@ type Users interface { | ||||
| 	UpdateWebSite(name string, website string) error | ||||
| 	UpdateTwitter(name string, twitter string) error | ||||
| 	UpdatePassword(name string, password string) error | ||||
| 	UpdateQuota(name string, quota int) error | ||||
| 	UpdateQuota(name string, quota int64) error | ||||
| 	GroupList(name string) ([]string, error) | ||||
| 	GroupAdd(name string, group string) error | ||||
| 	GroupRemove(name string, group string) error | ||||
|   | ||||
							
								
								
									
										129
									
								
								types/mocks/Apps.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										129
									
								
								types/mocks/Apps.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,129 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // Apps is an autogenerated mock type for the Apps type | ||||
| type Apps struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // Disable provides a mock function with given fields: name | ||||
| func (_m *Apps) Disable(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Enable provides a mock function with given fields: name | ||||
| func (_m *Apps) Enable(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Infos provides a mock function with given fields: name | ||||
| func (_m *Apps) Infos(name string) (types.App, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 types.App | ||||
| 	if rf, ok := ret.Get(0).(func(string) types.App); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(types.App) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *Apps) List() ([]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func() []string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // ListDisabled provides a mock function with given fields: | ||||
| func (_m *Apps) ListDisabled() ([]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func() []string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // ListEnabled provides a mock function with given fields: | ||||
| func (_m *Apps) ListEnabled() ([]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func() []string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
							
								
								
									
										151
									
								
								types/mocks/AppsConfig.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								types/mocks/AppsConfig.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
|  | ||||
| // AppsConfig is an autogenerated mock type for the AppsConfig type | ||||
| type AppsConfig struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // DeleteValue provides a mock function with given fields: id, key, value | ||||
| func (_m *AppsConfig) DeleteValue(id string, key string, value string) error { | ||||
| 	ret := _m.Called(id, key, value) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string, string) error); ok { | ||||
| 		r0 = rf(id, key, value) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Details provides a mock function with given fields: appID | ||||
| func (_m *AppsConfig) Details(appID string) (map[string]string, error) { | ||||
| 	ret := _m.Called(appID) | ||||
|  | ||||
| 	var r0 map[string]string | ||||
| 	if rf, ok := ret.Get(0).(func(string) map[string]string); ok { | ||||
| 		r0 = rf(appID) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(map[string]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(appID) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Get provides a mock function with given fields: | ||||
| func (_m *AppsConfig) Get() (map[string]map[string]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 map[string]map[string]string | ||||
| 	if rf, ok := ret.Get(0).(func() map[string]map[string]string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(map[string]map[string]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Keys provides a mock function with given fields: id | ||||
| func (_m *AppsConfig) Keys(id string) ([]string, error) { | ||||
| 	ret := _m.Called(id) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(id) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(id) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *AppsConfig) List() ([]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func() []string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // SetValue provides a mock function with given fields: id, key, value | ||||
| func (_m *AppsConfig) SetValue(id string, key string, value string) error { | ||||
| 	ret := _m.Called(id, key, value) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string, string) error); ok { | ||||
| 		r0 = rf(id, key, value) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Value provides a mock function with given fields: id, key | ||||
| func (_m *AppsConfig) Value(id string, key string) (string, error) { | ||||
| 	ret := _m.Called(id, key) | ||||
|  | ||||
| 	var r0 string | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) string); ok { | ||||
| 		r0 = rf(id, key) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(string) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string, string) error); ok { | ||||
| 		r1 = rf(id, key) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
							
								
								
									
										38
									
								
								types/mocks/Auth.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								types/mocks/Auth.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
|  | ||||
| // Auth is an autogenerated mock type for the Auth type | ||||
| type Auth struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // Login provides a mock function with given fields: username, password | ||||
| func (_m *Auth) Login(username string, password string) error { | ||||
| 	ret := _m.Called(username, password) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(username, password) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Logout provides a mock function with given fields: | ||||
| func (_m *Auth) Logout() error { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func() error); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
							
								
								
									
										151
									
								
								types/mocks/Client.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								types/mocks/Client.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,151 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // Client is an autogenerated mock type for the Client type | ||||
| type Client struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // Apps provides a mock function with given fields: | ||||
| func (_m *Client) Apps() types.Apps { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.Apps | ||||
| 	if rf, ok := ret.Get(0).(func() types.Apps); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.Apps) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // AppsConfig provides a mock function with given fields: | ||||
| func (_m *Client) AppsConfig() types.AppsConfig { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.AppsConfig | ||||
| 	if rf, ok := ret.Get(0).(func() types.AppsConfig); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.AppsConfig) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // GroupFolders provides a mock function with given fields: | ||||
| func (_m *Client) GroupFolders() types.GroupFolders { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.GroupFolders | ||||
| 	if rf, ok := ret.Get(0).(func() types.GroupFolders); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.GroupFolders) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Groups provides a mock function with given fields: | ||||
| func (_m *Client) Groups() types.Groups { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.Groups | ||||
| 	if rf, ok := ret.Get(0).(func() types.Groups); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.Groups) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Login provides a mock function with given fields: username, password | ||||
| func (_m *Client) Login(username string, password string) error { | ||||
| 	ret := _m.Called(username, password) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(username, password) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Logout provides a mock function with given fields: | ||||
| func (_m *Client) Logout() error { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func() error); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Notifications provides a mock function with given fields: | ||||
| func (_m *Client) Notifications() types.Notifications { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.Notifications | ||||
| 	if rf, ok := ret.Get(0).(func() types.Notifications); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.Notifications) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Shares provides a mock function with given fields: | ||||
| func (_m *Client) Shares() types.Shares { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.Shares | ||||
| 	if rf, ok := ret.Get(0).(func() types.Shares); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.Shares) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Users provides a mock function with given fields: | ||||
| func (_m *Client) Users() types.Users { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 types.Users | ||||
| 	if rf, ok := ret.Get(0).(func() types.Users); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(types.Users) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
							
								
								
									
										146
									
								
								types/mocks/GroupFolders.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								types/mocks/GroupFolders.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,146 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // GroupFolders is an autogenerated mock type for the GroupFolders type | ||||
| type GroupFolders struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // AddGroup provides a mock function with given fields: folderID, groupName | ||||
| func (_m *GroupFolders) AddGroup(folderID int, groupName string) error { | ||||
| 	ret := _m.Called(folderID, groupName) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, string) error); ok { | ||||
| 		r0 = rf(folderID, groupName) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Create provides a mock function with given fields: name | ||||
| func (_m *GroupFolders) Create(name string) (int, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 int | ||||
| 	if rf, ok := ret.Get(0).(func(string) int); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(int) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Get provides a mock function with given fields: id | ||||
| func (_m *GroupFolders) Get(id int) (types.GroupFolder, error) { | ||||
| 	ret := _m.Called(id) | ||||
|  | ||||
| 	var r0 types.GroupFolder | ||||
| 	if rf, ok := ret.Get(0).(func(int) types.GroupFolder); ok { | ||||
| 		r0 = rf(id) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(types.GroupFolder) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(int) error); ok { | ||||
| 		r1 = rf(id) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *GroupFolders) List() (map[int]types.GroupFolder, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 map[int]types.GroupFolder | ||||
| 	if rf, ok := ret.Get(0).(func() map[int]types.GroupFolder); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(map[int]types.GroupFolder) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // RemoveGroup provides a mock function with given fields: folderID, groupName | ||||
| func (_m *GroupFolders) RemoveGroup(folderID int, groupName string) error { | ||||
| 	ret := _m.Called(folderID, groupName) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, string) error); ok { | ||||
| 		r0 = rf(folderID, groupName) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Rename provides a mock function with given fields: groupID, name | ||||
| func (_m *GroupFolders) Rename(groupID int, name string) error { | ||||
| 	ret := _m.Called(groupID, name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, string) error); ok { | ||||
| 		r0 = rf(groupID, name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // SetGroupPermissions provides a mock function with given fields: folderID, groupName, permission | ||||
| func (_m *GroupFolders) SetGroupPermissions(folderID int, groupName string, permission types.SharePermission) error { | ||||
| 	ret := _m.Called(folderID, groupName, permission) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, string, types.SharePermission) error); ok { | ||||
| 		r0 = rf(folderID, groupName, permission) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // SetQuota provides a mock function with given fields: folderID, quota | ||||
| func (_m *GroupFolders) SetQuota(folderID int, quota int) error { | ||||
| 	ret := _m.Called(folderID, quota) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, int) error); ok { | ||||
| 		r0 = rf(folderID, quota) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
							
								
								
									
										154
									
								
								types/mocks/Groups.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										154
									
								
								types/mocks/Groups.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,154 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // Groups is an autogenerated mock type for the Groups type | ||||
| type Groups struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // Create provides a mock function with given fields: name | ||||
| func (_m *Groups) Create(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Delete provides a mock function with given fields: name | ||||
| func (_m *Groups) Delete(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *Groups) List() ([]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func() []string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // ListDetails provides a mock function with given fields: search | ||||
| func (_m *Groups) ListDetails(search string) ([]types.Group, error) { | ||||
| 	ret := _m.Called(search) | ||||
|  | ||||
| 	var r0 []types.Group | ||||
| 	if rf, ok := ret.Get(0).(func(string) []types.Group); ok { | ||||
| 		r0 = rf(search) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]types.Group) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(search) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Search provides a mock function with given fields: search | ||||
| func (_m *Groups) Search(search string) ([]string, error) { | ||||
| 	ret := _m.Called(search) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(search) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(search) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // SubAdminList provides a mock function with given fields: name | ||||
| func (_m *Groups) SubAdminList(name string) ([]string, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Users provides a mock function with given fields: name | ||||
| func (_m *Groups) Users(name string) ([]string, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
							
								
								
									
										125
									
								
								types/mocks/Notifications.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										125
									
								
								types/mocks/Notifications.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,125 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // Notifications is an autogenerated mock type for the Notifications type | ||||
| type Notifications struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // AdminAvailable provides a mock function with given fields: | ||||
| func (_m *Notifications) AdminAvailable() error { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func() error); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Available provides a mock function with given fields: | ||||
| func (_m *Notifications) Available() error { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func() error); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Create provides a mock function with given fields: userID, title, message | ||||
| func (_m *Notifications) Create(userID string, title string, message string) error { | ||||
| 	ret := _m.Called(userID, title, message) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string, string) error); ok { | ||||
| 		r0 = rf(userID, title, message) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Delete provides a mock function with given fields: id | ||||
| func (_m *Notifications) Delete(id int) error { | ||||
| 	ret := _m.Called(id) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int) error); ok { | ||||
| 		r0 = rf(id) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // DeleteAll provides a mock function with given fields: | ||||
| func (_m *Notifications) DeleteAll() error { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func() error); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Get provides a mock function with given fields: id | ||||
| func (_m *Notifications) Get(id int) (types.Notification, error) { | ||||
| 	ret := _m.Called(id) | ||||
|  | ||||
| 	var r0 types.Notification | ||||
| 	if rf, ok := ret.Get(0).(func(int) types.Notification); ok { | ||||
| 		r0 = rf(id) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(types.Notification) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(int) error); ok { | ||||
| 		r1 = rf(id) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *Notifications) List() ([]types.Notification, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []types.Notification | ||||
| 	if rf, ok := ret.Get(0).(func() []types.Notification); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]types.Notification) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
							
								
								
									
										183
									
								
								types/mocks/Shares.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										183
									
								
								types/mocks/Shares.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,183 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // Shares is an autogenerated mock type for the Shares type | ||||
| type Shares struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // Create provides a mock function with given fields: path, shareType, permission, shareWith, publicUpload, password | ||||
| func (_m *Shares) Create(path string, shareType types.ShareType, permission types.SharePermission, shareWith string, publicUpload bool, password string) (types.Share, error) { | ||||
| 	ret := _m.Called(path, shareType, permission, shareWith, publicUpload, password) | ||||
|  | ||||
| 	var r0 types.Share | ||||
| 	if rf, ok := ret.Get(0).(func(string, types.ShareType, types.SharePermission, string, bool, string) types.Share); ok { | ||||
| 		r0 = rf(path, shareType, permission, shareWith, publicUpload, password) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(types.Share) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string, types.ShareType, types.SharePermission, string, bool, string) error); ok { | ||||
| 		r1 = rf(path, shareType, permission, shareWith, publicUpload, password) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Delete provides a mock function with given fields: shareID | ||||
| func (_m *Shares) Delete(shareID int) error { | ||||
| 	ret := _m.Called(shareID) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int) error); ok { | ||||
| 		r0 = rf(shareID) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Get provides a mock function with given fields: shareID | ||||
| func (_m *Shares) Get(shareID string) (types.Share, error) { | ||||
| 	ret := _m.Called(shareID) | ||||
|  | ||||
| 	var r0 types.Share | ||||
| 	if rf, ok := ret.Get(0).(func(string) types.Share); ok { | ||||
| 		r0 = rf(shareID) | ||||
| 	} else { | ||||
| 		r0 = ret.Get(0).(types.Share) | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(shareID) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // GetFromPath provides a mock function with given fields: path, reshares, subfiles | ||||
| func (_m *Shares) GetFromPath(path string, reshares bool, subfiles bool) ([]types.Share, error) { | ||||
| 	ret := _m.Called(path, reshares, subfiles) | ||||
|  | ||||
| 	var r0 []types.Share | ||||
| 	if rf, ok := ret.Get(0).(func(string, bool, bool) []types.Share); ok { | ||||
| 		r0 = rf(path, reshares, subfiles) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]types.Share) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string, bool, bool) error); ok { | ||||
| 		r1 = rf(path, reshares, subfiles) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *Shares) List() ([]types.Share, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []types.Share | ||||
| 	if rf, ok := ret.Get(0).(func() []types.Share); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]types.Share) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Update provides a mock function with given fields: shareUpdate | ||||
| func (_m *Shares) Update(shareUpdate types.ShareUpdate) error { | ||||
| 	ret := _m.Called(shareUpdate) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(types.ShareUpdate) error); ok { | ||||
| 		r0 = rf(shareUpdate) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateExpireDate provides a mock function with given fields: shareID, expireDate | ||||
| func (_m *Shares) UpdateExpireDate(shareID int, expireDate string) error { | ||||
| 	ret := _m.Called(shareID, expireDate) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, string) error); ok { | ||||
| 		r0 = rf(shareID, expireDate) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdatePassword provides a mock function with given fields: shareID, password | ||||
| func (_m *Shares) UpdatePassword(shareID int, password string) error { | ||||
| 	ret := _m.Called(shareID, password) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, string) error); ok { | ||||
| 		r0 = rf(shareID, password) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdatePermissions provides a mock function with given fields: shareID, permissions | ||||
| func (_m *Shares) UpdatePermissions(shareID int, permissions types.SharePermission) error { | ||||
| 	ret := _m.Called(shareID, permissions) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, types.SharePermission) error); ok { | ||||
| 		r0 = rf(shareID, permissions) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdatePublicUpload provides a mock function with given fields: shareID, public | ||||
| func (_m *Shares) UpdatePublicUpload(shareID int, public bool) error { | ||||
| 	ret := _m.Called(shareID, public) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(int, bool) error); ok { | ||||
| 		r0 = rf(shareID, public) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
							
								
								
									
										436
									
								
								types/mocks/Users.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										436
									
								
								types/mocks/Users.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,436 @@ | ||||
| // Code generated by mockery v1.0.0. DO NOT EDIT. | ||||
|  | ||||
| package mocks | ||||
|  | ||||
| import mock "github.com/stretchr/testify/mock" | ||||
| import types "gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
|  | ||||
| // Users is an autogenerated mock type for the Users type | ||||
| type Users struct { | ||||
| 	mock.Mock | ||||
| } | ||||
|  | ||||
| // Create provides a mock function with given fields: username, password, user | ||||
| func (_m *Users) Create(username string, password string, user *types.UserDetails) error { | ||||
| 	ret := _m.Called(username, password, user) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string, *types.UserDetails) error); ok { | ||||
| 		r0 = rf(username, password, user) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // CreateBatchWithoutPassword provides a mock function with given fields: users | ||||
| func (_m *Users) CreateBatchWithoutPassword(users []types.User) error { | ||||
| 	ret := _m.Called(users) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func([]types.User) error); ok { | ||||
| 		r0 = rf(users) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // CreateWithoutPassword provides a mock function with given fields: username, email, displayName, quota, language, groups | ||||
| func (_m *Users) CreateWithoutPassword(username string, email string, displayName string, quota string, language string, groups ...string) error { | ||||
| 	_va := make([]interface{}, len(groups)) | ||||
| 	for _i := range groups { | ||||
| 		_va[_i] = groups[_i] | ||||
| 	} | ||||
| 	var _ca []interface{} | ||||
| 	_ca = append(_ca, username, email, displayName, quota, language) | ||||
| 	_ca = append(_ca, _va...) | ||||
| 	ret := _m.Called(_ca...) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string, string, string, string, ...string) error); ok { | ||||
| 		r0 = rf(username, email, displayName, quota, language, groups...) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Delete provides a mock function with given fields: name | ||||
| func (_m *Users) Delete(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Disable provides a mock function with given fields: name | ||||
| func (_m *Users) Disable(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Enable provides a mock function with given fields: name | ||||
| func (_m *Users) Enable(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Get provides a mock function with given fields: name | ||||
| func (_m *Users) Get(name string) (*types.UserDetails, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 *types.UserDetails | ||||
| 	if rf, ok := ret.Get(0).(func(string) *types.UserDetails); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(*types.UserDetails) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // GroupAdd provides a mock function with given fields: name, group | ||||
| func (_m *Users) GroupAdd(name string, group string) error { | ||||
| 	ret := _m.Called(name, group) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, group) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // GroupDemote provides a mock function with given fields: name, group | ||||
| func (_m *Users) GroupDemote(name string, group string) error { | ||||
| 	ret := _m.Called(name, group) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, group) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // GroupList provides a mock function with given fields: name | ||||
| func (_m *Users) GroupList(name string) ([]string, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // GroupPromote provides a mock function with given fields: name, group | ||||
| func (_m *Users) GroupPromote(name string, group string) error { | ||||
| 	ret := _m.Called(name, group) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, group) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // GroupRemove provides a mock function with given fields: name, group | ||||
| func (_m *Users) GroupRemove(name string, group string) error { | ||||
| 	ret := _m.Called(name, group) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, group) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // GroupSubAdminList provides a mock function with given fields: name | ||||
| func (_m *Users) GroupSubAdminList(name string) ([]string, error) { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(name) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // List provides a mock function with given fields: | ||||
| func (_m *Users) List() ([]string, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func() []string); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // ListDetails provides a mock function with given fields: | ||||
| func (_m *Users) ListDetails() (map[string]types.UserDetails, error) { | ||||
| 	ret := _m.Called() | ||||
|  | ||||
| 	var r0 map[string]types.UserDetails | ||||
| 	if rf, ok := ret.Get(0).(func() map[string]types.UserDetails); ok { | ||||
| 		r0 = rf() | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).(map[string]types.UserDetails) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func() error); ok { | ||||
| 		r1 = rf() | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // Search provides a mock function with given fields: search | ||||
| func (_m *Users) Search(search string) ([]string, error) { | ||||
| 	ret := _m.Called(search) | ||||
|  | ||||
| 	var r0 []string | ||||
| 	if rf, ok := ret.Get(0).(func(string) []string); ok { | ||||
| 		r0 = rf(search) | ||||
| 	} else { | ||||
| 		if ret.Get(0) != nil { | ||||
| 			r0 = ret.Get(0).([]string) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	var r1 error | ||||
| 	if rf, ok := ret.Get(1).(func(string) error); ok { | ||||
| 		r1 = rf(search) | ||||
| 	} else { | ||||
| 		r1 = ret.Error(1) | ||||
| 	} | ||||
|  | ||||
| 	return r0, r1 | ||||
| } | ||||
|  | ||||
| // SendWelcomeEmail provides a mock function with given fields: name | ||||
| func (_m *Users) SendWelcomeEmail(name string) error { | ||||
| 	ret := _m.Called(name) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string) error); ok { | ||||
| 		r0 = rf(name) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // Update provides a mock function with given fields: user | ||||
| func (_m *Users) Update(user *types.UserDetails) error { | ||||
| 	ret := _m.Called(user) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(*types.UserDetails) error); ok { | ||||
| 		r0 = rf(user) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateAddress provides a mock function with given fields: name, address | ||||
| func (_m *Users) UpdateAddress(name string, address string) error { | ||||
| 	ret := _m.Called(name, address) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, address) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateDisplayName provides a mock function with given fields: name, displayName | ||||
| func (_m *Users) UpdateDisplayName(name string, displayName string) error { | ||||
| 	ret := _m.Called(name, displayName) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, displayName) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateEmail provides a mock function with given fields: name, email | ||||
| func (_m *Users) UpdateEmail(name string, email string) error { | ||||
| 	ret := _m.Called(name, email) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, email) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdatePassword provides a mock function with given fields: name, password | ||||
| func (_m *Users) UpdatePassword(name string, password string) error { | ||||
| 	ret := _m.Called(name, password) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, password) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdatePhone provides a mock function with given fields: name, phone | ||||
| func (_m *Users) UpdatePhone(name string, phone string) error { | ||||
| 	ret := _m.Called(name, phone) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, phone) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateQuota provides a mock function with given fields: name, quota | ||||
| func (_m *Users) UpdateQuota(name string, quota int64) error { | ||||
| 	ret := _m.Called(name, quota) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, int64) error); ok { | ||||
| 		r0 = rf(name, quota) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateTwitter provides a mock function with given fields: name, twitter | ||||
| func (_m *Users) UpdateTwitter(name string, twitter string) error { | ||||
| 	ret := _m.Called(name, twitter) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, twitter) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
|  | ||||
| // UpdateWebSite provides a mock function with given fields: name, website | ||||
| func (_m *Users) UpdateWebSite(name string, website string) error { | ||||
| 	ret := _m.Called(name, website) | ||||
|  | ||||
| 	var r0 error | ||||
| 	if rf, ok := ret.Get(0).(func(string, string) error); ok { | ||||
| 		r0 = rf(name, website) | ||||
| 	} else { | ||||
| 		r0 = ret.Error(0) | ||||
| 	} | ||||
|  | ||||
| 	return r0 | ||||
| } | ||||
| @@ -1,5 +1,7 @@ | ||||
| package types | ||||
|  | ||||
| import "strconv" | ||||
|  | ||||
| //User encapsulate the data needed to create a new Nextcloud's User | ||||
| type User struct { | ||||
| 	Username    string | ||||
| @@ -33,8 +35,15 @@ type UserDetails struct { | ||||
|  | ||||
| type Quota struct { | ||||
| 	Free     int64   `json:"free"` | ||||
| 	Used     int     `json:"used"` | ||||
| 	Used     int64   `json:"used"` | ||||
| 	Total    int64   `json:"total"` | ||||
| 	Relative float64 `json:"relative"` | ||||
| 	Quota    int     `json:"quota"` | ||||
| 	Quota    int64   `json:"quota"` | ||||
| } | ||||
|  | ||||
| func (q *Quota) String() string { | ||||
| 	if q.Quota < 0 { | ||||
| 		return "none" | ||||
| 	} | ||||
| 	return strconv.FormatInt(q.Quota, 10) | ||||
| } | ||||
|   | ||||
							
								
								
									
										57
									
								
								update_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								update_test.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| package gonextcloud | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"github.com/fatih/structs" | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud/types" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func TestUserUpdate(t *testing.T) { | ||||
| 	if err := initClient(); err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	username := fmt.Sprintf("%s-2", config.NotExistingUser) | ||||
| 	err := c.Users().Create(username, password, nil) | ||||
| 	if err != nil { | ||||
| 		t.FailNow() | ||||
| 	} | ||||
| 	err = c.Groups().Create(config.NotExistingGroup) | ||||
| 	if err != nil { | ||||
| 		t.FailNow() | ||||
| 	} | ||||
| 	user := &types.UserDetails{ | ||||
| 		ID:          username, | ||||
| 		Displayname: strings.ToUpper(username), | ||||
| 		Email:       "some@mail.com", | ||||
| 		Quota: types.Quota{ | ||||
| 			// Unlimited | ||||
| 			Quota: -3, | ||||
| 		}, | ||||
| 		Groups: []string{config.NotExistingGroup}, | ||||
| 	} | ||||
| 	s := time.Now() | ||||
| 	err = c.Users().Update(user) | ||||
| 	e := time.Now().Sub(s) | ||||
| 	fmt.Println(e.String()) | ||||
| 	assert.NoError(t, err) | ||||
| 	u, err := c.Users().Get(username) | ||||
| 	assert.NoError(t, err) | ||||
| 	o := structs.Map(user) | ||||
| 	r := structs.Map(u) | ||||
| 	for k := range o { | ||||
| 		if ignoredUserField(k) { | ||||
| 			continue | ||||
| 		} | ||||
| 		assert.Equal(t, o[k], r[k]) | ||||
| 	} | ||||
| 	// Clean up | ||||
| 	err = c.Users().Delete(username) | ||||
| 	assert.NoError(t, err) | ||||
| 	err = c.Groups().Delete(config.NotExistingGroup) | ||||
| 	assert.NoError(t, err) | ||||
|  | ||||
| } | ||||
							
								
								
									
										123
									
								
								users.go
									
									
									
									
									
								
							
							
						
						
									
										123
									
								
								users.go
									
									
									
									
									
								
							| @@ -2,7 +2,6 @@ package gonextcloud | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"github.com/fatih/structs" | ||||
| 	req "github.com/levigross/grequests" | ||||
| 	"github.com/pkg/errors" | ||||
| 	"github.com/sirupsen/logrus" | ||||
| @@ -10,7 +9,6 @@ import ( | ||||
| 	"net/http" | ||||
| 	"net/url" | ||||
| 	"path" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| ) | ||||
| @@ -190,29 +188,112 @@ func (u *Users) SendWelcomeEmail(name string) error { | ||||
| } | ||||
|  | ||||
| //Update takes a *types.Users struct to update the user's information | ||||
| // Updatable fields: Email, Displayname, Phone, Address, Website, Twitter, Quota, Groups | ||||
| func (u *Users) Update(user *types.UserDetails) error { | ||||
| 	m := structs.Map(user) | ||||
| 	errs := make(chan types.UpdateError) | ||||
| 	// Get user to update only modified fields | ||||
| 	original, err := u.Get(user.ID) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	errs := make(chan *types.UpdateError) | ||||
| 	var wg sync.WaitGroup | ||||
| 	for k := range m { | ||||
| 		if !ignoredUserField(k) && m[k].(string) != "" { | ||||
| 	update := func(key string, value string) { | ||||
| 		defer wg.Done() | ||||
| 		if err := u.updateAttribute(user.ID, strings.ToLower(key), value); err != nil { | ||||
| 			errs <- &types.UpdateError{ | ||||
| 				Field: key, | ||||
| 				Error: err, | ||||
| 			} | ||||
| 		} | ||||
| 		errs <- nil | ||||
| 	} | ||||
| 	//  Email | ||||
| 	if user.Email != original.Email { | ||||
| 		wg.Add(1) | ||||
| 		go update("Email", user.Email) | ||||
| 	} | ||||
| 	//  Displayname | ||||
| 	if user.Displayname != original.Displayname { | ||||
| 		wg.Add(1) | ||||
| 		go update("Displayname", user.Displayname) | ||||
| 	} | ||||
| 	//  Phone | ||||
| 	if user.Phone != original.Phone { | ||||
| 		wg.Add(1) | ||||
| 		go update("Phone", user.Phone) | ||||
| 	} | ||||
| 	//  Address | ||||
| 	if user.Address != original.Address { | ||||
| 		wg.Add(1) | ||||
| 		go update("Address", user.Address) | ||||
| 	} | ||||
| 	//  Website | ||||
| 	if user.Website != original.Website { | ||||
| 		wg.Add(1) | ||||
| 		go update("Website", user.Website) | ||||
| 	} | ||||
| 	//  Twitter | ||||
| 	if user.Twitter != original.Twitter { | ||||
| 		wg.Add(1) | ||||
| 		go update("Twitter", user.Twitter) | ||||
| 	} | ||||
| 	//  Quota | ||||
| 	if user.Quota.Quota != original.Quota.Quota { | ||||
| 		var value string | ||||
| 		// If empty | ||||
| 		if user.Quota == (types.Quota{}) { | ||||
| 			value = "default" | ||||
| 		} else { | ||||
| 			value = user.Quota.String() | ||||
| 		} | ||||
| 		wg.Add(1) | ||||
| 		go update("Quota", value) | ||||
| 	} | ||||
| 	// Groups | ||||
| 	// Group removed | ||||
| 	for _, g := range original.Groups { | ||||
| 		if !contains(user.Groups, g) { | ||||
| 			wg.Add(1) | ||||
| 			go func(key string, value string) { | ||||
| 			go func(gr string) { | ||||
| 				defer wg.Done() | ||||
| 				if err := u.updateAttribute(user.ID, strings.ToLower(key), value); err != nil { | ||||
| 					errs <- types.UpdateError{ | ||||
| 						Field: key, | ||||
| 				if err := u.GroupRemove(user.ID, gr); err != nil { | ||||
| 					errs <- &types.UpdateError{ | ||||
| 						Field: "Groups/" + gr, | ||||
| 						Error: err, | ||||
| 					} | ||||
| 				} | ||||
| 			}(k, m[k].(string)) | ||||
| 				errs <- nil | ||||
| 			}(g) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// Group Added | ||||
| 	for _, g := range user.Groups { | ||||
| 		if !contains(original.Groups, g) { | ||||
| 			wg.Add(1) | ||||
| 			go func(gr string) { | ||||
| 				defer wg.Done() | ||||
| 				if err := u.GroupAdd(user.ID, gr); err != nil { | ||||
| 					errs <- &types.UpdateError{ | ||||
| 						Field: "Groups/" + gr, | ||||
| 						Error: err, | ||||
| 					} | ||||
| 				} | ||||
| 				errs <- nil | ||||
| 			}(g) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	go func() { | ||||
| 		wg.Wait() | ||||
| 		close(errs) | ||||
| 	}() | ||||
| 	return types.NewUpdateError(errs) | ||||
| 	// Warning : we actually need to check the *err | ||||
| 	if err := types.NewUpdateError(errs); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| //UpdateEmail update the user's email | ||||
| @@ -250,9 +331,10 @@ func (u *Users) UpdatePassword(name string, password string) error { | ||||
| 	return u.updateAttribute(name, "password", password) | ||||
| } | ||||
|  | ||||
| //UpdateQuota update the user's quota (bytes) | ||||
| func (u *Users) UpdateQuota(name string, quota int) error { | ||||
| 	return u.updateAttribute(name, "quota", strconv.Itoa(quota)) | ||||
| //UpdateQuota update the user's quota (bytes). Set negative quota for unlimited | ||||
| func (u *Users) UpdateQuota(name string, quota int64) error { | ||||
| 	q := types.Quota{Quota: quota} | ||||
| 	return u.updateAttribute(name, "quota", q.String()) | ||||
| } | ||||
|  | ||||
| //GroupList lists the user's groups | ||||
| @@ -338,9 +420,18 @@ func (u *Users) baseRequest(method string, ro *req.RequestOptions, subRoutes ... | ||||
| } | ||||
|  | ||||
| func ignoredUserField(key string) bool { | ||||
| 	keys := []string{"ID", "Quota", "Enabled", "Groups", "Language"} | ||||
| 	keys := []string{"Email", "Displayname", "Phone", "Address", "Website", "Twitter", "Quota", "Groups"} | ||||
| 	for _, k := range keys { | ||||
| 		if key == k { | ||||
| 			return false | ||||
| 		} | ||||
| 	} | ||||
| 	return true | ||||
| } | ||||
|  | ||||
| func contains(slice []string, e string) bool { | ||||
| 	for _, s := range slice { | ||||
| 		if e == s { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
|   | ||||
							
								
								
									
										2
									
								
								utils.go
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								utils.go
									
									
									
									
									
								
							| @@ -55,5 +55,7 @@ func reformatJSON(json string) string { | ||||
| 	json = strings.Replace(json, "\"false\"", "false", -1) | ||||
| 	// Nextcloud encode quota as an empty array for never connected users | ||||
| 	json = strings.Replace(json, "\"quota\":[],", "", -1) | ||||
| 	// Nextcloud send admin unlimited quota as -3, others as "none" : replace with negative value | ||||
| 	json = strings.Replace(json, "\"quota\":\"none\"", "\"quota\":-3", -1) | ||||
| 	return json | ||||
| } | ||||
|   | ||||
							
								
								
									
										165
									
								
								vendor/github.com/sirupsen/logrus/CHANGELOG.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										165
									
								
								vendor/github.com/sirupsen/logrus/CHANGELOG.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,165 @@ | ||||
| # 1.2.0 | ||||
| This new release introduces: | ||||
|   * A new method `SetReportCaller` in the `Logger` to enable the file, line and calling function from which the trace has been issued | ||||
|   * A new trace level named `Trace` whose level is below `Debug` | ||||
|   * A configurable exit function to be called upon a Fatal trace | ||||
|   * The `Level` object now implements `encoding.TextUnmarshaler` interface | ||||
|  | ||||
| # 1.1.1 | ||||
| This is a bug fix release. | ||||
|   * fix the build break on Solaris | ||||
|   * don't drop a whole trace in JSONFormatter when a field param is a function pointer which can not be serialized | ||||
|  | ||||
| # 1.1.0 | ||||
| This new release introduces: | ||||
|   * several fixes: | ||||
|     * a fix for a race condition on entry formatting | ||||
|     * proper cleanup of previously used entries before putting them back in the pool | ||||
|     * the extra new line at the end of message in text formatter has been removed | ||||
|   * a new global public API to check if a level is activated: IsLevelEnabled | ||||
|   * the following methods have been added to the Logger object | ||||
|     * IsLevelEnabled | ||||
|     * SetFormatter | ||||
|     * SetOutput | ||||
|     * ReplaceHooks | ||||
|   * introduction of go module | ||||
|   * an indent configuration for the json formatter | ||||
|   * output colour support for windows | ||||
|   * the field sort function is now configurable for text formatter | ||||
|   * the CLICOLOR and CLICOLOR\_FORCE environment variable support in text formater | ||||
|  | ||||
| # 1.0.6 | ||||
|  | ||||
| This new release introduces: | ||||
|   * a new api WithTime which allows to easily force the time of the log entry | ||||
|     which is mostly useful for logger wrapper | ||||
|   * a fix reverting the immutability of the entry given as parameter to the hooks | ||||
|     a new configuration field of the json formatter in order to put all the fields | ||||
|     in a nested dictionnary | ||||
|   * a new SetOutput method in the Logger | ||||
|   * a new configuration of the textformatter to configure the name of the default keys | ||||
|   * a new configuration of the text formatter to disable the level truncation | ||||
|  | ||||
| # 1.0.5 | ||||
|  | ||||
| * Fix hooks race (#707) | ||||
| * Fix panic deadlock (#695) | ||||
|  | ||||
| # 1.0.4 | ||||
|  | ||||
| * Fix race when adding hooks (#612) | ||||
| * Fix terminal check in AppEngine (#635) | ||||
|  | ||||
| # 1.0.3 | ||||
|  | ||||
| * Replace example files with testable examples | ||||
|  | ||||
| # 1.0.2 | ||||
|  | ||||
| * bug: quote non-string values in text formatter (#583) | ||||
| * Make (*Logger) SetLevel a public method | ||||
|  | ||||
| # 1.0.1 | ||||
|  | ||||
| * bug: fix escaping in text formatter (#575) | ||||
|  | ||||
| # 1.0.0 | ||||
|  | ||||
| * Officially changed name to lower-case | ||||
| * bug: colors on Windows 10 (#541) | ||||
| * bug: fix race in accessing level (#512) | ||||
|  | ||||
| # 0.11.5 | ||||
|  | ||||
| * feature: add writer and writerlevel to entry (#372) | ||||
|  | ||||
| # 0.11.4 | ||||
|  | ||||
| * bug: fix undefined variable on solaris (#493) | ||||
|  | ||||
| # 0.11.3 | ||||
|  | ||||
| * formatter: configure quoting of empty values (#484) | ||||
| * formatter: configure quoting character (default is `"`) (#484) | ||||
| * bug: fix not importing io correctly in non-linux environments (#481) | ||||
|  | ||||
| # 0.11.2 | ||||
|  | ||||
| * bug: fix windows terminal detection (#476) | ||||
|  | ||||
| # 0.11.1 | ||||
|  | ||||
| * bug: fix tty detection with custom out (#471) | ||||
|  | ||||
| # 0.11.0 | ||||
|  | ||||
| * performance: Use bufferpool to allocate (#370) | ||||
| * terminal: terminal detection for app-engine (#343) | ||||
| * feature: exit handler (#375) | ||||
|  | ||||
| # 0.10.0 | ||||
|  | ||||
| * feature: Add a test hook (#180) | ||||
| * feature: `ParseLevel` is now case-insensitive (#326) | ||||
| * feature: `FieldLogger` interface that generalizes `Logger` and `Entry` (#308) | ||||
| * performance: avoid re-allocations on `WithFields` (#335) | ||||
|  | ||||
| # 0.9.0 | ||||
|  | ||||
| * logrus/text_formatter: don't emit empty msg | ||||
| * logrus/hooks/airbrake: move out of main repository | ||||
| * logrus/hooks/sentry: move out of main repository | ||||
| * logrus/hooks/papertrail: move out of main repository | ||||
| * logrus/hooks/bugsnag: move out of main repository | ||||
| * logrus/core: run tests with `-race` | ||||
| * logrus/core: detect TTY based on `stderr` | ||||
| * logrus/core: support `WithError` on logger | ||||
| * logrus/core: Solaris support | ||||
|  | ||||
| # 0.8.7 | ||||
|  | ||||
| * logrus/core: fix possible race (#216) | ||||
| * logrus/doc: small typo fixes and doc improvements | ||||
|  | ||||
|  | ||||
| # 0.8.6 | ||||
|  | ||||
| * hooks/raven: allow passing an initialized client | ||||
|  | ||||
| # 0.8.5 | ||||
|  | ||||
| * logrus/core: revert #208 | ||||
|  | ||||
| # 0.8.4 | ||||
|  | ||||
| * formatter/text: fix data race (#218) | ||||
|  | ||||
| # 0.8.3 | ||||
|  | ||||
| * logrus/core: fix entry log level (#208) | ||||
| * logrus/core: improve performance of text formatter by 40% | ||||
| * logrus/core: expose `LevelHooks` type | ||||
| * logrus/core: add support for DragonflyBSD and NetBSD | ||||
| * formatter/text: print structs more verbosely | ||||
|  | ||||
| # 0.8.2 | ||||
|  | ||||
| * logrus: fix more Fatal family functions | ||||
|  | ||||
| # 0.8.1 | ||||
|  | ||||
| * logrus: fix not exiting on `Fatalf` and `Fatalln` | ||||
|  | ||||
| # 0.8.0 | ||||
|  | ||||
| * logrus: defaults to stderr instead of stdout | ||||
| * hooks/sentry: add special field for `*http.Request` | ||||
| * formatter/text: ignore Windows for colors | ||||
|  | ||||
| # 0.7.3 | ||||
|  | ||||
| * formatter/\*: allow configuration of timestamp layout | ||||
|  | ||||
| # 0.7.2 | ||||
|  | ||||
| * formatter/text: Add configuration option for time format (#158) | ||||
							
								
								
									
										21
									
								
								vendor/github.com/sirupsen/logrus/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/github.com/sirupsen/logrus/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| The MIT License (MIT) | ||||
|  | ||||
| Copyright (c) 2014 Simon Eskildsen | ||||
|  | ||||
| Permission is hereby granted, free of charge, to any person obtaining a copy | ||||
| of this software and associated documentation files (the "Software"), to deal | ||||
| in the Software without restriction, including without limitation the rights | ||||
| to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||
| copies of the Software, and to permit persons to whom the Software is | ||||
| furnished to do so, subject to the following conditions: | ||||
|  | ||||
| The above copyright notice and this permission notice shall be included in | ||||
| all copies or substantial portions of the Software. | ||||
|  | ||||
| THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||
| FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||
| AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||
| LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||
| OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||
| THE SOFTWARE. | ||||
							
								
								
									
										494
									
								
								vendor/github.com/sirupsen/logrus/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										494
									
								
								vendor/github.com/sirupsen/logrus/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,494 @@ | ||||
| # Logrus <img src="http://i.imgur.com/hTeVwmJ.png" width="40" height="40" alt=":walrus:" class="emoji" title=":walrus:"/> [](https://travis-ci.org/sirupsen/logrus) [](https://godoc.org/github.com/sirupsen/logrus) | ||||
|  | ||||
| Logrus is a structured logger for Go (golang), completely API compatible with | ||||
| the standard library logger. | ||||
|  | ||||
| **Seeing weird case-sensitive problems?** It's in the past been possible to | ||||
| import Logrus as both upper- and lower-case. Due to the Go package environment, | ||||
| this caused issues in the community and we needed a standard. Some environments | ||||
| experienced problems with the upper-case variant, so the lower-case was decided. | ||||
| Everything using `logrus` will need to use the lower-case: | ||||
| `github.com/sirupsen/logrus`. Any package that isn't, should be changed. | ||||
|  | ||||
| To fix Glide, see [these | ||||
| comments](https://github.com/sirupsen/logrus/issues/553#issuecomment-306591437). | ||||
| For an in-depth explanation of the casing issue, see [this | ||||
| comment](https://github.com/sirupsen/logrus/issues/570#issuecomment-313933276). | ||||
|  | ||||
| **Are you interested in assisting in maintaining Logrus?** Currently I have a | ||||
| lot of obligations, and I am unable to provide Logrus with the maintainership it | ||||
| needs. If you'd like to help, please reach out to me at `simon at author's | ||||
| username dot com`. | ||||
|  | ||||
| Nicely color-coded in development (when a TTY is attached, otherwise just | ||||
| plain text): | ||||
|  | ||||
|  | ||||
|  | ||||
| With `log.SetFormatter(&log.JSONFormatter{})`, for easy parsing by logstash | ||||
| or Splunk: | ||||
|  | ||||
| ```json | ||||
| {"animal":"walrus","level":"info","msg":"A group of walrus emerges from the | ||||
| ocean","size":10,"time":"2014-03-10 19:57:38.562264131 -0400 EDT"} | ||||
|  | ||||
| {"level":"warning","msg":"The group's number increased tremendously!", | ||||
| "number":122,"omg":true,"time":"2014-03-10 19:57:38.562471297 -0400 EDT"} | ||||
|  | ||||
| {"animal":"walrus","level":"info","msg":"A giant walrus appears!", | ||||
| "size":10,"time":"2014-03-10 19:57:38.562500591 -0400 EDT"} | ||||
|  | ||||
| {"animal":"walrus","level":"info","msg":"Tremendously sized cow enters the ocean.", | ||||
| "size":9,"time":"2014-03-10 19:57:38.562527896 -0400 EDT"} | ||||
|  | ||||
| {"level":"fatal","msg":"The ice breaks!","number":100,"omg":true, | ||||
| "time":"2014-03-10 19:57:38.562543128 -0400 EDT"} | ||||
| ``` | ||||
|  | ||||
| With the default `log.SetFormatter(&log.TextFormatter{})` when a TTY is not | ||||
| attached, the output is compatible with the | ||||
| [logfmt](http://godoc.org/github.com/kr/logfmt) format: | ||||
|  | ||||
| ```text | ||||
| time="2015-03-26T01:27:38-04:00" level=debug msg="Started observing beach" animal=walrus number=8 | ||||
| time="2015-03-26T01:27:38-04:00" level=info msg="A group of walrus emerges from the ocean" animal=walrus size=10 | ||||
| time="2015-03-26T01:27:38-04:00" level=warning msg="The group's number increased tremendously!" number=122 omg=true | ||||
| time="2015-03-26T01:27:38-04:00" level=debug msg="Temperature changes" temperature=-4 | ||||
| time="2015-03-26T01:27:38-04:00" level=panic msg="It's over 9000!" animal=orca size=9009 | ||||
| time="2015-03-26T01:27:38-04:00" level=fatal msg="The ice breaks!" err=&{0x2082280c0 map[animal:orca size:9009] 2015-03-26 01:27:38.441574009 -0400 EDT panic It's over 9000!} number=100 omg=true | ||||
| ``` | ||||
| To ensure this behaviour even if a TTY is attached, set your formatter as follows: | ||||
|  | ||||
| ```go | ||||
| 	log.SetFormatter(&log.TextFormatter{ | ||||
| 		DisableColors: true, | ||||
| 		FullTimestamp: true, | ||||
| 	}) | ||||
| ``` | ||||
|  | ||||
| #### Logging Method Name | ||||
|  | ||||
| If you wish to add the calling method as a field, instruct the logger via: | ||||
| ```go | ||||
| log.SetReportCaller(true) | ||||
| ``` | ||||
| This adds the caller as 'method' like so: | ||||
|  | ||||
| ```json | ||||
| {"animal":"penguin","level":"fatal","method":"github.com/sirupsen/arcticcreatures.migrate","msg":"a penguin swims by", | ||||
| "time":"2014-03-10 19:57:38.562543129 -0400 EDT"} | ||||
| ``` | ||||
|  | ||||
| ```text | ||||
| time="2015-03-26T01:27:38-04:00" level=fatal method=github.com/sirupsen/arcticcreatures.migrate msg="a penguin swims by" animal=penguin | ||||
| ``` | ||||
| Note that this does add measurable overhead - the cost will depend on the version of Go, but is | ||||
| between 20 and 40% in recent tests with 1.6 and 1.7.  You can validate this in your | ||||
| environment via benchmarks:  | ||||
| ``` | ||||
| go test -bench=.*CallerTracing | ||||
| ``` | ||||
|  | ||||
|  | ||||
| #### Case-sensitivity | ||||
|  | ||||
| The organization's name was changed to lower-case--and this will not be changed | ||||
| back. If you are getting import conflicts due to case sensitivity, please use | ||||
| the lower-case import: `github.com/sirupsen/logrus`. | ||||
|  | ||||
| #### Example | ||||
|  | ||||
| The simplest way to use Logrus is simply the package-level exported logger: | ||||
|  | ||||
| ```go | ||||
| package main | ||||
|  | ||||
| import ( | ||||
|   log "github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
|   log.WithFields(log.Fields{ | ||||
|     "animal": "walrus", | ||||
|   }).Info("A walrus appears") | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Note that it's completely api-compatible with the stdlib logger, so you can | ||||
| replace your `log` imports everywhere with `log "github.com/sirupsen/logrus"` | ||||
| and you'll now have the flexibility of Logrus. You can customize it all you | ||||
| want: | ||||
|  | ||||
| ```go | ||||
| package main | ||||
|  | ||||
| import ( | ||||
|   "os" | ||||
|   log "github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
|   // Log as JSON instead of the default ASCII formatter. | ||||
|   log.SetFormatter(&log.JSONFormatter{}) | ||||
|  | ||||
|   // Output to stdout instead of the default stderr | ||||
|   // Can be any io.Writer, see below for File example | ||||
|   log.SetOutput(os.Stdout) | ||||
|  | ||||
|   // Only log the warning severity or above. | ||||
|   log.SetLevel(log.WarnLevel) | ||||
| } | ||||
|  | ||||
| func main() { | ||||
|   log.WithFields(log.Fields{ | ||||
|     "animal": "walrus", | ||||
|     "size":   10, | ||||
|   }).Info("A group of walrus emerges from the ocean") | ||||
|  | ||||
|   log.WithFields(log.Fields{ | ||||
|     "omg":    true, | ||||
|     "number": 122, | ||||
|   }).Warn("The group's number increased tremendously!") | ||||
|  | ||||
|   log.WithFields(log.Fields{ | ||||
|     "omg":    true, | ||||
|     "number": 100, | ||||
|   }).Fatal("The ice breaks!") | ||||
|  | ||||
|   // A common pattern is to re-use fields between logging statements by re-using | ||||
|   // the logrus.Entry returned from WithFields() | ||||
|   contextLogger := log.WithFields(log.Fields{ | ||||
|     "common": "this is a common field", | ||||
|     "other": "I also should be logged always", | ||||
|   }) | ||||
|  | ||||
|   contextLogger.Info("I'll be logged with common and other field") | ||||
|   contextLogger.Info("Me too") | ||||
| } | ||||
| ``` | ||||
|  | ||||
| For more advanced usage such as logging to multiple locations from the same | ||||
| application, you can also create an instance of the `logrus` Logger: | ||||
|  | ||||
| ```go | ||||
| package main | ||||
|  | ||||
| import ( | ||||
|   "os" | ||||
|   "github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| // Create a new instance of the logger. You can have any number of instances. | ||||
| var log = logrus.New() | ||||
|  | ||||
| func main() { | ||||
|   // The API for setting attributes is a little different than the package level | ||||
|   // exported logger. See Godoc. | ||||
|   log.Out = os.Stdout | ||||
|  | ||||
|   // You could set this to any `io.Writer` such as a file | ||||
|   // file, err := os.OpenFile("logrus.log", os.O_CREATE|os.O_WRONLY, 0666) | ||||
|   // if err == nil { | ||||
|   //  log.Out = file | ||||
|   // } else { | ||||
|   //  log.Info("Failed to log to file, using default stderr") | ||||
|   // } | ||||
|  | ||||
|   log.WithFields(logrus.Fields{ | ||||
|     "animal": "walrus", | ||||
|     "size":   10, | ||||
|   }).Info("A group of walrus emerges from the ocean") | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### Fields | ||||
|  | ||||
| Logrus encourages careful, structured logging through logging fields instead of | ||||
| long, unparseable error messages. For example, instead of: `log.Fatalf("Failed | ||||
| to send event %s to topic %s with key %d")`, you should log the much more | ||||
| discoverable: | ||||
|  | ||||
| ```go | ||||
| log.WithFields(log.Fields{ | ||||
|   "event": event, | ||||
|   "topic": topic, | ||||
|   "key": key, | ||||
| }).Fatal("Failed to send event") | ||||
| ``` | ||||
|  | ||||
| We've found this API forces you to think about logging in a way that produces | ||||
| much more useful logging messages. We've been in countless situations where just | ||||
| a single added field to a log statement that was already there would've saved us | ||||
| hours. The `WithFields` call is optional. | ||||
|  | ||||
| In general, with Logrus using any of the `printf`-family functions should be | ||||
| seen as a hint you should add a field, however, you can still use the | ||||
| `printf`-family functions with Logrus. | ||||
|  | ||||
| #### Default Fields | ||||
|  | ||||
| Often it's helpful to have fields _always_ attached to log statements in an | ||||
| application or parts of one. For example, you may want to always log the | ||||
| `request_id` and `user_ip` in the context of a request. Instead of writing | ||||
| `log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip})` on | ||||
| every line, you can create a `logrus.Entry` to pass around instead: | ||||
|  | ||||
| ```go | ||||
| requestLogger := log.WithFields(log.Fields{"request_id": request_id, "user_ip": user_ip}) | ||||
| requestLogger.Info("something happened on that request") # will log request_id and user_ip | ||||
| requestLogger.Warn("something not great happened") | ||||
| ``` | ||||
|  | ||||
| #### Hooks | ||||
|  | ||||
| You can add hooks for logging levels. For example to send errors to an exception | ||||
| tracking service on `Error`, `Fatal` and `Panic`, info to StatsD or log to | ||||
| multiple places simultaneously, e.g. syslog. | ||||
|  | ||||
| Logrus comes with [built-in hooks](hooks/). Add those, or your custom hook, in | ||||
| `init`: | ||||
|  | ||||
| ```go | ||||
| import ( | ||||
|   log "github.com/sirupsen/logrus" | ||||
|   "gopkg.in/gemnasium/logrus-airbrake-hook.v2" // the package is named "airbrake" | ||||
|   logrus_syslog "github.com/sirupsen/logrus/hooks/syslog" | ||||
|   "log/syslog" | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
|  | ||||
|   // Use the Airbrake hook to report errors that have Error severity or above to | ||||
|   // an exception tracker. You can create custom hooks, see the Hooks section. | ||||
|   log.AddHook(airbrake.NewHook(123, "xyz", "production")) | ||||
|  | ||||
|   hook, err := logrus_syslog.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") | ||||
|   if err != nil { | ||||
|     log.Error("Unable to connect to local syslog daemon") | ||||
|   } else { | ||||
|     log.AddHook(hook) | ||||
|   } | ||||
| } | ||||
| ``` | ||||
| Note: Syslog hook also support connecting to local syslog (Ex. "/dev/log" or "/var/run/syslog" or "/var/run/log"). For the detail, please check the [syslog hook README](hooks/syslog/README.md). | ||||
|  | ||||
| A list of currently known of service hook can be found in this wiki [page](https://github.com/sirupsen/logrus/wiki/Hooks) | ||||
|  | ||||
|  | ||||
| #### Level logging | ||||
|  | ||||
| Logrus has seven logging levels: Trace, Debug, Info, Warning, Error, Fatal and Panic. | ||||
|  | ||||
| ```go | ||||
| log.Trace("Something very low level.") | ||||
| log.Debug("Useful debugging information.") | ||||
| log.Info("Something noteworthy happened!") | ||||
| log.Warn("You should probably take a look at this.") | ||||
| log.Error("Something failed but I'm not quitting.") | ||||
| // Calls os.Exit(1) after logging | ||||
| log.Fatal("Bye.") | ||||
| // Calls panic() after logging | ||||
| log.Panic("I'm bailing.") | ||||
| ``` | ||||
|  | ||||
| You can set the logging level on a `Logger`, then it will only log entries with | ||||
| that severity or anything above it: | ||||
|  | ||||
| ```go | ||||
| // Will log anything that is info or above (warn, error, fatal, panic). Default. | ||||
| log.SetLevel(log.InfoLevel) | ||||
| ``` | ||||
|  | ||||
| It may be useful to set `log.Level = logrus.DebugLevel` in a debug or verbose | ||||
| environment if your application has that. | ||||
|  | ||||
| #### Entries | ||||
|  | ||||
| Besides the fields added with `WithField` or `WithFields` some fields are | ||||
| automatically added to all logging events: | ||||
|  | ||||
| 1. `time`. The timestamp when the entry was created. | ||||
| 2. `msg`. The logging message passed to `{Info,Warn,Error,Fatal,Panic}` after | ||||
|    the `AddFields` call. E.g. `Failed to send event.` | ||||
| 3. `level`. The logging level. E.g. `info`. | ||||
|  | ||||
| #### Environments | ||||
|  | ||||
| Logrus has no notion of environment. | ||||
|  | ||||
| If you wish for hooks and formatters to only be used in specific environments, | ||||
| you should handle that yourself. For example, if your application has a global | ||||
| variable `Environment`, which is a string representation of the environment you | ||||
| could do: | ||||
|  | ||||
| ```go | ||||
| import ( | ||||
|   log "github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| init() { | ||||
|   // do something here to set environment depending on an environment variable | ||||
|   // or command-line flag | ||||
|   if Environment == "production" { | ||||
|     log.SetFormatter(&log.JSONFormatter{}) | ||||
|   } else { | ||||
|     // The TextFormatter is default, you don't actually have to do this. | ||||
|     log.SetFormatter(&log.TextFormatter{}) | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| This configuration is how `logrus` was intended to be used, but JSON in | ||||
| production is mostly only useful if you do log aggregation with tools like | ||||
| Splunk or Logstash. | ||||
|  | ||||
| #### Formatters | ||||
|  | ||||
| The built-in logging formatters are: | ||||
|  | ||||
| * `logrus.TextFormatter`. Logs the event in colors if stdout is a tty, otherwise | ||||
|   without colors. | ||||
|   * *Note:* to force colored output when there is no TTY, set the `ForceColors` | ||||
|     field to `true`.  To force no colored output even if there is a TTY  set the | ||||
|     `DisableColors` field to `true`. For Windows, see | ||||
|     [github.com/mattn/go-colorable](https://github.com/mattn/go-colorable). | ||||
|   * When colors are enabled, levels are truncated to 4 characters by default. To disable | ||||
|     truncation set the `DisableLevelTruncation` field to `true`. | ||||
|   * All options are listed in the [generated docs](https://godoc.org/github.com/sirupsen/logrus#TextFormatter). | ||||
| * `logrus.JSONFormatter`. Logs fields as JSON. | ||||
|   * All options are listed in the [generated docs](https://godoc.org/github.com/sirupsen/logrus#JSONFormatter). | ||||
|  | ||||
| Third party logging formatters: | ||||
|  | ||||
| * [`FluentdFormatter`](https://github.com/joonix/log). Formats entries that can be parsed by Kubernetes and Google Container Engine. | ||||
| * [`GELF`](https://github.com/fabienm/go-logrus-formatters). Formats entries so they comply to Graylog's [GELF 1.1 specification](http://docs.graylog.org/en/2.4/pages/gelf.html). | ||||
| * [`logstash`](https://github.com/bshuster-repo/logrus-logstash-hook). Logs fields as [Logstash](http://logstash.net) Events. | ||||
| * [`prefixed`](https://github.com/x-cray/logrus-prefixed-formatter). Displays log entry source along with alternative layout. | ||||
| * [`zalgo`](https://github.com/aybabtme/logzalgo). Invoking the P͉̫o̳̼̊w̖͈̰͎e̬͔̭͂r͚̼̹̲ ̫͓͉̳͈ō̠͕͖̚f̝͍̠ ͕̲̞͖͑Z̖̫̤̫ͪa͉̬͈̗l͖͎g̳̥o̰̥̅!̣͔̲̻͊̄ ̙̘̦̹̦. | ||||
|  | ||||
| You can define your formatter by implementing the `Formatter` interface, | ||||
| requiring a `Format` method. `Format` takes an `*Entry`. `entry.Data` is a | ||||
| `Fields` type (`map[string]interface{}`) with all your fields as well as the | ||||
| default ones (see Entries section above): | ||||
|  | ||||
| ```go | ||||
| type MyJSONFormatter struct { | ||||
| } | ||||
|  | ||||
| log.SetFormatter(new(MyJSONFormatter)) | ||||
|  | ||||
| func (f *MyJSONFormatter) Format(entry *Entry) ([]byte, error) { | ||||
|   // Note this doesn't include Time, Level and Message which are available on | ||||
|   // the Entry. Consult `godoc` on information about those fields or read the | ||||
|   // source of the official loggers. | ||||
|   serialized, err := json.Marshal(entry.Data) | ||||
|     if err != nil { | ||||
|       return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) | ||||
|     } | ||||
|   return append(serialized, '\n'), nil | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### Logger as an `io.Writer` | ||||
|  | ||||
| Logrus can be transformed into an `io.Writer`. That writer is the end of an `io.Pipe` and it is your responsibility to close it. | ||||
|  | ||||
| ```go | ||||
| w := logger.Writer() | ||||
| defer w.Close() | ||||
|  | ||||
| srv := http.Server{ | ||||
|     // create a stdlib log.Logger that writes to | ||||
|     // logrus.Logger. | ||||
|     ErrorLog: log.New(w, "", 0), | ||||
| } | ||||
| ``` | ||||
|  | ||||
| Each line written to that writer will be printed the usual way, using formatters | ||||
| and hooks. The level for those entries is `info`. | ||||
|  | ||||
| This means that we can override the standard library logger easily: | ||||
|  | ||||
| ```go | ||||
| logger := logrus.New() | ||||
| logger.Formatter = &logrus.JSONFormatter{} | ||||
|  | ||||
| // Use logrus for standard log output | ||||
| // Note that `log` here references stdlib's log | ||||
| // Not logrus imported under the name `log`. | ||||
| log.SetOutput(logger.Writer()) | ||||
| ``` | ||||
|  | ||||
| #### Rotation | ||||
|  | ||||
| Log rotation is not provided with Logrus. Log rotation should be done by an | ||||
| external program (like `logrotate(8)`) that can compress and delete old log | ||||
| entries. It should not be a feature of the application-level logger. | ||||
|  | ||||
| #### Tools | ||||
|  | ||||
| | Tool | Description | | ||||
| | ---- | ----------- | | ||||
| |[Logrus Mate](https://github.com/gogap/logrus_mate)|Logrus mate is a tool for Logrus to manage loggers, you can initial logger's level, hook and formatter by config file, the logger will generated with different config at different environment.| | ||||
| |[Logrus Viper Helper](https://github.com/heirko/go-contrib/tree/master/logrusHelper)|An Helper around Logrus to wrap with spf13/Viper to load configuration with fangs! And to simplify Logrus configuration use some behavior of [Logrus Mate](https://github.com/gogap/logrus_mate). [sample](https://github.com/heirko/iris-contrib/blob/master/middleware/logrus-logger/example) | | ||||
|  | ||||
| #### Testing | ||||
|  | ||||
| Logrus has a built in facility for asserting the presence of log messages. This is implemented through the `test` hook and provides: | ||||
|  | ||||
| * decorators for existing logger (`test.NewLocal` and `test.NewGlobal`) which basically just add the `test` hook | ||||
| * a test logger (`test.NewNullLogger`) that just records log messages (and does not output any): | ||||
|  | ||||
| ```go | ||||
| import( | ||||
|   "github.com/sirupsen/logrus" | ||||
|   "github.com/sirupsen/logrus/hooks/test" | ||||
|   "github.com/stretchr/testify/assert" | ||||
|   "testing" | ||||
| ) | ||||
|  | ||||
| func TestSomething(t*testing.T){ | ||||
|   logger, hook := test.NewNullLogger() | ||||
|   logger.Error("Helloerror") | ||||
|  | ||||
|   assert.Equal(t, 1, len(hook.Entries)) | ||||
|   assert.Equal(t, logrus.ErrorLevel, hook.LastEntry().Level) | ||||
|   assert.Equal(t, "Helloerror", hook.LastEntry().Message) | ||||
|  | ||||
|   hook.Reset() | ||||
|   assert.Nil(t, hook.LastEntry()) | ||||
| } | ||||
| ``` | ||||
|  | ||||
| #### Fatal handlers | ||||
|  | ||||
| Logrus can register one or more functions that will be called when any `fatal` | ||||
| level message is logged. The registered handlers will be executed before | ||||
| logrus performs a `os.Exit(1)`. This behavior may be helpful if callers need | ||||
| to gracefully shutdown. Unlike a `panic("Something went wrong...")` call which can be intercepted with a deferred `recover` a call to `os.Exit(1)` can not be intercepted. | ||||
|  | ||||
| ``` | ||||
| ... | ||||
| handler := func() { | ||||
|   // gracefully shutdown something... | ||||
| } | ||||
| logrus.RegisterExitHandler(handler) | ||||
| ... | ||||
| ``` | ||||
|  | ||||
| #### Thread safety | ||||
|  | ||||
| By default, Logger is protected by a mutex for concurrent writes. The mutex is held when calling hooks and writing logs. | ||||
| If you are sure such locking is not needed, you can call logger.SetNoLock() to disable the locking. | ||||
|  | ||||
| Situation when locking is not needed includes: | ||||
|  | ||||
| * You have no hooks registered, or hooks calling is already thread-safe. | ||||
|  | ||||
| * Writing to logger.Out is already thread-safe, for example: | ||||
|  | ||||
|   1) logger.Out is protected by locks. | ||||
|  | ||||
|   2) logger.Out is a os.File handler opened with `O_APPEND` flag, and every write is smaller than 4k. (This allow multi-thread/multi-process writing) | ||||
|  | ||||
|      (Refer to http://www.notthewizard.com/2014/06/17/are-files-appends-really-atomic/) | ||||
							
								
								
									
										64
									
								
								vendor/github.com/sirupsen/logrus/alt_exit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								vendor/github.com/sirupsen/logrus/alt_exit.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| package logrus | ||||
|  | ||||
| // The following code was sourced and modified from the | ||||
| // https://github.com/tebeka/atexit package governed by the following license: | ||||
| // | ||||
| // Copyright (c) 2012 Miki Tebeka <miki.tebeka@gmail.com>. | ||||
| // | ||||
| // Permission is hereby granted, free of charge, to any person obtaining a copy of | ||||
| // this software and associated documentation files (the "Software"), to deal in | ||||
| // the Software without restriction, including without limitation the rights to | ||||
| // use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of | ||||
| // the Software, and to permit persons to whom the Software is furnished to do so, | ||||
| // subject to the following conditions: | ||||
| // | ||||
| // The above copyright notice and this permission notice shall be included in all | ||||
| // copies or substantial portions of the Software. | ||||
| // | ||||
| // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||
| // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS | ||||
| // FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR | ||||
| // COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER | ||||
| // IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN | ||||
| // CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| ) | ||||
|  | ||||
| var handlers = []func(){} | ||||
|  | ||||
| func runHandler(handler func()) { | ||||
| 	defer func() { | ||||
| 		if err := recover(); err != nil { | ||||
| 			fmt.Fprintln(os.Stderr, "Error: Logrus exit handler error:", err) | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	handler() | ||||
| } | ||||
|  | ||||
| func runHandlers() { | ||||
| 	for _, handler := range handlers { | ||||
| 		runHandler(handler) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Exit runs all the Logrus atexit handlers and then terminates the program using os.Exit(code) | ||||
| func Exit(code int) { | ||||
| 	runHandlers() | ||||
| 	os.Exit(code) | ||||
| } | ||||
|  | ||||
| // RegisterExitHandler adds a Logrus Exit handler, call logrus.Exit to invoke | ||||
| // all handlers. The handlers will also be invoked when any Fatal log entry is | ||||
| // made. | ||||
| // | ||||
| // This method is useful when a caller wishes to use logrus to log a fatal | ||||
| // message but also needs to gracefully shutdown. An example usecase could be | ||||
| // closing database connections, or sending a alert that the application is | ||||
| // closing. | ||||
| func RegisterExitHandler(handler func()) { | ||||
| 	handlers = append(handlers, handler) | ||||
| } | ||||
							
								
								
									
										99
									
								
								vendor/github.com/sirupsen/logrus/alt_exit_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								vendor/github.com/sirupsen/logrus/alt_exit_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io/ioutil" | ||||
| 	"log" | ||||
| 	"os" | ||||
| 	"os/exec" | ||||
| 	"path/filepath" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func TestRegister(t *testing.T) { | ||||
| 	current := len(handlers) | ||||
| 	RegisterExitHandler(func() {}) | ||||
| 	if len(handlers) != current+1 { | ||||
| 		t.Fatalf("expected %d handlers, got %d", current+1, len(handlers)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestHandler(t *testing.T) { | ||||
| 	testprog := testprogleader | ||||
| 	testprog = append(testprog, getPackage()...) | ||||
| 	testprog = append(testprog, testprogtrailer...) | ||||
| 	tempDir, err := ioutil.TempDir("", "test_handler") | ||||
| 	if err != nil { | ||||
| 		log.Fatalf("can't create temp dir. %q", err) | ||||
| 	} | ||||
| 	defer os.RemoveAll(tempDir) | ||||
|  | ||||
| 	gofile := filepath.Join(tempDir, "gofile.go") | ||||
| 	if err := ioutil.WriteFile(gofile, testprog, 0666); err != nil { | ||||
| 		t.Fatalf("can't create go file. %q", err) | ||||
| 	} | ||||
|  | ||||
| 	outfile := filepath.Join(tempDir, "outfile.out") | ||||
| 	arg := time.Now().UTC().String() | ||||
| 	err = exec.Command("go", "run", gofile, outfile, arg).Run() | ||||
| 	if err == nil { | ||||
| 		t.Fatalf("completed normally, should have failed") | ||||
| 	} | ||||
|  | ||||
| 	data, err := ioutil.ReadFile(outfile) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("can't read output file %s. %q", outfile, err) | ||||
| 	} | ||||
|  | ||||
| 	if string(data) != arg { | ||||
| 		t.Fatalf("bad data. Expected %q, got %q", data, arg) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // getPackage returns the name of the current package, which makes running this | ||||
| // test in a fork simpler | ||||
| func getPackage() []byte { | ||||
| 	pc, _, _, _ := runtime.Caller(0) | ||||
| 	fullFuncName := runtime.FuncForPC(pc).Name() | ||||
| 	idx := strings.LastIndex(fullFuncName, ".") | ||||
| 	return []byte(fullFuncName[:idx]) // trim off function details | ||||
| } | ||||
|  | ||||
| var testprogleader = []byte(` | ||||
| // Test program for atexit, gets output file and data as arguments and writes | ||||
| // data to output file in atexit handler. | ||||
| package main | ||||
|  | ||||
| import ( | ||||
| 	"`) | ||||
| var testprogtrailer = []byte( | ||||
| 	`" | ||||
| 	"flag" | ||||
| 	"fmt" | ||||
| 	"io/ioutil" | ||||
| ) | ||||
|  | ||||
| var outfile = "" | ||||
| var data = "" | ||||
|  | ||||
| func handler() { | ||||
| 	ioutil.WriteFile(outfile, []byte(data), 0666) | ||||
| } | ||||
|  | ||||
| func badHandler() { | ||||
| 	n := 0 | ||||
| 	fmt.Println(1/n) | ||||
| } | ||||
|  | ||||
| func main() { | ||||
| 	flag.Parse() | ||||
| 	outfile = flag.Arg(0) | ||||
| 	data = flag.Arg(1) | ||||
|  | ||||
| 	logrus.RegisterExitHandler(handler) | ||||
| 	logrus.RegisterExitHandler(badHandler) | ||||
| 	logrus.Fatal("Bye bye") | ||||
| } | ||||
| `) | ||||
							
								
								
									
										14
									
								
								vendor/github.com/sirupsen/logrus/appveyor.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								vendor/github.com/sirupsen/logrus/appveyor.yml
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| version: "{build}" | ||||
| platform: x64 | ||||
| clone_folder: c:\gopath\src\github.com\sirupsen\logrus | ||||
| environment:   | ||||
|   GOPATH: c:\gopath | ||||
| branches:   | ||||
|   only: | ||||
|     - master | ||||
| install:   | ||||
|   - set PATH=%GOPATH%\bin;c:\go\bin;%PATH% | ||||
|   - go version | ||||
| build_script:   | ||||
|   - go get -t | ||||
|   - go test | ||||
							
								
								
									
										26
									
								
								vendor/github.com/sirupsen/logrus/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								vendor/github.com/sirupsen/logrus/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| /* | ||||
| Package logrus is a structured logger for Go, completely API compatible with the standard library logger. | ||||
|  | ||||
|  | ||||
| The simplest way to use Logrus is simply the package-level exported logger: | ||||
|  | ||||
|   package main | ||||
|  | ||||
|   import ( | ||||
|     log "github.com/sirupsen/logrus" | ||||
|   ) | ||||
|  | ||||
|   func main() { | ||||
|     log.WithFields(log.Fields{ | ||||
|       "animal": "walrus", | ||||
|       "number": 1, | ||||
|       "size":   10, | ||||
|     }).Info("A walrus appears") | ||||
|   } | ||||
|  | ||||
| Output: | ||||
|   time="2015-09-07T08:48:33Z" level=info msg="A walrus appears" animal=walrus number=1 size=10 | ||||
|  | ||||
| For a full guide visit https://github.com/sirupsen/logrus | ||||
| */ | ||||
| package logrus | ||||
							
								
								
									
										419
									
								
								vendor/github.com/sirupsen/logrus/entry.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										419
									
								
								vendor/github.com/sirupsen/logrus/entry.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,419 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"reflect" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	bufferPool *sync.Pool | ||||
|  | ||||
| 	// qualified package name, cached at first use | ||||
| 	logrusPackage string | ||||
|  | ||||
| 	// Positions in the call stack when tracing to report the calling method | ||||
| 	minimumCallerDepth int | ||||
|  | ||||
| 	// Used for caller information initialisation | ||||
| 	callerInitOnce sync.Once | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	maximumCallerDepth int = 25 | ||||
| 	knownLogrusFrames  int = 4 | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
| 	bufferPool = &sync.Pool{ | ||||
| 		New: func() interface{} { | ||||
| 			return new(bytes.Buffer) | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	// start at the bottom of the stack before the package-name cache is primed | ||||
| 	minimumCallerDepth = 1 | ||||
| } | ||||
|  | ||||
| // Defines the key when adding errors using WithError. | ||||
| var ErrorKey = "error" | ||||
|  | ||||
| // An entry is the final or intermediate Logrus logging entry. It contains all | ||||
| // the fields passed with WithField{,s}. It's finally logged when Trace, Debug, | ||||
| // Info, Warn, Error, Fatal or Panic is called on it. These objects can be | ||||
| // reused and passed around as much as you wish to avoid field duplication. | ||||
| type Entry struct { | ||||
| 	Logger *Logger | ||||
|  | ||||
| 	// Contains all the fields set by the user. | ||||
| 	Data Fields | ||||
|  | ||||
| 	// Time at which the log entry was created | ||||
| 	Time time.Time | ||||
|  | ||||
| 	// Level the log entry was logged at: Trace, Debug, Info, Warn, Error, Fatal or Panic | ||||
| 	// This field will be set on entry firing and the value will be equal to the one in Logger struct field. | ||||
| 	Level Level | ||||
|  | ||||
| 	// Calling method, with package name | ||||
| 	Caller *runtime.Frame | ||||
|  | ||||
| 	// Message passed to Trace, Debug, Info, Warn, Error, Fatal or Panic | ||||
| 	Message string | ||||
|  | ||||
| 	// When formatter is called in entry.log(), a Buffer may be set to entry | ||||
| 	Buffer *bytes.Buffer | ||||
|  | ||||
| 	// err may contain a field formatting error | ||||
| 	err string | ||||
| } | ||||
|  | ||||
| func NewEntry(logger *Logger) *Entry { | ||||
| 	return &Entry{ | ||||
| 		Logger: logger, | ||||
| 		// Default is three fields, plus one optional.  Give a little extra room. | ||||
| 		Data: make(Fields, 6), | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Returns the string representation from the reader and ultimately the | ||||
| // formatter. | ||||
| func (entry *Entry) String() (string, error) { | ||||
| 	serialized, err := entry.Logger.Formatter.Format(entry) | ||||
| 	if err != nil { | ||||
| 		return "", err | ||||
| 	} | ||||
| 	str := string(serialized) | ||||
| 	return str, nil | ||||
| } | ||||
|  | ||||
| // Add an error as single field (using the key defined in ErrorKey) to the Entry. | ||||
| func (entry *Entry) WithError(err error) *Entry { | ||||
| 	return entry.WithField(ErrorKey, err) | ||||
| } | ||||
|  | ||||
| // Add a single field to the Entry. | ||||
| func (entry *Entry) WithField(key string, value interface{}) *Entry { | ||||
| 	return entry.WithFields(Fields{key: value}) | ||||
| } | ||||
|  | ||||
| // Add a map of fields to the Entry. | ||||
| func (entry *Entry) WithFields(fields Fields) *Entry { | ||||
| 	data := make(Fields, len(entry.Data)+len(fields)) | ||||
| 	for k, v := range entry.Data { | ||||
| 		data[k] = v | ||||
| 	} | ||||
| 	fieldErr := entry.err | ||||
| 	for k, v := range fields { | ||||
| 		isErrField := false | ||||
| 		if t := reflect.TypeOf(v); t != nil { | ||||
| 			switch t.Kind() { | ||||
| 			case reflect.Func: | ||||
| 				isErrField = true | ||||
| 			case reflect.Ptr: | ||||
| 				isErrField = t.Elem().Kind() == reflect.Func | ||||
| 			} | ||||
| 		} | ||||
| 		if isErrField { | ||||
| 			tmp := fmt.Sprintf("can not add field %q", k) | ||||
| 			if fieldErr != "" { | ||||
| 				fieldErr = entry.err + ", " + tmp | ||||
| 			} else { | ||||
| 				fieldErr = tmp | ||||
| 			} | ||||
| 		} else { | ||||
| 			data[k] = v | ||||
| 		} | ||||
| 	} | ||||
| 	return &Entry{Logger: entry.Logger, Data: data, Time: entry.Time, err: fieldErr} | ||||
| } | ||||
|  | ||||
| // Overrides the time of the Entry. | ||||
| func (entry *Entry) WithTime(t time.Time) *Entry { | ||||
| 	return &Entry{Logger: entry.Logger, Data: entry.Data, Time: t, err: entry.err} | ||||
| } | ||||
|  | ||||
| // getPackageName reduces a fully qualified function name to the package name | ||||
| // There really ought to be to be a better way... | ||||
| func getPackageName(f string) string { | ||||
| 	for { | ||||
| 		lastPeriod := strings.LastIndex(f, ".") | ||||
| 		lastSlash := strings.LastIndex(f, "/") | ||||
| 		if lastPeriod > lastSlash { | ||||
| 			f = f[:lastPeriod] | ||||
| 		} else { | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return f | ||||
| } | ||||
|  | ||||
| // getCaller retrieves the name of the first non-logrus calling function | ||||
| func getCaller() *runtime.Frame { | ||||
| 	// Restrict the lookback frames to avoid runaway lookups | ||||
| 	pcs := make([]uintptr, maximumCallerDepth) | ||||
| 	depth := runtime.Callers(minimumCallerDepth, pcs) | ||||
| 	frames := runtime.CallersFrames(pcs[:depth]) | ||||
|  | ||||
| 	// cache this package's fully-qualified name | ||||
| 	callerInitOnce.Do(func() { | ||||
| 		logrusPackage = getPackageName(runtime.FuncForPC(pcs[0]).Name()) | ||||
|  | ||||
| 		// now that we have the cache, we can skip a minimum count of known-logrus functions | ||||
| 		// XXX this is dubious, the number of frames may vary store an entry in a logger interface | ||||
| 		minimumCallerDepth = knownLogrusFrames | ||||
| 	}) | ||||
|  | ||||
| 	for f, again := frames.Next(); again; f, again = frames.Next() { | ||||
| 		pkg := getPackageName(f.Function) | ||||
|  | ||||
| 		// If the caller isn't part of this package, we're done | ||||
| 		if pkg != logrusPackage { | ||||
| 			return &f | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// if we got here, we failed to find the caller's context | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (entry Entry) HasCaller() (has bool) { | ||||
| 	return entry.Logger != nil && | ||||
| 		entry.Logger.ReportCaller && | ||||
| 		entry.Caller != nil | ||||
| } | ||||
|  | ||||
| // This function is not declared with a pointer value because otherwise | ||||
| // race conditions will occur when using multiple goroutines | ||||
| func (entry Entry) log(level Level, msg string) { | ||||
| 	var buffer *bytes.Buffer | ||||
|  | ||||
| 	// Default to now, but allow users to override if they want. | ||||
| 	// | ||||
| 	// We don't have to worry about polluting future calls to Entry#log() | ||||
| 	// with this assignment because this function is declared with a | ||||
| 	// non-pointer receiver. | ||||
| 	if entry.Time.IsZero() { | ||||
| 		entry.Time = time.Now() | ||||
| 	} | ||||
|  | ||||
| 	entry.Level = level | ||||
| 	entry.Message = msg | ||||
| 	if entry.Logger.ReportCaller { | ||||
| 		entry.Caller = getCaller() | ||||
| 	} | ||||
|  | ||||
| 	entry.fireHooks() | ||||
|  | ||||
| 	buffer = bufferPool.Get().(*bytes.Buffer) | ||||
| 	buffer.Reset() | ||||
| 	defer bufferPool.Put(buffer) | ||||
| 	entry.Buffer = buffer | ||||
|  | ||||
| 	entry.write() | ||||
|  | ||||
| 	entry.Buffer = nil | ||||
|  | ||||
| 	// To avoid Entry#log() returning a value that only would make sense for | ||||
| 	// panic() to use in Entry#Panic(), we avoid the allocation by checking | ||||
| 	// directly here. | ||||
| 	if level <= PanicLevel { | ||||
| 		panic(&entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) fireHooks() { | ||||
| 	entry.Logger.mu.Lock() | ||||
| 	defer entry.Logger.mu.Unlock() | ||||
| 	err := entry.Logger.Hooks.Fire(entry.Level, entry) | ||||
| 	if err != nil { | ||||
| 		fmt.Fprintf(os.Stderr, "Failed to fire hook: %v\n", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) write() { | ||||
| 	entry.Logger.mu.Lock() | ||||
| 	defer entry.Logger.mu.Unlock() | ||||
| 	serialized, err := entry.Logger.Formatter.Format(entry) | ||||
| 	if err != nil { | ||||
| 		fmt.Fprintf(os.Stderr, "Failed to obtain reader, %v\n", err) | ||||
| 	} else { | ||||
| 		_, err = entry.Logger.Out.Write(serialized) | ||||
| 		if err != nil { | ||||
| 			fmt.Fprintf(os.Stderr, "Failed to write to log, %v\n", err) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Trace(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(TraceLevel) { | ||||
| 		entry.log(TraceLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Debug(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(DebugLevel) { | ||||
| 		entry.log(DebugLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Print(args ...interface{}) { | ||||
| 	entry.Info(args...) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Info(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(InfoLevel) { | ||||
| 		entry.log(InfoLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Warn(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry.log(WarnLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Warning(args ...interface{}) { | ||||
| 	entry.Warn(args...) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Error(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(ErrorLevel) { | ||||
| 		entry.log(ErrorLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Fatal(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(FatalLevel) { | ||||
| 		entry.log(FatalLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| 	entry.Logger.Exit(1) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Panic(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(PanicLevel) { | ||||
| 		entry.log(PanicLevel, fmt.Sprint(args...)) | ||||
| 	} | ||||
| 	panic(fmt.Sprint(args...)) | ||||
| } | ||||
|  | ||||
| // Entry Printf family functions | ||||
|  | ||||
| func (entry *Entry) Tracef(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(TraceLevel) { | ||||
| 		entry.Trace(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Debugf(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(DebugLevel) { | ||||
| 		entry.Debug(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Infof(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(InfoLevel) { | ||||
| 		entry.Info(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Printf(format string, args ...interface{}) { | ||||
| 	entry.Infof(format, args...) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Warnf(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry.Warn(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Warningf(format string, args ...interface{}) { | ||||
| 	entry.Warnf(format, args...) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Errorf(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(ErrorLevel) { | ||||
| 		entry.Error(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Fatalf(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(FatalLevel) { | ||||
| 		entry.Fatal(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| 	entry.Logger.Exit(1) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Panicf(format string, args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(PanicLevel) { | ||||
| 		entry.Panic(fmt.Sprintf(format, args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Entry Println family functions | ||||
|  | ||||
| func (entry *Entry) Traceln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(TraceLevel) { | ||||
| 		entry.Trace(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Debugln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(DebugLevel) { | ||||
| 		entry.Debug(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Infoln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(InfoLevel) { | ||||
| 		entry.Info(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Println(args ...interface{}) { | ||||
| 	entry.Infoln(args...) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Warnln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry.Warn(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Warningln(args ...interface{}) { | ||||
| 	entry.Warnln(args...) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Errorln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(ErrorLevel) { | ||||
| 		entry.Error(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Fatalln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(FatalLevel) { | ||||
| 		entry.Fatal(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| 	entry.Logger.Exit(1) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Panicln(args ...interface{}) { | ||||
| 	if entry.Logger.IsLevelEnabled(PanicLevel) { | ||||
| 		entry.Panic(entry.sprintlnn(args...)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Sprintlnn => Sprint no newline. This is to get the behavior of how | ||||
| // fmt.Sprintln where spaces are always added between operands, regardless of | ||||
| // their type. Instead of vendoring the Sprintln implementation to spare a | ||||
| // string allocation, we do the simplest thing. | ||||
| func (entry *Entry) sprintlnn(args ...interface{}) string { | ||||
| 	msg := fmt.Sprintln(args...) | ||||
| 	return msg[:len(msg)-1] | ||||
| } | ||||
							
								
								
									
										141
									
								
								vendor/github.com/sirupsen/logrus/entry_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										141
									
								
								vendor/github.com/sirupsen/logrus/entry_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,141 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| ) | ||||
|  | ||||
| func TestEntryWithError(t *testing.T) { | ||||
|  | ||||
| 	assert := assert.New(t) | ||||
|  | ||||
| 	defer func() { | ||||
| 		ErrorKey = "error" | ||||
| 	}() | ||||
|  | ||||
| 	err := fmt.Errorf("kaboom at layer %d", 4711) | ||||
|  | ||||
| 	assert.Equal(err, WithError(err).Data["error"]) | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &bytes.Buffer{} | ||||
| 	entry := NewEntry(logger) | ||||
|  | ||||
| 	assert.Equal(err, entry.WithError(err).Data["error"]) | ||||
|  | ||||
| 	ErrorKey = "err" | ||||
|  | ||||
| 	assert.Equal(err, entry.WithError(err).Data["err"]) | ||||
|  | ||||
| } | ||||
|  | ||||
| func TestEntryPanicln(t *testing.T) { | ||||
| 	errBoom := fmt.Errorf("boom time") | ||||
|  | ||||
| 	defer func() { | ||||
| 		p := recover() | ||||
| 		assert.NotNil(t, p) | ||||
|  | ||||
| 		switch pVal := p.(type) { | ||||
| 		case *Entry: | ||||
| 			assert.Equal(t, "kaboom", pVal.Message) | ||||
| 			assert.Equal(t, errBoom, pVal.Data["err"]) | ||||
| 		default: | ||||
| 			t.Fatalf("want type *Entry, got %T: %#v", pVal, pVal) | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &bytes.Buffer{} | ||||
| 	entry := NewEntry(logger) | ||||
| 	entry.WithField("err", errBoom).Panicln("kaboom") | ||||
| } | ||||
|  | ||||
| func TestEntryPanicf(t *testing.T) { | ||||
| 	errBoom := fmt.Errorf("boom again") | ||||
|  | ||||
| 	defer func() { | ||||
| 		p := recover() | ||||
| 		assert.NotNil(t, p) | ||||
|  | ||||
| 		switch pVal := p.(type) { | ||||
| 		case *Entry: | ||||
| 			assert.Equal(t, "kaboom true", pVal.Message) | ||||
| 			assert.Equal(t, errBoom, pVal.Data["err"]) | ||||
| 		default: | ||||
| 			t.Fatalf("want type *Entry, got %T: %#v", pVal, pVal) | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &bytes.Buffer{} | ||||
| 	entry := NewEntry(logger) | ||||
| 	entry.WithField("err", errBoom).Panicf("kaboom %v", true) | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	badMessage   = "this is going to panic" | ||||
| 	panicMessage = "this is broken" | ||||
| ) | ||||
|  | ||||
| type panickyHook struct{} | ||||
|  | ||||
| func (p *panickyHook) Levels() []Level { | ||||
| 	return []Level{InfoLevel} | ||||
| } | ||||
|  | ||||
| func (p *panickyHook) Fire(entry *Entry) error { | ||||
| 	if entry.Message == badMessage { | ||||
| 		panic(panicMessage) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func TestEntryHooksPanic(t *testing.T) { | ||||
| 	logger := New() | ||||
| 	logger.Out = &bytes.Buffer{} | ||||
| 	logger.Level = InfoLevel | ||||
| 	logger.Hooks.Add(&panickyHook{}) | ||||
|  | ||||
| 	defer func() { | ||||
| 		p := recover() | ||||
| 		assert.NotNil(t, p) | ||||
| 		assert.Equal(t, panicMessage, p) | ||||
|  | ||||
| 		entry := NewEntry(logger) | ||||
| 		entry.Info("another message") | ||||
| 	}() | ||||
|  | ||||
| 	entry := NewEntry(logger) | ||||
| 	entry.Info(badMessage) | ||||
| } | ||||
|  | ||||
| func TestEntryWithIncorrectField(t *testing.T) { | ||||
| 	assert := assert.New(t) | ||||
|  | ||||
| 	fn := func() {} | ||||
|  | ||||
| 	e := Entry{} | ||||
| 	eWithFunc := e.WithFields(Fields{"func": fn}) | ||||
| 	eWithFuncPtr := e.WithFields(Fields{"funcPtr": &fn}) | ||||
|  | ||||
| 	assert.Equal(eWithFunc.err, `can not add field "func"`) | ||||
| 	assert.Equal(eWithFuncPtr.err, `can not add field "funcPtr"`) | ||||
|  | ||||
| 	eWithFunc = eWithFunc.WithField("not_a_func", "it is a string") | ||||
| 	eWithFuncPtr = eWithFuncPtr.WithField("not_a_func", "it is a string") | ||||
|  | ||||
| 	assert.Equal(eWithFunc.err, `can not add field "func"`) | ||||
| 	assert.Equal(eWithFuncPtr.err, `can not add field "funcPtr"`) | ||||
|  | ||||
| 	eWithFunc = eWithFunc.WithTime(time.Now()) | ||||
| 	eWithFuncPtr = eWithFuncPtr.WithTime(time.Now()) | ||||
|  | ||||
| 	assert.Equal(eWithFunc.err, `can not add field "func"`) | ||||
| 	assert.Equal(eWithFuncPtr.err, `can not add field "funcPtr"`) | ||||
| } | ||||
							
								
								
									
										77
									
								
								vendor/github.com/sirupsen/logrus/example_basic_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										77
									
								
								vendor/github.com/sirupsen/logrus/example_basic_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| package logrus_test | ||||
|  | ||||
| import ( | ||||
| 	"os" | ||||
|  | ||||
| 	"github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| func Example_basic() { | ||||
| 	var log = logrus.New() | ||||
| 	log.Formatter = new(logrus.JSONFormatter) | ||||
| 	log.Formatter = new(logrus.TextFormatter)                     //default | ||||
| 	log.Formatter.(*logrus.TextFormatter).DisableColors = true    // remove colors | ||||
| 	log.Formatter.(*logrus.TextFormatter).DisableTimestamp = true // remove timestamp from test output | ||||
| 	log.Level = logrus.TraceLevel | ||||
| 	log.Out = os.Stdout | ||||
|  | ||||
| 	// file, err := os.OpenFile("logrus.log", os.O_CREATE|os.O_WRONLY, 0666) | ||||
| 	// if err == nil { | ||||
| 	// 	log.Out = file | ||||
| 	// } else { | ||||
| 	// 	log.Info("Failed to log to file, using default stderr") | ||||
| 	// } | ||||
|  | ||||
| 	defer func() { | ||||
| 		err := recover() | ||||
| 		if err != nil { | ||||
| 			entry := err.(*logrus.Entry) | ||||
| 			log.WithFields(logrus.Fields{ | ||||
| 				"omg":         true, | ||||
| 				"err_animal":  entry.Data["animal"], | ||||
| 				"err_size":    entry.Data["size"], | ||||
| 				"err_level":   entry.Level, | ||||
| 				"err_message": entry.Message, | ||||
| 				"number":      100, | ||||
| 			}).Error("The ice breaks!") // or use Fatal() to force the process to exit with a nonzero code | ||||
| 		} | ||||
| 	}() | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"animal": "walrus", | ||||
| 		"number": 0, | ||||
| 	}).Trace("Went to the beach") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"animal": "walrus", | ||||
| 		"number": 8, | ||||
| 	}).Debug("Started observing beach") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"animal": "walrus", | ||||
| 		"size":   10, | ||||
| 	}).Info("A group of walrus emerges from the ocean") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"omg":    true, | ||||
| 		"number": 122, | ||||
| 	}).Warn("The group's number increased tremendously!") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"temperature": -4, | ||||
| 	}).Debug("Temperature changes") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"animal": "orca", | ||||
| 		"size":   9009, | ||||
| 	}).Panic("It's over 9000!") | ||||
|  | ||||
| 	// Output: | ||||
| 	// level=trace msg="Went to the beach" animal=walrus number=0 | ||||
| 	// level=debug msg="Started observing beach" animal=walrus number=8 | ||||
| 	// level=info msg="A group of walrus emerges from the ocean" animal=walrus size=10 | ||||
| 	// level=warning msg="The group's number increased tremendously!" number=122 omg=true | ||||
| 	// level=debug msg="Temperature changes" temperature=-4 | ||||
| 	// level=panic msg="It's over 9000!" animal=orca size=9009 | ||||
| 	// level=error msg="The ice breaks!" err_animal=orca err_level=panic err_message="It's over 9000!" err_size=9009 number=100 omg=true | ||||
| } | ||||
							
								
								
									
										36
									
								
								vendor/github.com/sirupsen/logrus/example_global_hook_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								vendor/github.com/sirupsen/logrus/example_global_hook_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| package logrus_test | ||||
|  | ||||
| import ( | ||||
| 	"github.com/sirupsen/logrus" | ||||
| 	"os" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	mystring string | ||||
| ) | ||||
|  | ||||
| type GlobalHook struct { | ||||
| } | ||||
|  | ||||
| func (h *GlobalHook) Levels() []logrus.Level { | ||||
| 	return logrus.AllLevels | ||||
| } | ||||
|  | ||||
| func (h *GlobalHook) Fire(e *logrus.Entry) error { | ||||
| 	e.Data["mystring"] = mystring | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func Example() { | ||||
| 	l := logrus.New() | ||||
| 	l.Out = os.Stdout | ||||
| 	l.Formatter = &logrus.TextFormatter{DisableTimestamp: true, DisableColors: true} | ||||
| 	l.AddHook(&GlobalHook{}) | ||||
| 	mystring = "first value" | ||||
| 	l.Info("first log") | ||||
| 	mystring = "another value" | ||||
| 	l.Info("second log") | ||||
| 	// Output: | ||||
| 	// level=info msg="first log" mystring="first value" | ||||
| 	// level=info msg="second log" mystring="another value" | ||||
| } | ||||
							
								
								
									
										43
									
								
								vendor/github.com/sirupsen/logrus/example_hook_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								vendor/github.com/sirupsen/logrus/example_hook_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| // +build !windows | ||||
|  | ||||
| package logrus_test | ||||
|  | ||||
| import ( | ||||
| 	"log/syslog" | ||||
| 	"os" | ||||
|  | ||||
| 	"github.com/sirupsen/logrus" | ||||
| 	slhooks "github.com/sirupsen/logrus/hooks/syslog" | ||||
| ) | ||||
|  | ||||
| // An example on how to use a hook | ||||
| func Example_hook() { | ||||
| 	var log = logrus.New() | ||||
| 	log.Formatter = new(logrus.TextFormatter)                     // default | ||||
| 	log.Formatter.(*logrus.TextFormatter).DisableColors = true    // remove colors | ||||
| 	log.Formatter.(*logrus.TextFormatter).DisableTimestamp = true // remove timestamp from test output | ||||
| 	if sl, err := slhooks.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, ""); err == nil { | ||||
| 		log.Hooks.Add(sl) | ||||
| 	} | ||||
| 	log.Out = os.Stdout | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"animal": "walrus", | ||||
| 		"size":   10, | ||||
| 	}).Info("A group of walrus emerges from the ocean") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"omg":    true, | ||||
| 		"number": 122, | ||||
| 	}).Warn("The group's number increased tremendously!") | ||||
|  | ||||
| 	log.WithFields(logrus.Fields{ | ||||
| 		"omg":    true, | ||||
| 		"number": 100, | ||||
| 	}).Error("The ice breaks!") | ||||
|  | ||||
| 	// Output: | ||||
| 	// level=info msg="A group of walrus emerges from the ocean" animal=walrus size=10 | ||||
| 	// level=warning msg="The group's number increased tremendously!" number=122 omg=true | ||||
| 	// level=error msg="The ice breaks!" number=100 omg=true | ||||
| } | ||||
							
								
								
									
										219
									
								
								vendor/github.com/sirupsen/logrus/exported.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										219
									
								
								vendor/github.com/sirupsen/logrus/exported.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,219 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	// std is the name of the standard logger in stdlib `log` | ||||
| 	std = New() | ||||
| ) | ||||
|  | ||||
| func StandardLogger() *Logger { | ||||
| 	return std | ||||
| } | ||||
|  | ||||
| // SetOutput sets the standard logger output. | ||||
| func SetOutput(out io.Writer) { | ||||
| 	std.SetOutput(out) | ||||
| } | ||||
|  | ||||
| // SetFormatter sets the standard logger formatter. | ||||
| func SetFormatter(formatter Formatter) { | ||||
| 	std.SetFormatter(formatter) | ||||
| } | ||||
|  | ||||
| // SetReportCaller sets whether the standard logger will include the calling | ||||
| // method as a field. | ||||
| func SetReportCaller(include bool) { | ||||
| 	std.SetReportCaller(include) | ||||
| } | ||||
|  | ||||
| // SetLevel sets the standard logger level. | ||||
| func SetLevel(level Level) { | ||||
| 	std.SetLevel(level) | ||||
| } | ||||
|  | ||||
| // GetLevel returns the standard logger level. | ||||
| func GetLevel() Level { | ||||
| 	return std.GetLevel() | ||||
| } | ||||
|  | ||||
| // IsLevelEnabled checks if the log level of the standard logger is greater than the level param | ||||
| func IsLevelEnabled(level Level) bool { | ||||
| 	return std.IsLevelEnabled(level) | ||||
| } | ||||
|  | ||||
| // AddHook adds a hook to the standard logger hooks. | ||||
| func AddHook(hook Hook) { | ||||
| 	std.AddHook(hook) | ||||
| } | ||||
|  | ||||
| // WithError creates an entry from the standard logger and adds an error to it, using the value defined in ErrorKey as key. | ||||
| func WithError(err error) *Entry { | ||||
| 	return std.WithField(ErrorKey, err) | ||||
| } | ||||
|  | ||||
| // WithField creates an entry from the standard logger and adds a field to | ||||
| // it. If you want multiple fields, use `WithFields`. | ||||
| // | ||||
| // Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal | ||||
| // or Panic on the Entry it returns. | ||||
| func WithField(key string, value interface{}) *Entry { | ||||
| 	return std.WithField(key, value) | ||||
| } | ||||
|  | ||||
| // WithFields creates an entry from the standard logger and adds multiple | ||||
| // fields to it. This is simply a helper for `WithField`, invoking it | ||||
| // once for each field. | ||||
| // | ||||
| // Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal | ||||
| // or Panic on the Entry it returns. | ||||
| func WithFields(fields Fields) *Entry { | ||||
| 	return std.WithFields(fields) | ||||
| } | ||||
|  | ||||
| // WithTime creats an entry from the standard logger and overrides the time of | ||||
| // logs generated with it. | ||||
| // | ||||
| // Note that it doesn't log until you call Debug, Print, Info, Warn, Fatal | ||||
| // or Panic on the Entry it returns. | ||||
| func WithTime(t time.Time) *Entry { | ||||
| 	return std.WithTime(t) | ||||
| } | ||||
|  | ||||
| // Trace logs a message at level Trace on the standard logger. | ||||
| func Trace(args ...interface{}) { | ||||
| 	std.Trace(args...) | ||||
| } | ||||
|  | ||||
| // Debug logs a message at level Debug on the standard logger. | ||||
| func Debug(args ...interface{}) { | ||||
| 	std.Debug(args...) | ||||
| } | ||||
|  | ||||
| // Print logs a message at level Info on the standard logger. | ||||
| func Print(args ...interface{}) { | ||||
| 	std.Print(args...) | ||||
| } | ||||
|  | ||||
| // Info logs a message at level Info on the standard logger. | ||||
| func Info(args ...interface{}) { | ||||
| 	std.Info(args...) | ||||
| } | ||||
|  | ||||
| // Warn logs a message at level Warn on the standard logger. | ||||
| func Warn(args ...interface{}) { | ||||
| 	std.Warn(args...) | ||||
| } | ||||
|  | ||||
| // Warning logs a message at level Warn on the standard logger. | ||||
| func Warning(args ...interface{}) { | ||||
| 	std.Warning(args...) | ||||
| } | ||||
|  | ||||
| // Error logs a message at level Error on the standard logger. | ||||
| func Error(args ...interface{}) { | ||||
| 	std.Error(args...) | ||||
| } | ||||
|  | ||||
| // Panic logs a message at level Panic on the standard logger. | ||||
| func Panic(args ...interface{}) { | ||||
| 	std.Panic(args...) | ||||
| } | ||||
|  | ||||
| // Fatal logs a message at level Fatal on the standard logger then the process will exit with status set to 1. | ||||
| func Fatal(args ...interface{}) { | ||||
| 	std.Fatal(args...) | ||||
| } | ||||
|  | ||||
| // Tracef logs a message at level Trace on the standard logger. | ||||
| func Tracef(format string, args ...interface{}) { | ||||
| 	std.Tracef(format, args...) | ||||
| } | ||||
|  | ||||
| // Debugf logs a message at level Debug on the standard logger. | ||||
| func Debugf(format string, args ...interface{}) { | ||||
| 	std.Debugf(format, args...) | ||||
| } | ||||
|  | ||||
| // Printf logs a message at level Info on the standard logger. | ||||
| func Printf(format string, args ...interface{}) { | ||||
| 	std.Printf(format, args...) | ||||
| } | ||||
|  | ||||
| // Infof logs a message at level Info on the standard logger. | ||||
| func Infof(format string, args ...interface{}) { | ||||
| 	std.Infof(format, args...) | ||||
| } | ||||
|  | ||||
| // Warnf logs a message at level Warn on the standard logger. | ||||
| func Warnf(format string, args ...interface{}) { | ||||
| 	std.Warnf(format, args...) | ||||
| } | ||||
|  | ||||
| // Warningf logs a message at level Warn on the standard logger. | ||||
| func Warningf(format string, args ...interface{}) { | ||||
| 	std.Warningf(format, args...) | ||||
| } | ||||
|  | ||||
| // Errorf logs a message at level Error on the standard logger. | ||||
| func Errorf(format string, args ...interface{}) { | ||||
| 	std.Errorf(format, args...) | ||||
| } | ||||
|  | ||||
| // Panicf logs a message at level Panic on the standard logger. | ||||
| func Panicf(format string, args ...interface{}) { | ||||
| 	std.Panicf(format, args...) | ||||
| } | ||||
|  | ||||
| // Fatalf logs a message at level Fatal on the standard logger then the process will exit with status set to 1. | ||||
| func Fatalf(format string, args ...interface{}) { | ||||
| 	std.Fatalf(format, args...) | ||||
| } | ||||
|  | ||||
| // Traceln logs a message at level Trace on the standard logger. | ||||
| func Traceln(args ...interface{}) { | ||||
| 	std.Traceln(args...) | ||||
| } | ||||
|  | ||||
| // Debugln logs a message at level Debug on the standard logger. | ||||
| func Debugln(args ...interface{}) { | ||||
| 	std.Debugln(args...) | ||||
| } | ||||
|  | ||||
| // Println logs a message at level Info on the standard logger. | ||||
| func Println(args ...interface{}) { | ||||
| 	std.Println(args...) | ||||
| } | ||||
|  | ||||
| // Infoln logs a message at level Info on the standard logger. | ||||
| func Infoln(args ...interface{}) { | ||||
| 	std.Infoln(args...) | ||||
| } | ||||
|  | ||||
| // Warnln logs a message at level Warn on the standard logger. | ||||
| func Warnln(args ...interface{}) { | ||||
| 	std.Warnln(args...) | ||||
| } | ||||
|  | ||||
| // Warningln logs a message at level Warn on the standard logger. | ||||
| func Warningln(args ...interface{}) { | ||||
| 	std.Warningln(args...) | ||||
| } | ||||
|  | ||||
| // Errorln logs a message at level Error on the standard logger. | ||||
| func Errorln(args ...interface{}) { | ||||
| 	std.Errorln(args...) | ||||
| } | ||||
|  | ||||
| // Panicln logs a message at level Panic on the standard logger. | ||||
| func Panicln(args ...interface{}) { | ||||
| 	std.Panicln(args...) | ||||
| } | ||||
|  | ||||
| // Fatalln logs a message at level Fatal on the standard logger then the process will exit with status set to 1. | ||||
| func Fatalln(args ...interface{}) { | ||||
| 	std.Fatalln(args...) | ||||
| } | ||||
							
								
								
									
										78
									
								
								vendor/github.com/sirupsen/logrus/formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								vendor/github.com/sirupsen/logrus/formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,78 @@ | ||||
| package logrus | ||||
|  | ||||
| import "time" | ||||
|  | ||||
| // Default key names for the default fields | ||||
| const ( | ||||
| 	defaultTimestampFormat = time.RFC3339 | ||||
| 	FieldKeyMsg            = "msg" | ||||
| 	FieldKeyLevel          = "level" | ||||
| 	FieldKeyTime           = "time" | ||||
| 	FieldKeyLogrusError    = "logrus_error" | ||||
| 	FieldKeyFunc           = "func" | ||||
| 	FieldKeyFile           = "file" | ||||
| ) | ||||
|  | ||||
| // The Formatter interface is used to implement a custom Formatter. It takes an | ||||
| // `Entry`. It exposes all the fields, including the default ones: | ||||
| // | ||||
| // * `entry.Data["msg"]`. The message passed from Info, Warn, Error .. | ||||
| // * `entry.Data["time"]`. The timestamp. | ||||
| // * `entry.Data["level"]. The level the entry was logged at. | ||||
| // | ||||
| // Any additional fields added with `WithField` or `WithFields` are also in | ||||
| // `entry.Data`. Format is expected to return an array of bytes which are then | ||||
| // logged to `logger.Out`. | ||||
| type Formatter interface { | ||||
| 	Format(*Entry) ([]byte, error) | ||||
| } | ||||
|  | ||||
| // This is to not silently overwrite `time`, `msg`, `func` and `level` fields when | ||||
| // dumping it. If this code wasn't there doing: | ||||
| // | ||||
| //  logrus.WithField("level", 1).Info("hello") | ||||
| // | ||||
| // Would just silently drop the user provided level. Instead with this code | ||||
| // it'll logged as: | ||||
| // | ||||
| //  {"level": "info", "fields.level": 1, "msg": "hello", "time": "..."} | ||||
| // | ||||
| // It's not exported because it's still using Data in an opinionated way. It's to | ||||
| // avoid code duplication between the two default formatters. | ||||
| func prefixFieldClashes(data Fields, fieldMap FieldMap, reportCaller bool) { | ||||
| 	timeKey := fieldMap.resolve(FieldKeyTime) | ||||
| 	if t, ok := data[timeKey]; ok { | ||||
| 		data["fields."+timeKey] = t | ||||
| 		delete(data, timeKey) | ||||
| 	} | ||||
|  | ||||
| 	msgKey := fieldMap.resolve(FieldKeyMsg) | ||||
| 	if m, ok := data[msgKey]; ok { | ||||
| 		data["fields."+msgKey] = m | ||||
| 		delete(data, msgKey) | ||||
| 	} | ||||
|  | ||||
| 	levelKey := fieldMap.resolve(FieldKeyLevel) | ||||
| 	if l, ok := data[levelKey]; ok { | ||||
| 		data["fields."+levelKey] = l | ||||
| 		delete(data, levelKey) | ||||
| 	} | ||||
|  | ||||
| 	logrusErrKey := fieldMap.resolve(FieldKeyLogrusError) | ||||
| 	if l, ok := data[logrusErrKey]; ok { | ||||
| 		data["fields."+logrusErrKey] = l | ||||
| 		delete(data, logrusErrKey) | ||||
| 	} | ||||
|  | ||||
| 	// If reportCaller is not set, 'func' will not conflict. | ||||
| 	if reportCaller { | ||||
| 		funcKey := fieldMap.resolve(FieldKeyFunc) | ||||
| 		if l, ok := data[funcKey]; ok { | ||||
| 			data["fields."+funcKey] = l | ||||
| 		} | ||||
| 		fileKey := fieldMap.resolve(FieldKeyFile) | ||||
| 		if l, ok := data[fileKey]; ok { | ||||
| 			data["fields."+fileKey] = l | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										101
									
								
								vendor/github.com/sirupsen/logrus/formatter_bench_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								vendor/github.com/sirupsen/logrus/formatter_bench_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,101 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| // smallFields is a small size data set for benchmarking | ||||
| var smallFields = Fields{ | ||||
| 	"foo":   "bar", | ||||
| 	"baz":   "qux", | ||||
| 	"one":   "two", | ||||
| 	"three": "four", | ||||
| } | ||||
|  | ||||
| // largeFields is a large size data set for benchmarking | ||||
| var largeFields = Fields{ | ||||
| 	"foo":       "bar", | ||||
| 	"baz":       "qux", | ||||
| 	"one":       "two", | ||||
| 	"three":     "four", | ||||
| 	"five":      "six", | ||||
| 	"seven":     "eight", | ||||
| 	"nine":      "ten", | ||||
| 	"eleven":    "twelve", | ||||
| 	"thirteen":  "fourteen", | ||||
| 	"fifteen":   "sixteen", | ||||
| 	"seventeen": "eighteen", | ||||
| 	"nineteen":  "twenty", | ||||
| 	"a":         "b", | ||||
| 	"c":         "d", | ||||
| 	"e":         "f", | ||||
| 	"g":         "h", | ||||
| 	"i":         "j", | ||||
| 	"k":         "l", | ||||
| 	"m":         "n", | ||||
| 	"o":         "p", | ||||
| 	"q":         "r", | ||||
| 	"s":         "t", | ||||
| 	"u":         "v", | ||||
| 	"w":         "x", | ||||
| 	"y":         "z", | ||||
| 	"this":      "will", | ||||
| 	"make":      "thirty", | ||||
| 	"entries":   "yeah", | ||||
| } | ||||
|  | ||||
| var errorFields = Fields{ | ||||
| 	"foo": fmt.Errorf("bar"), | ||||
| 	"baz": fmt.Errorf("qux"), | ||||
| } | ||||
|  | ||||
| func BenchmarkErrorTextFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &TextFormatter{DisableColors: true}, errorFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkSmallTextFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &TextFormatter{DisableColors: true}, smallFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkLargeTextFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &TextFormatter{DisableColors: true}, largeFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkSmallColoredTextFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &TextFormatter{ForceColors: true}, smallFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkLargeColoredTextFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &TextFormatter{ForceColors: true}, largeFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkSmallJSONFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &JSONFormatter{}, smallFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkLargeJSONFormatter(b *testing.B) { | ||||
| 	doBenchmark(b, &JSONFormatter{}, largeFields) | ||||
| } | ||||
|  | ||||
| func doBenchmark(b *testing.B, formatter Formatter, fields Fields) { | ||||
| 	logger := New() | ||||
|  | ||||
| 	entry := &Entry{ | ||||
| 		Time:    time.Time{}, | ||||
| 		Level:   InfoLevel, | ||||
| 		Message: "message", | ||||
| 		Data:    fields, | ||||
| 		Logger:  logger, | ||||
| 	} | ||||
| 	var d []byte | ||||
| 	var err error | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		d, err = formatter.Format(entry) | ||||
| 		if err != nil { | ||||
| 			b.Fatal(err) | ||||
| 		} | ||||
| 		b.SetBytes(int64(len(d))) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										11
									
								
								vendor/github.com/sirupsen/logrus/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								vendor/github.com/sirupsen/logrus/go.mod
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| module github.com/sirupsen/logrus | ||||
|  | ||||
| require ( | ||||
| 	github.com/davecgh/go-spew v1.1.1 // indirect | ||||
| 	github.com/konsorten/go-windows-terminal-sequences v1.0.1 | ||||
| 	github.com/pmezard/go-difflib v1.0.0 // indirect | ||||
| 	github.com/stretchr/objx v0.1.1 // indirect | ||||
| 	github.com/stretchr/testify v1.2.2 | ||||
| 	golang.org/x/crypto v0.0.0-20180904163835-0709b304e793 | ||||
| 	golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33 | ||||
| ) | ||||
							
								
								
									
										15
									
								
								vendor/github.com/sirupsen/logrus/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								vendor/github.com/sirupsen/logrus/go.sum
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= | ||||
| github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||
| github.com/konsorten/go-windows-terminal-sequences v0.0.0-20180402223658-b729f2633dfe h1:CHRGQ8V7OlCYtwaKPJi3iA7J+YdNKdo8j7nG5IgDhjs= | ||||
| github.com/konsorten/go-windows-terminal-sequences v0.0.0-20180402223658-b729f2633dfe/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= | ||||
| github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= | ||||
| github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||
| github.com/stretchr/objx v0.1.1 h1:2vfRuCMp5sSVIDSqO8oNnWJq7mPa6KVP3iPIwFBuy8A= | ||||
| github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||
| github.com/stretchr/testify v1.2.2 h1:bSDNvY7ZPG5RlJ8otE/7V6gMiyenm9RtJ7IUVIAoJ1w= | ||||
| github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= | ||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793 h1:u+LnwYTOOW7Ukr/fppxEb1Nwz0AtPflrblfvUudpo+I= | ||||
| golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= | ||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33 h1:I6FyU15t786LL7oL/hn43zqTuEGr4PN7F4XJ1p4E3Y8= | ||||
| golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||
							
								
								
									
										192
									
								
								vendor/github.com/sirupsen/logrus/hook_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										192
									
								
								vendor/github.com/sirupsen/logrus/hook_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,192 @@ | ||||
| package logrus_test | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"github.com/stretchr/testify/require" | ||||
|  | ||||
| 	. "github.com/sirupsen/logrus" | ||||
| 	. "github.com/sirupsen/logrus/internal/testutils" | ||||
| ) | ||||
|  | ||||
| type TestHook struct { | ||||
| 	Fired bool | ||||
| } | ||||
|  | ||||
| func (hook *TestHook) Fire(entry *Entry) error { | ||||
| 	hook.Fired = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (hook *TestHook) Levels() []Level { | ||||
| 	return []Level{ | ||||
| 		TraceLevel, | ||||
| 		DebugLevel, | ||||
| 		InfoLevel, | ||||
| 		WarnLevel, | ||||
| 		ErrorLevel, | ||||
| 		FatalLevel, | ||||
| 		PanicLevel, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestHookFires(t *testing.T) { | ||||
| 	hook := new(TestHook) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Hooks.Add(hook) | ||||
| 		assert.Equal(t, hook.Fired, false) | ||||
|  | ||||
| 		log.Print("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, hook.Fired, true) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| type ModifyHook struct { | ||||
| } | ||||
|  | ||||
| func (hook *ModifyHook) Fire(entry *Entry) error { | ||||
| 	entry.Data["wow"] = "whale" | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (hook *ModifyHook) Levels() []Level { | ||||
| 	return []Level{ | ||||
| 		TraceLevel, | ||||
| 		DebugLevel, | ||||
| 		InfoLevel, | ||||
| 		WarnLevel, | ||||
| 		ErrorLevel, | ||||
| 		FatalLevel, | ||||
| 		PanicLevel, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestHookCanModifyEntry(t *testing.T) { | ||||
| 	hook := new(ModifyHook) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Hooks.Add(hook) | ||||
| 		log.WithField("wow", "elephant").Print("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, fields["wow"], "whale") | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestCanFireMultipleHooks(t *testing.T) { | ||||
| 	hook1 := new(ModifyHook) | ||||
| 	hook2 := new(TestHook) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Hooks.Add(hook1) | ||||
| 		log.Hooks.Add(hook2) | ||||
|  | ||||
| 		log.WithField("wow", "elephant").Print("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, fields["wow"], "whale") | ||||
| 		assert.Equal(t, hook2.Fired, true) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| type SingleLevelModifyHook struct { | ||||
| 	ModifyHook | ||||
| } | ||||
|  | ||||
| func (h *SingleLevelModifyHook) Levels() []Level { | ||||
| 	return []Level{InfoLevel} | ||||
| } | ||||
|  | ||||
| func TestHookEntryIsPristine(t *testing.T) { | ||||
| 	l := New() | ||||
| 	b := &bytes.Buffer{} | ||||
| 	l.Formatter = &JSONFormatter{} | ||||
| 	l.Out = b | ||||
| 	l.AddHook(&SingleLevelModifyHook{}) | ||||
|  | ||||
| 	l.Error("error message") | ||||
| 	data := map[string]string{} | ||||
| 	err := json.Unmarshal(b.Bytes(), &data) | ||||
| 	require.NoError(t, err) | ||||
| 	_, ok := data["wow"] | ||||
| 	require.False(t, ok) | ||||
| 	b.Reset() | ||||
|  | ||||
| 	l.Info("error message") | ||||
| 	data = map[string]string{} | ||||
| 	err = json.Unmarshal(b.Bytes(), &data) | ||||
| 	require.NoError(t, err) | ||||
| 	_, ok = data["wow"] | ||||
| 	require.True(t, ok) | ||||
| 	b.Reset() | ||||
|  | ||||
| 	l.Error("error message") | ||||
| 	data = map[string]string{} | ||||
| 	err = json.Unmarshal(b.Bytes(), &data) | ||||
| 	require.NoError(t, err) | ||||
| 	_, ok = data["wow"] | ||||
| 	require.False(t, ok) | ||||
| 	b.Reset() | ||||
| } | ||||
|  | ||||
| type ErrorHook struct { | ||||
| 	Fired bool | ||||
| } | ||||
|  | ||||
| func (hook *ErrorHook) Fire(entry *Entry) error { | ||||
| 	hook.Fired = true | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| func (hook *ErrorHook) Levels() []Level { | ||||
| 	return []Level{ | ||||
| 		ErrorLevel, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorHookShouldntFireOnInfo(t *testing.T) { | ||||
| 	hook := new(ErrorHook) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Hooks.Add(hook) | ||||
| 		log.Info("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, hook.Fired, false) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestErrorHookShouldFireOnError(t *testing.T) { | ||||
| 	hook := new(ErrorHook) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Hooks.Add(hook) | ||||
| 		log.Error("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, hook.Fired, true) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestAddHookRace(t *testing.T) { | ||||
| 	var wg sync.WaitGroup | ||||
| 	wg.Add(2) | ||||
| 	hook := new(ErrorHook) | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		go func() { | ||||
| 			defer wg.Done() | ||||
| 			log.AddHook(hook) | ||||
| 		}() | ||||
| 		go func() { | ||||
| 			defer wg.Done() | ||||
| 			log.Error("test") | ||||
| 		}() | ||||
| 		wg.Wait() | ||||
| 	}, func(fields Fields) { | ||||
| 		// the line may have been logged | ||||
| 		// before the hook was added, so we can't | ||||
| 		// actually assert on the hook | ||||
| 	}) | ||||
| } | ||||
							
								
								
									
										34
									
								
								vendor/github.com/sirupsen/logrus/hooks.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								vendor/github.com/sirupsen/logrus/hooks.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| package logrus | ||||
|  | ||||
| // A hook to be fired when logging on the logging levels returned from | ||||
| // `Levels()` on your implementation of the interface. Note that this is not | ||||
| // fired in a goroutine or a channel with workers, you should handle such | ||||
| // functionality yourself if your call is non-blocking and you don't wish for | ||||
| // the logging calls for levels returned from `Levels()` to block. | ||||
| type Hook interface { | ||||
| 	Levels() []Level | ||||
| 	Fire(*Entry) error | ||||
| } | ||||
|  | ||||
| // Internal type for storing the hooks on a logger instance. | ||||
| type LevelHooks map[Level][]Hook | ||||
|  | ||||
| // Add a hook to an instance of logger. This is called with | ||||
| // `log.Hooks.Add(new(MyHook))` where `MyHook` implements the `Hook` interface. | ||||
| func (hooks LevelHooks) Add(hook Hook) { | ||||
| 	for _, level := range hook.Levels() { | ||||
| 		hooks[level] = append(hooks[level], hook) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Fire all the hooks for the passed level. Used by `entry.log` to fire | ||||
| // appropriate hooks for a log entry. | ||||
| func (hooks LevelHooks) Fire(level Level, entry *Entry) error { | ||||
| 	for _, hook := range hooks[level] { | ||||
| 		if err := hook.Fire(entry); err != nil { | ||||
| 			return err | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
							
								
								
									
										39
									
								
								vendor/github.com/sirupsen/logrus/hooks/syslog/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										39
									
								
								vendor/github.com/sirupsen/logrus/hooks/syslog/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,39 @@ | ||||
| # Syslog Hooks for Logrus <img src="http://i.imgur.com/hTeVwmJ.png" width="40" height="40" alt=":walrus:" class="emoji" title=":walrus:"/> | ||||
|  | ||||
| ## Usage | ||||
|  | ||||
| ```go | ||||
| import ( | ||||
|   "log/syslog" | ||||
|   "github.com/sirupsen/logrus" | ||||
|   lSyslog "github.com/sirupsen/logrus/hooks/syslog" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
|   log       := logrus.New() | ||||
|   hook, err := lSyslog.NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") | ||||
|  | ||||
|   if err == nil { | ||||
|     log.Hooks.Add(hook) | ||||
|   } | ||||
| } | ||||
| ``` | ||||
|  | ||||
| If you want to connect to local syslog (Ex. "/dev/log" or "/var/run/syslog" or "/var/run/log"). Just assign empty string to the first two parameters of `NewSyslogHook`. It should look like the following. | ||||
|  | ||||
| ```go | ||||
| import ( | ||||
|   "log/syslog" | ||||
|   "github.com/sirupsen/logrus" | ||||
|   lSyslog "github.com/sirupsen/logrus/hooks/syslog" | ||||
| ) | ||||
|  | ||||
| func main() { | ||||
|   log       := logrus.New() | ||||
|   hook, err := lSyslog.NewSyslogHook("", "", syslog.LOG_INFO, "") | ||||
|  | ||||
|   if err == nil { | ||||
|     log.Hooks.Add(hook) | ||||
|   } | ||||
| } | ||||
| ``` | ||||
							
								
								
									
										55
									
								
								vendor/github.com/sirupsen/logrus/hooks/syslog/syslog.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								vendor/github.com/sirupsen/logrus/hooks/syslog/syslog.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| // +build !windows,!nacl,!plan9 | ||||
|  | ||||
| package syslog | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"log/syslog" | ||||
| 	"os" | ||||
|  | ||||
| 	"github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| // SyslogHook to send logs via syslog. | ||||
| type SyslogHook struct { | ||||
| 	Writer        *syslog.Writer | ||||
| 	SyslogNetwork string | ||||
| 	SyslogRaddr   string | ||||
| } | ||||
|  | ||||
| // Creates a hook to be added to an instance of logger. This is called with | ||||
| // `hook, err := NewSyslogHook("udp", "localhost:514", syslog.LOG_DEBUG, "")` | ||||
| // `if err == nil { log.Hooks.Add(hook) }` | ||||
| func NewSyslogHook(network, raddr string, priority syslog.Priority, tag string) (*SyslogHook, error) { | ||||
| 	w, err := syslog.Dial(network, raddr, priority, tag) | ||||
| 	return &SyslogHook{w, network, raddr}, err | ||||
| } | ||||
|  | ||||
| func (hook *SyslogHook) Fire(entry *logrus.Entry) error { | ||||
| 	line, err := entry.String() | ||||
| 	if err != nil { | ||||
| 		fmt.Fprintf(os.Stderr, "Unable to read entry, %v", err) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	switch entry.Level { | ||||
| 	case logrus.PanicLevel: | ||||
| 		return hook.Writer.Crit(line) | ||||
| 	case logrus.FatalLevel: | ||||
| 		return hook.Writer.Crit(line) | ||||
| 	case logrus.ErrorLevel: | ||||
| 		return hook.Writer.Err(line) | ||||
| 	case logrus.WarnLevel: | ||||
| 		return hook.Writer.Warning(line) | ||||
| 	case logrus.InfoLevel: | ||||
| 		return hook.Writer.Info(line) | ||||
| 	case logrus.DebugLevel, logrus.TraceLevel: | ||||
| 		return hook.Writer.Debug(line) | ||||
| 	default: | ||||
| 		return nil | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (hook *SyslogHook) Levels() []logrus.Level { | ||||
| 	return logrus.AllLevels | ||||
| } | ||||
							
								
								
									
										29
									
								
								vendor/github.com/sirupsen/logrus/hooks/syslog/syslog_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/github.com/sirupsen/logrus/hooks/syslog/syslog_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // +build !windows,!nacl,!plan9 | ||||
|  | ||||
| package syslog | ||||
|  | ||||
| import ( | ||||
| 	"log/syslog" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/sirupsen/logrus" | ||||
| ) | ||||
|  | ||||
| func TestLocalhostAddAndPrint(t *testing.T) { | ||||
| 	log := logrus.New() | ||||
| 	hook, err := NewSyslogHook("udp", "localhost:514", syslog.LOG_INFO, "") | ||||
|  | ||||
| 	if err != nil { | ||||
| 		t.Errorf("Unable to connect to local syslog.") | ||||
| 	} | ||||
|  | ||||
| 	log.Hooks.Add(hook) | ||||
|  | ||||
| 	for _, level := range hook.Levels() { | ||||
| 		if len(log.Hooks[level]) != 1 { | ||||
| 			t.Errorf("SyslogHook was not added. The length of log.Hooks[%v]: %v", level, len(log.Hooks[level])) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	log.Info("Congratulations!") | ||||
| } | ||||
							
								
								
									
										58
									
								
								vendor/github.com/sirupsen/logrus/internal/testutils/testutils.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								vendor/github.com/sirupsen/logrus/internal/testutils/testutils.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| package testutils | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"strconv" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
|  | ||||
| 	. "github.com/sirupsen/logrus" | ||||
|  | ||||
| 	"github.com/stretchr/testify/require" | ||||
| ) | ||||
|  | ||||
| func LogAndAssertJSON(t *testing.T, log func(*Logger), assertions func(fields Fields)) { | ||||
| 	var buffer bytes.Buffer | ||||
| 	var fields Fields | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = new(JSONFormatter) | ||||
|  | ||||
| 	log(logger) | ||||
|  | ||||
| 	err := json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	require.Nil(t, err) | ||||
|  | ||||
| 	assertions(fields) | ||||
| } | ||||
|  | ||||
| func LogAndAssertText(t *testing.T, log func(*Logger), assertions func(fields map[string]string)) { | ||||
| 	var buffer bytes.Buffer | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = &TextFormatter{ | ||||
| 		DisableColors: true, | ||||
| 	} | ||||
|  | ||||
| 	log(logger) | ||||
|  | ||||
| 	fields := make(map[string]string) | ||||
| 	for _, kv := range strings.Split(buffer.String(), " ") { | ||||
| 		if !strings.Contains(kv, "=") { | ||||
| 			continue | ||||
| 		} | ||||
| 		kvArr := strings.Split(kv, "=") | ||||
| 		key := strings.TrimSpace(kvArr[0]) | ||||
| 		val := kvArr[1] | ||||
| 		if kvArr[1][0] == '"' { | ||||
| 			var err error | ||||
| 			val, err = strconv.Unquote(val) | ||||
| 			require.NoError(t, err) | ||||
| 		} | ||||
| 		fields[key] = val | ||||
| 	} | ||||
| 	assertions(fields) | ||||
| } | ||||
							
								
								
									
										105
									
								
								vendor/github.com/sirupsen/logrus/json_formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								vendor/github.com/sirupsen/logrus/json_formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,105 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| ) | ||||
|  | ||||
| type fieldKey string | ||||
|  | ||||
| // FieldMap allows customization of the key names for default fields. | ||||
| type FieldMap map[fieldKey]string | ||||
|  | ||||
| func (f FieldMap) resolve(key fieldKey) string { | ||||
| 	if k, ok := f[key]; ok { | ||||
| 		return k | ||||
| 	} | ||||
|  | ||||
| 	return string(key) | ||||
| } | ||||
|  | ||||
| // JSONFormatter formats logs into parsable json | ||||
| type JSONFormatter struct { | ||||
| 	// TimestampFormat sets the format used for marshaling timestamps. | ||||
| 	TimestampFormat string | ||||
|  | ||||
| 	// DisableTimestamp allows disabling automatic timestamps in output | ||||
| 	DisableTimestamp bool | ||||
|  | ||||
| 	// DataKey allows users to put all the log entry parameters into a nested dictionary at a given key. | ||||
| 	DataKey string | ||||
|  | ||||
| 	// FieldMap allows users to customize the names of keys for default fields. | ||||
| 	// As an example: | ||||
| 	// formatter := &JSONFormatter{ | ||||
| 	//   	FieldMap: FieldMap{ | ||||
| 	// 		 FieldKeyTime:  "@timestamp", | ||||
| 	// 		 FieldKeyLevel: "@level", | ||||
| 	// 		 FieldKeyMsg:   "@message", | ||||
| 	// 		 FieldKeyFunc:  "@caller", | ||||
| 	//    }, | ||||
| 	// } | ||||
| 	FieldMap FieldMap | ||||
|  | ||||
| 	// PrettyPrint will indent all json logs | ||||
| 	PrettyPrint bool | ||||
| } | ||||
|  | ||||
| // Format renders a single log entry | ||||
| func (f *JSONFormatter) Format(entry *Entry) ([]byte, error) { | ||||
| 	data := make(Fields, len(entry.Data)+4) | ||||
| 	for k, v := range entry.Data { | ||||
| 		switch v := v.(type) { | ||||
| 		case error: | ||||
| 			// Otherwise errors are ignored by `encoding/json` | ||||
| 			// https://github.com/sirupsen/logrus/issues/137 | ||||
| 			data[k] = v.Error() | ||||
| 		default: | ||||
| 			data[k] = v | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if f.DataKey != "" { | ||||
| 		newData := make(Fields, 4) | ||||
| 		newData[f.DataKey] = data | ||||
| 		data = newData | ||||
| 	} | ||||
|  | ||||
| 	prefixFieldClashes(data, f.FieldMap, entry.HasCaller()) | ||||
|  | ||||
| 	timestampFormat := f.TimestampFormat | ||||
| 	if timestampFormat == "" { | ||||
| 		timestampFormat = defaultTimestampFormat | ||||
| 	} | ||||
|  | ||||
| 	if entry.err != "" { | ||||
| 		data[f.FieldMap.resolve(FieldKeyLogrusError)] = entry.err | ||||
| 	} | ||||
| 	if !f.DisableTimestamp { | ||||
| 		data[f.FieldMap.resolve(FieldKeyTime)] = entry.Time.Format(timestampFormat) | ||||
| 	} | ||||
| 	data[f.FieldMap.resolve(FieldKeyMsg)] = entry.Message | ||||
| 	data[f.FieldMap.resolve(FieldKeyLevel)] = entry.Level.String() | ||||
| 	if entry.HasCaller() { | ||||
| 		data[f.FieldMap.resolve(FieldKeyFunc)] = entry.Caller.Function | ||||
| 		data[f.FieldMap.resolve(FieldKeyFile)] = fmt.Sprintf("%s:%d", entry.Caller.File, entry.Caller.Line) | ||||
| 	} | ||||
|  | ||||
| 	var b *bytes.Buffer | ||||
| 	if entry.Buffer != nil { | ||||
| 		b = entry.Buffer | ||||
| 	} else { | ||||
| 		b = &bytes.Buffer{} | ||||
| 	} | ||||
|  | ||||
| 	encoder := json.NewEncoder(b) | ||||
| 	if f.PrettyPrint { | ||||
| 		encoder.SetIndent("", "  ") | ||||
| 	} | ||||
| 	if err := encoder.Encode(data); err != nil { | ||||
| 		return nil, fmt.Errorf("Failed to marshal fields to JSON, %v", err) | ||||
| 	} | ||||
|  | ||||
| 	return b.Bytes(), nil | ||||
| } | ||||
							
								
								
									
										346
									
								
								vendor/github.com/sirupsen/logrus/json_formatter_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										346
									
								
								vendor/github.com/sirupsen/logrus/json_formatter_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,346 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"runtime" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| func TestErrorNotLost(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("error", errors.New("wild walrus"))) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["error"] != "wild walrus" { | ||||
| 		t.Fatal("Error field not set") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorNotLostOnFieldNotNamedError(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("omg", errors.New("wild walrus"))) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["omg"] != "wild walrus" { | ||||
| 		t.Fatal("Error field not set") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldClashWithTime(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("time", "right now!")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["fields.time"] != "right now!" { | ||||
| 		t.Fatal("fields.time not set to original time field") | ||||
| 	} | ||||
|  | ||||
| 	if entry["time"] != "0001-01-01T00:00:00Z" { | ||||
| 		t.Fatal("time field not set to current time, was: ", entry["time"]) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldClashWithMsg(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("msg", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["fields.msg"] != "something" { | ||||
| 		t.Fatal("fields.msg not set to original msg field") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldClashWithLevel(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("level", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["fields.level"] != "something" { | ||||
| 		t.Fatal("fields.level not set to original level field") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldClashWithRemappedFields(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{ | ||||
| 		FieldMap: FieldMap{ | ||||
| 			FieldKeyTime:  "@timestamp", | ||||
| 			FieldKeyLevel: "@level", | ||||
| 			FieldKeyMsg:   "@message", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	b, err := formatter.Format(WithFields(Fields{ | ||||
| 		"@timestamp": "@timestamp", | ||||
| 		"@level":     "@level", | ||||
| 		"@message":   "@message", | ||||
| 		"timestamp":  "timestamp", | ||||
| 		"level":      "level", | ||||
| 		"msg":        "msg", | ||||
| 	})) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	for _, field := range []string{"timestamp", "level", "msg"} { | ||||
| 		if entry[field] != field { | ||||
| 			t.Errorf("Expected field %v to be untouched; got %v", field, entry[field]) | ||||
| 		} | ||||
|  | ||||
| 		remappedKey := fmt.Sprintf("fields.%s", field) | ||||
| 		if remapped, ok := entry[remappedKey]; ok { | ||||
| 			t.Errorf("Expected %s to be empty; got %v", remappedKey, remapped) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, field := range []string{"@timestamp", "@level", "@message"} { | ||||
| 		if entry[field] == field { | ||||
| 			t.Errorf("Expected field %v to be mapped to an Entry value", field) | ||||
| 		} | ||||
|  | ||||
| 		remappedKey := fmt.Sprintf("fields.%s", field) | ||||
| 		if remapped, ok := entry[remappedKey]; ok { | ||||
| 			if remapped != field { | ||||
| 				t.Errorf("Expected field %v to be copied to %s; got %v", field, remappedKey, remapped) | ||||
| 			} | ||||
| 		} else { | ||||
| 			t.Errorf("Expected field %v to be copied to %s; was absent", field, remappedKey) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldsInNestedDictionary(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{ | ||||
| 		DataKey: "args", | ||||
| 	} | ||||
|  | ||||
| 	logEntry := WithFields(Fields{ | ||||
| 		"level": "level", | ||||
| 		"test":  "test", | ||||
| 	}) | ||||
| 	logEntry.Level = InfoLevel | ||||
|  | ||||
| 	b, err := formatter.Format(logEntry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	args := entry["args"].(map[string]interface{}) | ||||
|  | ||||
| 	for _, field := range []string{"test", "level"} { | ||||
| 		if value, present := args[field]; !present || value != field { | ||||
| 			t.Errorf("Expected field %v to be present under 'args'; untouched", field) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	for _, field := range []string{"test", "fields.level"} { | ||||
| 		if _, present := entry[field]; present { | ||||
| 			t.Errorf("Expected field %v not to be present at top level", field) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	// with nested object, "level" shouldn't clash | ||||
| 	if entry["level"] != "info" { | ||||
| 		t.Errorf("Expected 'level' field to contain 'info'") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEntryEndsWithNewline(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("level", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if b[len(b)-1] != '\n' { | ||||
| 		t.Fatal("Expected JSON log entry to end with a newline") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONMessageKey(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{ | ||||
| 		FieldMap: FieldMap{ | ||||
| 			FieldKeyMsg: "message", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	b, err := formatter.Format(&Entry{Message: "oh hai"}) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
| 	s := string(b) | ||||
| 	if !(strings.Contains(s, "message") && strings.Contains(s, "oh hai")) { | ||||
| 		t.Fatal("Expected JSON to format message key") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONLevelKey(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{ | ||||
| 		FieldMap: FieldMap{ | ||||
| 			FieldKeyLevel: "somelevel", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("level", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
| 	s := string(b) | ||||
| 	if !strings.Contains(s, "somelevel") { | ||||
| 		t.Fatal("Expected JSON to format level key") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONTimeKey(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{ | ||||
| 		FieldMap: FieldMap{ | ||||
| 			FieldKeyTime: "timeywimey", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("level", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
| 	s := string(b) | ||||
| 	if !strings.Contains(s, "timeywimey") { | ||||
| 		t.Fatal("Expected JSON to format time key") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldDoesNotClashWithCaller(t *testing.T) { | ||||
| 	SetReportCaller(false) | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("func", "howdy pardner")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["func"] != "howdy pardner" { | ||||
| 		t.Fatal("func field replaced when ReportCaller=false") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFieldClashWithCaller(t *testing.T) { | ||||
| 	SetReportCaller(true) | ||||
| 	formatter := &JSONFormatter{} | ||||
| 	e := WithField("func", "howdy pardner") | ||||
| 	e.Caller = &runtime.Frame{Function: "somefunc"} | ||||
| 	b, err := formatter.Format(e) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	entry := make(map[string]interface{}) | ||||
| 	err = json.Unmarshal(b, &entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to unmarshal formatted entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	if entry["fields.func"] != "howdy pardner" { | ||||
| 		t.Fatalf("fields.func not set to original func field when ReportCaller=true (got '%s')", | ||||
| 			entry["fields.func"]) | ||||
| 	} | ||||
|  | ||||
| 	if entry["func"] != "somefunc" { | ||||
| 		t.Fatalf("func not set as expected when ReportCaller=true (got '%s')", | ||||
| 			entry["func"]) | ||||
| 	} | ||||
|  | ||||
| 	SetReportCaller(false) // return to default value | ||||
| } | ||||
|  | ||||
| func TestJSONDisableTimestamp(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{ | ||||
| 		DisableTimestamp: true, | ||||
| 	} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("level", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
| 	s := string(b) | ||||
| 	if strings.Contains(s, FieldKeyTime) { | ||||
| 		t.Error("Did not prevent timestamp", s) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEnableTimestamp(t *testing.T) { | ||||
| 	formatter := &JSONFormatter{} | ||||
|  | ||||
| 	b, err := formatter.Format(WithField("level", "something")) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
| 	s := string(b) | ||||
| 	if !strings.Contains(s, FieldKeyTime) { | ||||
| 		t.Error("Timestamp not present", s) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										415
									
								
								vendor/github.com/sirupsen/logrus/logger.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										415
									
								
								vendor/github.com/sirupsen/logrus/logger.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,415 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"sync" | ||||
| 	"sync/atomic" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| type Logger struct { | ||||
| 	// The logs are `io.Copy`'d to this in a mutex. It's common to set this to a | ||||
| 	// file, or leave it default which is `os.Stderr`. You can also set this to | ||||
| 	// something more adventurous, such as logging to Kafka. | ||||
| 	Out io.Writer | ||||
| 	// Hooks for the logger instance. These allow firing events based on logging | ||||
| 	// levels and log entries. For example, to send errors to an error tracking | ||||
| 	// service, log to StatsD or dump the core on fatal errors. | ||||
| 	Hooks LevelHooks | ||||
| 	// All log entries pass through the formatter before logged to Out. The | ||||
| 	// included formatters are `TextFormatter` and `JSONFormatter` for which | ||||
| 	// TextFormatter is the default. In development (when a TTY is attached) it | ||||
| 	// logs with colors, but to a file it wouldn't. You can easily implement your | ||||
| 	// own that implements the `Formatter` interface, see the `README` or included | ||||
| 	// formatters for examples. | ||||
| 	Formatter Formatter | ||||
|  | ||||
| 	// Flag for whether to log caller info (off by default) | ||||
| 	ReportCaller bool | ||||
|  | ||||
| 	// The logging level the logger should log at. This is typically (and defaults | ||||
| 	// to) `logrus.Info`, which allows Info(), Warn(), Error() and Fatal() to be | ||||
| 	// logged. | ||||
| 	Level Level | ||||
| 	// Used to sync writing to the log. Locking is enabled by Default | ||||
| 	mu MutexWrap | ||||
| 	// Reusable empty entry | ||||
| 	entryPool sync.Pool | ||||
| 	// Function to exit the application, defaults to `os.Exit()` | ||||
| 	ExitFunc exitFunc | ||||
| } | ||||
|  | ||||
| type exitFunc func(int) | ||||
|  | ||||
| type MutexWrap struct { | ||||
| 	lock     sync.Mutex | ||||
| 	disabled bool | ||||
| } | ||||
|  | ||||
| func (mw *MutexWrap) Lock() { | ||||
| 	if !mw.disabled { | ||||
| 		mw.lock.Lock() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (mw *MutexWrap) Unlock() { | ||||
| 	if !mw.disabled { | ||||
| 		mw.lock.Unlock() | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (mw *MutexWrap) Disable() { | ||||
| 	mw.disabled = true | ||||
| } | ||||
|  | ||||
| // Creates a new logger. Configuration should be set by changing `Formatter`, | ||||
| // `Out` and `Hooks` directly on the default logger instance. You can also just | ||||
| // instantiate your own: | ||||
| // | ||||
| //    var log = &Logger{ | ||||
| //      Out: os.Stderr, | ||||
| //      Formatter: new(JSONFormatter), | ||||
| //      Hooks: make(LevelHooks), | ||||
| //      Level: logrus.DebugLevel, | ||||
| //    } | ||||
| // | ||||
| // It's recommended to make this a global instance called `log`. | ||||
| func New() *Logger { | ||||
| 	return &Logger{ | ||||
| 		Out:          os.Stderr, | ||||
| 		Formatter:    new(TextFormatter), | ||||
| 		Hooks:        make(LevelHooks), | ||||
| 		Level:        InfoLevel, | ||||
| 		ExitFunc:     os.Exit, | ||||
| 		ReportCaller: false, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) newEntry() *Entry { | ||||
| 	entry, ok := logger.entryPool.Get().(*Entry) | ||||
| 	if ok { | ||||
| 		return entry | ||||
| 	} | ||||
| 	return NewEntry(logger) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) releaseEntry(entry *Entry) { | ||||
| 	entry.Data = map[string]interface{}{} | ||||
| 	logger.entryPool.Put(entry) | ||||
| } | ||||
|  | ||||
| // Adds a field to the log entry, note that it doesn't log until you call | ||||
| // Debug, Print, Info, Warn, Error, Fatal or Panic. It only creates a log entry. | ||||
| // If you want multiple fields, use `WithFields`. | ||||
| func (logger *Logger) WithField(key string, value interface{}) *Entry { | ||||
| 	entry := logger.newEntry() | ||||
| 	defer logger.releaseEntry(entry) | ||||
| 	return entry.WithField(key, value) | ||||
| } | ||||
|  | ||||
| // Adds a struct of fields to the log entry. All it does is call `WithField` for | ||||
| // each `Field`. | ||||
| func (logger *Logger) WithFields(fields Fields) *Entry { | ||||
| 	entry := logger.newEntry() | ||||
| 	defer logger.releaseEntry(entry) | ||||
| 	return entry.WithFields(fields) | ||||
| } | ||||
|  | ||||
| // Add an error as single field to the log entry.  All it does is call | ||||
| // `WithError` for the given `error`. | ||||
| func (logger *Logger) WithError(err error) *Entry { | ||||
| 	entry := logger.newEntry() | ||||
| 	defer logger.releaseEntry(entry) | ||||
| 	return entry.WithError(err) | ||||
| } | ||||
|  | ||||
| // Overrides the time of the log entry. | ||||
| func (logger *Logger) WithTime(t time.Time) *Entry { | ||||
| 	entry := logger.newEntry() | ||||
| 	defer logger.releaseEntry(entry) | ||||
| 	return entry.WithTime(t) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Tracef(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(TraceLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Tracef(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Debugf(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(DebugLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Debugf(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Infof(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(InfoLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Infof(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Printf(format string, args ...interface{}) { | ||||
| 	entry := logger.newEntry() | ||||
| 	entry.Printf(format, args...) | ||||
| 	logger.releaseEntry(entry) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Warnf(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Warnf(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Warningf(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Warnf(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Errorf(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(ErrorLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Errorf(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Fatalf(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(FatalLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Fatalf(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| 	logger.Exit(1) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Panicf(format string, args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(PanicLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Panicf(format, args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Trace(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(TraceLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Trace(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Debug(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(DebugLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Debug(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Info(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(InfoLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Info(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Print(args ...interface{}) { | ||||
| 	entry := logger.newEntry() | ||||
| 	entry.Info(args...) | ||||
| 	logger.releaseEntry(entry) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Warn(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Warn(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Warning(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Warn(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Error(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(ErrorLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Error(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Fatal(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(FatalLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Fatal(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| 	logger.Exit(1) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Panic(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(PanicLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Panic(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Traceln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(TraceLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Traceln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Debugln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(DebugLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Debugln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Infoln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(InfoLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Infoln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Println(args ...interface{}) { | ||||
| 	entry := logger.newEntry() | ||||
| 	entry.Println(args...) | ||||
| 	logger.releaseEntry(entry) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Warnln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Warnln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Warningln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(WarnLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Warnln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Errorln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(ErrorLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Errorln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Fatalln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(FatalLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Fatalln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| 	logger.Exit(1) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Panicln(args ...interface{}) { | ||||
| 	if logger.IsLevelEnabled(PanicLevel) { | ||||
| 		entry := logger.newEntry() | ||||
| 		entry.Panicln(args...) | ||||
| 		logger.releaseEntry(entry) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (logger *Logger) Exit(code int) { | ||||
| 	runHandlers() | ||||
| 	if logger.ExitFunc == nil { | ||||
| 		logger.ExitFunc = os.Exit | ||||
| 	} | ||||
| 	logger.ExitFunc(code) | ||||
| } | ||||
|  | ||||
| //When file is opened with appending mode, it's safe to | ||||
| //write concurrently to a file (within 4k message on Linux). | ||||
| //In these cases user can choose to disable the lock. | ||||
| func (logger *Logger) SetNoLock() { | ||||
| 	logger.mu.Disable() | ||||
| } | ||||
|  | ||||
| func (logger *Logger) level() Level { | ||||
| 	return Level(atomic.LoadUint32((*uint32)(&logger.Level))) | ||||
| } | ||||
|  | ||||
| // SetLevel sets the logger level. | ||||
| func (logger *Logger) SetLevel(level Level) { | ||||
| 	atomic.StoreUint32((*uint32)(&logger.Level), uint32(level)) | ||||
| } | ||||
|  | ||||
| // GetLevel returns the logger level. | ||||
| func (logger *Logger) GetLevel() Level { | ||||
| 	return logger.level() | ||||
| } | ||||
|  | ||||
| // AddHook adds a hook to the logger hooks. | ||||
| func (logger *Logger) AddHook(hook Hook) { | ||||
| 	logger.mu.Lock() | ||||
| 	defer logger.mu.Unlock() | ||||
| 	logger.Hooks.Add(hook) | ||||
| } | ||||
|  | ||||
| // IsLevelEnabled checks if the log level of the logger is greater than the level param | ||||
| func (logger *Logger) IsLevelEnabled(level Level) bool { | ||||
| 	return logger.level() >= level | ||||
| } | ||||
|  | ||||
| // SetFormatter sets the logger formatter. | ||||
| func (logger *Logger) SetFormatter(formatter Formatter) { | ||||
| 	logger.mu.Lock() | ||||
| 	defer logger.mu.Unlock() | ||||
| 	logger.Formatter = formatter | ||||
| } | ||||
|  | ||||
| // SetOutput sets the logger output. | ||||
| func (logger *Logger) SetOutput(output io.Writer) { | ||||
| 	logger.mu.Lock() | ||||
| 	defer logger.mu.Unlock() | ||||
| 	logger.Out = output | ||||
| } | ||||
|  | ||||
| func (logger *Logger) SetReportCaller(reportCaller bool) { | ||||
| 	logger.mu.Lock() | ||||
| 	defer logger.mu.Unlock() | ||||
| 	logger.ReportCaller = reportCaller | ||||
| } | ||||
|  | ||||
| // ReplaceHooks replaces the logger hooks and returns the old ones | ||||
| func (logger *Logger) ReplaceHooks(hooks LevelHooks) LevelHooks { | ||||
| 	logger.mu.Lock() | ||||
| 	oldHooks := logger.Hooks | ||||
| 	logger.Hooks = hooks | ||||
| 	logger.mu.Unlock() | ||||
| 	return oldHooks | ||||
| } | ||||
							
								
								
									
										85
									
								
								vendor/github.com/sirupsen/logrus/logger_bench_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								vendor/github.com/sirupsen/logrus/logger_bench_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| // smallFields is a small size data set for benchmarking | ||||
| var loggerFields = Fields{ | ||||
| 	"foo":   "bar", | ||||
| 	"baz":   "qux", | ||||
| 	"one":   "two", | ||||
| 	"three": "four", | ||||
| } | ||||
|  | ||||
| func BenchmarkDummyLogger(b *testing.B) { | ||||
| 	nullf, err := os.OpenFile("/dev/null", os.O_WRONLY, 0666) | ||||
| 	if err != nil { | ||||
| 		b.Fatalf("%v", err) | ||||
| 	} | ||||
| 	defer nullf.Close() | ||||
| 	doLoggerBenchmark(b, nullf, &TextFormatter{DisableColors: true}, smallFields) | ||||
| } | ||||
|  | ||||
| func BenchmarkDummyLoggerNoLock(b *testing.B) { | ||||
| 	nullf, err := os.OpenFile("/dev/null", os.O_WRONLY|os.O_APPEND, 0666) | ||||
| 	if err != nil { | ||||
| 		b.Fatalf("%v", err) | ||||
| 	} | ||||
| 	defer nullf.Close() | ||||
| 	doLoggerBenchmarkNoLock(b, nullf, &TextFormatter{DisableColors: true}, smallFields) | ||||
| } | ||||
|  | ||||
| func doLoggerBenchmark(b *testing.B, out *os.File, formatter Formatter, fields Fields) { | ||||
| 	logger := Logger{ | ||||
| 		Out:       out, | ||||
| 		Level:     InfoLevel, | ||||
| 		Formatter: formatter, | ||||
| 	} | ||||
| 	entry := logger.WithFields(fields) | ||||
| 	b.RunParallel(func(pb *testing.PB) { | ||||
| 		for pb.Next() { | ||||
| 			entry.Info("aaa") | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func doLoggerBenchmarkNoLock(b *testing.B, out *os.File, formatter Formatter, fields Fields) { | ||||
| 	logger := Logger{ | ||||
| 		Out:       out, | ||||
| 		Level:     InfoLevel, | ||||
| 		Formatter: formatter, | ||||
| 	} | ||||
| 	logger.SetNoLock() | ||||
| 	entry := logger.WithFields(fields) | ||||
| 	b.RunParallel(func(pb *testing.PB) { | ||||
| 		for pb.Next() { | ||||
| 			entry.Info("aaa") | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func BenchmarkLoggerJSONFormatter(b *testing.B) { | ||||
| 	doLoggerBenchmarkWithFormatter(b, &JSONFormatter{}) | ||||
| } | ||||
|  | ||||
| func BenchmarkLoggerTextFormatter(b *testing.B) { | ||||
| 	doLoggerBenchmarkWithFormatter(b, &TextFormatter{}) | ||||
| } | ||||
|  | ||||
| func doLoggerBenchmarkWithFormatter(b *testing.B, f Formatter) { | ||||
| 	b.SetParallelism(100) | ||||
| 	log := New() | ||||
| 	log.Formatter = f | ||||
| 	log.Out = ioutil.Discard | ||||
| 	b.RunParallel(func(pb *testing.PB) { | ||||
| 		for pb.Next() { | ||||
| 			log. | ||||
| 				WithField("foo1", "bar1"). | ||||
| 				WithField("foo2", "bar2"). | ||||
| 				Info("this is a dummy log") | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
							
								
								
									
										42
									
								
								vendor/github.com/sirupsen/logrus/logger_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								vendor/github.com/sirupsen/logrus/logger_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"fmt" | ||||
| 	"testing" | ||||
|  | ||||
| 	"github.com/stretchr/testify/require" | ||||
| ) | ||||
|  | ||||
| func TestFieldValueError(t *testing.T) { | ||||
| 	buf := &bytes.Buffer{} | ||||
| 	l := &Logger{ | ||||
| 		Out:       buf, | ||||
| 		Formatter: new(JSONFormatter), | ||||
| 		Hooks:     make(LevelHooks), | ||||
| 		Level:     DebugLevel, | ||||
| 	} | ||||
| 	l.WithField("func", func() {}).Info("test") | ||||
| 	fmt.Println(buf.String()) | ||||
| 	var data map[string]interface{} | ||||
| 	json.Unmarshal(buf.Bytes(), &data) | ||||
| 	_, ok := data[FieldKeyLogrusError] | ||||
| 	require.True(t, ok) | ||||
| } | ||||
|  | ||||
| func TestNoFieldValueError(t *testing.T) { | ||||
| 	buf := &bytes.Buffer{} | ||||
| 	l := &Logger{ | ||||
| 		Out:       buf, | ||||
| 		Formatter: new(JSONFormatter), | ||||
| 		Hooks:     make(LevelHooks), | ||||
| 		Level:     DebugLevel, | ||||
| 	} | ||||
| 	l.WithField("str", "str").Info("test") | ||||
| 	fmt.Println(buf.String()) | ||||
| 	var data map[string]interface{} | ||||
| 	json.Unmarshal(buf.Bytes(), &data) | ||||
| 	_, ok := data[FieldKeyLogrusError] | ||||
| 	require.False(t, ok) | ||||
| } | ||||
							
								
								
									
										178
									
								
								vendor/github.com/sirupsen/logrus/logrus.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								vendor/github.com/sirupsen/logrus/logrus.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,178 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"log" | ||||
| 	"strings" | ||||
| ) | ||||
|  | ||||
| // Fields type, used to pass to `WithFields`. | ||||
| type Fields map[string]interface{} | ||||
|  | ||||
| // Level type | ||||
| type Level uint32 | ||||
|  | ||||
| // Convert the Level to a string. E.g. PanicLevel becomes "panic". | ||||
| func (level Level) String() string { | ||||
| 	switch level { | ||||
| 	case TraceLevel: | ||||
| 		return "trace" | ||||
| 	case DebugLevel: | ||||
| 		return "debug" | ||||
| 	case InfoLevel: | ||||
| 		return "info" | ||||
| 	case WarnLevel: | ||||
| 		return "warning" | ||||
| 	case ErrorLevel: | ||||
| 		return "error" | ||||
| 	case FatalLevel: | ||||
| 		return "fatal" | ||||
| 	case PanicLevel: | ||||
| 		return "panic" | ||||
| 	} | ||||
|  | ||||
| 	return "unknown" | ||||
| } | ||||
|  | ||||
| // ParseLevel takes a string level and returns the Logrus log level constant. | ||||
| func ParseLevel(lvl string) (Level, error) { | ||||
| 	switch strings.ToLower(lvl) { | ||||
| 	case "panic": | ||||
| 		return PanicLevel, nil | ||||
| 	case "fatal": | ||||
| 		return FatalLevel, nil | ||||
| 	case "error": | ||||
| 		return ErrorLevel, nil | ||||
| 	case "warn", "warning": | ||||
| 		return WarnLevel, nil | ||||
| 	case "info": | ||||
| 		return InfoLevel, nil | ||||
| 	case "debug": | ||||
| 		return DebugLevel, nil | ||||
| 	case "trace": | ||||
| 		return TraceLevel, nil | ||||
| 	} | ||||
|  | ||||
| 	var l Level | ||||
| 	return l, fmt.Errorf("not a valid logrus Level: %q", lvl) | ||||
| } | ||||
|  | ||||
| // UnmarshalText implements encoding.TextUnmarshaler. | ||||
| func (level *Level) UnmarshalText(text []byte) error { | ||||
| 	l, err := ParseLevel(string(text)) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	*level = Level(l) | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // A constant exposing all logging levels | ||||
| var AllLevels = []Level{ | ||||
| 	PanicLevel, | ||||
| 	FatalLevel, | ||||
| 	ErrorLevel, | ||||
| 	WarnLevel, | ||||
| 	InfoLevel, | ||||
| 	DebugLevel, | ||||
| 	TraceLevel, | ||||
| } | ||||
|  | ||||
| // These are the different logging levels. You can set the logging level to log | ||||
| // on your instance of logger, obtained with `logrus.New()`. | ||||
| const ( | ||||
| 	// PanicLevel level, highest level of severity. Logs and then calls panic with the | ||||
| 	// message passed to Debug, Info, ... | ||||
| 	PanicLevel Level = iota | ||||
| 	// FatalLevel level. Logs and then calls `logger.Exit(1)`. It will exit even if the | ||||
| 	// logging level is set to Panic. | ||||
| 	FatalLevel | ||||
| 	// ErrorLevel level. Logs. Used for errors that should definitely be noted. | ||||
| 	// Commonly used for hooks to send errors to an error tracking service. | ||||
| 	ErrorLevel | ||||
| 	// WarnLevel level. Non-critical entries that deserve eyes. | ||||
| 	WarnLevel | ||||
| 	// InfoLevel level. General operational entries about what's going on inside the | ||||
| 	// application. | ||||
| 	InfoLevel | ||||
| 	// DebugLevel level. Usually only enabled when debugging. Very verbose logging. | ||||
| 	DebugLevel | ||||
| 	// TraceLevel level. Designates finer-grained informational events than the Debug. | ||||
| 	TraceLevel | ||||
| ) | ||||
|  | ||||
| // Won't compile if StdLogger can't be realized by a log.Logger | ||||
| var ( | ||||
| 	_ StdLogger = &log.Logger{} | ||||
| 	_ StdLogger = &Entry{} | ||||
| 	_ StdLogger = &Logger{} | ||||
| ) | ||||
|  | ||||
| // StdLogger is what your logrus-enabled library should take, that way | ||||
| // it'll accept a stdlib logger and a logrus logger. There's no standard | ||||
| // interface, this is the closest we get, unfortunately. | ||||
| type StdLogger interface { | ||||
| 	Print(...interface{}) | ||||
| 	Printf(string, ...interface{}) | ||||
| 	Println(...interface{}) | ||||
|  | ||||
| 	Fatal(...interface{}) | ||||
| 	Fatalf(string, ...interface{}) | ||||
| 	Fatalln(...interface{}) | ||||
|  | ||||
| 	Panic(...interface{}) | ||||
| 	Panicf(string, ...interface{}) | ||||
| 	Panicln(...interface{}) | ||||
| } | ||||
|  | ||||
| // The FieldLogger interface generalizes the Entry and Logger types | ||||
| type FieldLogger interface { | ||||
| 	WithField(key string, value interface{}) *Entry | ||||
| 	WithFields(fields Fields) *Entry | ||||
| 	WithError(err error) *Entry | ||||
|  | ||||
| 	Debugf(format string, args ...interface{}) | ||||
| 	Infof(format string, args ...interface{}) | ||||
| 	Printf(format string, args ...interface{}) | ||||
| 	Warnf(format string, args ...interface{}) | ||||
| 	Warningf(format string, args ...interface{}) | ||||
| 	Errorf(format string, args ...interface{}) | ||||
| 	Fatalf(format string, args ...interface{}) | ||||
| 	Panicf(format string, args ...interface{}) | ||||
|  | ||||
| 	Debug(args ...interface{}) | ||||
| 	Info(args ...interface{}) | ||||
| 	Print(args ...interface{}) | ||||
| 	Warn(args ...interface{}) | ||||
| 	Warning(args ...interface{}) | ||||
| 	Error(args ...interface{}) | ||||
| 	Fatal(args ...interface{}) | ||||
| 	Panic(args ...interface{}) | ||||
|  | ||||
| 	Debugln(args ...interface{}) | ||||
| 	Infoln(args ...interface{}) | ||||
| 	Println(args ...interface{}) | ||||
| 	Warnln(args ...interface{}) | ||||
| 	Warningln(args ...interface{}) | ||||
| 	Errorln(args ...interface{}) | ||||
| 	Fatalln(args ...interface{}) | ||||
| 	Panicln(args ...interface{}) | ||||
|  | ||||
| 	// IsDebugEnabled() bool | ||||
| 	// IsInfoEnabled() bool | ||||
| 	// IsWarnEnabled() bool | ||||
| 	// IsErrorEnabled() bool | ||||
| 	// IsFatalEnabled() bool | ||||
| 	// IsPanicEnabled() bool | ||||
| } | ||||
|  | ||||
| // Ext1FieldLogger (the first extension to FieldLogger) is superfluous, it is | ||||
| // here for consistancy. Do not use. Use Logger or Entry instead. | ||||
| type Ext1FieldLogger interface { | ||||
| 	FieldLogger | ||||
| 	Tracef(format string, args ...interface{}) | ||||
| 	Trace(args ...interface{}) | ||||
| 	Traceln(args ...interface{}) | ||||
| } | ||||
							
								
								
									
										716
									
								
								vendor/github.com/sirupsen/logrus/logrus_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										716
									
								
								vendor/github.com/sirupsen/logrus/logrus_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,716 @@ | ||||
| package logrus_test | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"encoding/json" | ||||
| 	"io/ioutil" | ||||
| 	"os" | ||||
| 	"path/filepath" | ||||
| 	"sync" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"github.com/stretchr/testify/require" | ||||
|  | ||||
| 	. "github.com/sirupsen/logrus" | ||||
| 	. "github.com/sirupsen/logrus/internal/testutils" | ||||
| ) | ||||
|  | ||||
| // TestReportCaller verifies that when ReportCaller is set, the 'func' field | ||||
| // is added, and when it is unset it is not set or modified | ||||
| // Verify that functions within the Logrus package aren't considered when | ||||
| // discovering the caller. | ||||
| func TestReportCallerWhenConfigured(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.ReportCaller = false | ||||
| 		log.Print("testNoCaller") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "testNoCaller", fields["msg"]) | ||||
| 		assert.Equal(t, "info", fields["level"]) | ||||
| 		assert.Equal(t, nil, fields["func"]) | ||||
| 	}) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.ReportCaller = true | ||||
| 		log.Print("testWithCaller") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "testWithCaller", fields["msg"]) | ||||
| 		assert.Equal(t, "info", fields["level"]) | ||||
| 		assert.Equal(t, | ||||
| 			"github.com/sirupsen/logrus_test.TestReportCallerWhenConfigured.func3", fields["func"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func logSomething(t *testing.T, message string) Fields { | ||||
| 	var buffer bytes.Buffer | ||||
| 	var fields Fields | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = new(JSONFormatter) | ||||
| 	logger.ReportCaller = true | ||||
|  | ||||
| 	entry := logger.WithFields(Fields{ | ||||
| 		"foo": "bar", | ||||
| 	}) | ||||
|  | ||||
| 	entry.Info(message) | ||||
|  | ||||
| 	err := json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	assert.Nil(t, err) | ||||
|  | ||||
| 	return fields | ||||
| } | ||||
|  | ||||
| // TestReportCallerHelperDirect - verify reference when logging from a regular function | ||||
| func TestReportCallerHelperDirect(t *testing.T) { | ||||
| 	fields := logSomething(t, "direct") | ||||
|  | ||||
| 	assert.Equal(t, "direct", fields["msg"]) | ||||
| 	assert.Equal(t, "info", fields["level"]) | ||||
| 	assert.Regexp(t, "github.com/.*/logrus_test.logSomething", fields["func"]) | ||||
| } | ||||
|  | ||||
| // TestReportCallerHelperDirect - verify reference when logging from a function called via pointer | ||||
| func TestReportCallerHelperViaPointer(t *testing.T) { | ||||
| 	fptr := logSomething | ||||
| 	fields := fptr(t, "via pointer") | ||||
|  | ||||
| 	assert.Equal(t, "via pointer", fields["msg"]) | ||||
| 	assert.Equal(t, "info", fields["level"]) | ||||
| 	assert.Regexp(t, "github.com/.*/logrus_test.logSomething", fields["func"]) | ||||
| } | ||||
|  | ||||
| func TestPrint(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Print("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test", fields["msg"]) | ||||
| 		assert.Equal(t, "info", fields["level"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfo(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Info("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test", fields["msg"]) | ||||
| 		assert.Equal(t, "info", fields["level"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestWarn(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Warn("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test", fields["msg"]) | ||||
| 		assert.Equal(t, "warning", fields["level"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfolnShouldAddSpacesBetweenStrings(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Infoln("test", "test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test test", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfolnShouldAddSpacesBetweenStringAndNonstring(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Infoln("test", 10) | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test 10", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfolnShouldAddSpacesBetweenTwoNonStrings(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Infoln(10, 10) | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "10 10", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfoShouldAddSpacesBetweenTwoNonStrings(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Infoln(10, 10) | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "10 10", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfoShouldNotAddSpacesBetweenStringAndNonstring(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Info("test", 10) | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test10", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestInfoShouldNotAddSpacesBetweenStrings(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.Info("test", "test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "testtest", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestWithFieldsShouldAllowAssignments(t *testing.T) { | ||||
| 	var buffer bytes.Buffer | ||||
| 	var fields Fields | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = new(JSONFormatter) | ||||
|  | ||||
| 	localLog := logger.WithFields(Fields{ | ||||
| 		"key1": "value1", | ||||
| 	}) | ||||
|  | ||||
| 	localLog.WithField("key2", "value2").Info("test") | ||||
| 	err := json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	assert.Nil(t, err) | ||||
|  | ||||
| 	assert.Equal(t, "value2", fields["key2"]) | ||||
| 	assert.Equal(t, "value1", fields["key1"]) | ||||
|  | ||||
| 	buffer = bytes.Buffer{} | ||||
| 	fields = Fields{} | ||||
| 	localLog.Info("test") | ||||
| 	err = json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	assert.Nil(t, err) | ||||
|  | ||||
| 	_, ok := fields["key2"] | ||||
| 	assert.Equal(t, false, ok) | ||||
| 	assert.Equal(t, "value1", fields["key1"]) | ||||
| } | ||||
|  | ||||
| func TestUserSuppliedFieldDoesNotOverwriteDefaults(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithField("msg", "hello").Info("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test", fields["msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestUserSuppliedMsgFieldHasPrefix(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithField("msg", "hello").Info("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "test", fields["msg"]) | ||||
| 		assert.Equal(t, "hello", fields["fields.msg"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestUserSuppliedTimeFieldHasPrefix(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithField("time", "hello").Info("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "hello", fields["fields.time"]) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestUserSuppliedLevelFieldHasPrefix(t *testing.T) { | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithField("level", 1).Info("test") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, "info", fields["level"]) | ||||
| 		assert.Equal(t, 1.0, fields["fields.level"]) // JSON has floats only | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestDefaultFieldsAreNotPrefixed(t *testing.T) { | ||||
| 	LogAndAssertText(t, func(log *Logger) { | ||||
| 		ll := log.WithField("herp", "derp") | ||||
| 		ll.Info("hello") | ||||
| 		ll.Info("bye") | ||||
| 	}, func(fields map[string]string) { | ||||
| 		for _, fieldName := range []string{"fields.level", "fields.time", "fields.msg"} { | ||||
| 			if _, ok := fields[fieldName]; ok { | ||||
| 				t.Fatalf("should not have prefixed %q: %v", fieldName, fields) | ||||
| 			} | ||||
| 		} | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestWithTimeShouldOverrideTime(t *testing.T) { | ||||
| 	now := time.Now().Add(24 * time.Hour) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithTime(now).Info("foobar") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, fields["time"], now.Format(time.RFC3339)) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestWithTimeShouldNotOverrideFields(t *testing.T) { | ||||
| 	now := time.Now().Add(24 * time.Hour) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithField("herp", "derp").WithTime(now).Info("blah") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, fields["time"], now.Format(time.RFC3339)) | ||||
| 		assert.Equal(t, fields["herp"], "derp") | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestWithFieldShouldNotOverrideTime(t *testing.T) { | ||||
| 	now := time.Now().Add(24 * time.Hour) | ||||
|  | ||||
| 	LogAndAssertJSON(t, func(log *Logger) { | ||||
| 		log.WithTime(now).WithField("herp", "derp").Info("blah") | ||||
| 	}, func(fields Fields) { | ||||
| 		assert.Equal(t, fields["time"], now.Format(time.RFC3339)) | ||||
| 		assert.Equal(t, fields["herp"], "derp") | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestTimeOverrideMultipleLogs(t *testing.T) { | ||||
| 	var buffer bytes.Buffer | ||||
| 	var firstFields, secondFields Fields | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	formatter := new(JSONFormatter) | ||||
| 	formatter.TimestampFormat = time.StampMilli | ||||
| 	logger.Formatter = formatter | ||||
|  | ||||
| 	llog := logger.WithField("herp", "derp") | ||||
| 	llog.Info("foo") | ||||
|  | ||||
| 	err := json.Unmarshal(buffer.Bytes(), &firstFields) | ||||
| 	assert.NoError(t, err, "should have decoded first message") | ||||
|  | ||||
| 	buffer.Reset() | ||||
|  | ||||
| 	time.Sleep(10 * time.Millisecond) | ||||
| 	llog.Info("bar") | ||||
|  | ||||
| 	err = json.Unmarshal(buffer.Bytes(), &secondFields) | ||||
| 	assert.NoError(t, err, "should have decoded second message") | ||||
|  | ||||
| 	assert.NotEqual(t, firstFields["time"], secondFields["time"], "timestamps should not be equal") | ||||
| } | ||||
|  | ||||
| func TestDoubleLoggingDoesntPrefixPreviousFields(t *testing.T) { | ||||
|  | ||||
| 	var buffer bytes.Buffer | ||||
| 	var fields Fields | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = new(JSONFormatter) | ||||
|  | ||||
| 	llog := logger.WithField("context", "eating raw fish") | ||||
|  | ||||
| 	llog.Info("looks delicious") | ||||
|  | ||||
| 	err := json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	assert.NoError(t, err, "should have decoded first message") | ||||
| 	assert.Equal(t, len(fields), 4, "should only have msg/time/level/context fields") | ||||
| 	assert.Equal(t, fields["msg"], "looks delicious") | ||||
| 	assert.Equal(t, fields["context"], "eating raw fish") | ||||
|  | ||||
| 	buffer.Reset() | ||||
|  | ||||
| 	llog.Warn("omg it is!") | ||||
|  | ||||
| 	err = json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	assert.NoError(t, err, "should have decoded second message") | ||||
| 	assert.Equal(t, len(fields), 4, "should only have msg/time/level/context fields") | ||||
| 	assert.Equal(t, "omg it is!", fields["msg"]) | ||||
| 	assert.Equal(t, "eating raw fish", fields["context"]) | ||||
| 	assert.Nil(t, fields["fields.msg"], "should not have prefixed previous `msg` entry") | ||||
|  | ||||
| } | ||||
|  | ||||
| func TestNestedLoggingReportsCorrectCaller(t *testing.T) { | ||||
| 	var buffer bytes.Buffer | ||||
| 	var fields Fields | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = new(JSONFormatter) | ||||
| 	logger.ReportCaller = true | ||||
|  | ||||
| 	llog := logger.WithField("context", "eating raw fish") | ||||
|  | ||||
| 	llog.Info("looks delicious") | ||||
|  | ||||
| 	err := json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	require.NoError(t, err, "should have decoded first message") | ||||
| 	assert.Equal(t, 6, len(fields), "should have msg/time/level/func/context fields") | ||||
| 	assert.Equal(t, "looks delicious", fields["msg"]) | ||||
| 	assert.Equal(t, "eating raw fish", fields["context"]) | ||||
| 	assert.Equal(t, | ||||
| 		"github.com/sirupsen/logrus_test.TestNestedLoggingReportsCorrectCaller", fields["func"]) | ||||
| 	cwd, err := os.Getwd() | ||||
| 	require.NoError(t, err) | ||||
| 	assert.Equal(t, filepath.ToSlash(cwd+"/logrus_test.go:340"), filepath.ToSlash(fields["file"].(string))) | ||||
|  | ||||
| 	buffer.Reset() | ||||
|  | ||||
| 	logger.WithFields(Fields{ | ||||
| 		"Clyde": "Stubblefield", | ||||
| 	}).WithFields(Fields{ | ||||
| 		"Jab'o": "Starks", | ||||
| 	}).WithFields(Fields{ | ||||
| 		"uri": "https://www.youtube.com/watch?v=V5DTznu-9v0", | ||||
| 	}).WithFields(Fields{ | ||||
| 		"func": "y drummer", | ||||
| 	}).WithFields(Fields{ | ||||
| 		"James": "Brown", | ||||
| 	}).Print("The hardest workin' man in show business") | ||||
|  | ||||
| 	err = json.Unmarshal(buffer.Bytes(), &fields) | ||||
| 	assert.NoError(t, err, "should have decoded second message") | ||||
| 	assert.Equal(t, 11, len(fields), "should have all builtin fields plus foo,bar,baz,...") | ||||
| 	assert.Equal(t, "Stubblefield", fields["Clyde"]) | ||||
| 	assert.Equal(t, "Starks", fields["Jab'o"]) | ||||
| 	assert.Equal(t, "https://www.youtube.com/watch?v=V5DTznu-9v0", fields["uri"]) | ||||
| 	assert.Equal(t, "y drummer", fields["fields.func"]) | ||||
| 	assert.Equal(t, "Brown", fields["James"]) | ||||
| 	assert.Equal(t, "The hardest workin' man in show business", fields["msg"]) | ||||
| 	assert.Nil(t, fields["fields.msg"], "should not have prefixed previous `msg` entry") | ||||
| 	assert.Equal(t, | ||||
| 		"github.com/sirupsen/logrus_test.TestNestedLoggingReportsCorrectCaller", fields["func"]) | ||||
| 	require.NoError(t, err) | ||||
| 	assert.Equal(t, filepath.ToSlash(cwd+"/logrus_test.go:365"), filepath.ToSlash(fields["file"].(string))) | ||||
|  | ||||
| 	logger.ReportCaller = false // return to default value | ||||
| } | ||||
|  | ||||
| func logLoop(iterations int, reportCaller bool) { | ||||
| 	var buffer bytes.Buffer | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	logger.Formatter = new(JSONFormatter) | ||||
| 	logger.ReportCaller = reportCaller | ||||
|  | ||||
| 	for i := 0; i < iterations; i++ { | ||||
| 		logger.Infof("round %d of %d", i, iterations) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Assertions for upper bounds to reporting overhead | ||||
| func TestCallerReportingOverhead(t *testing.T) { | ||||
| 	iterations := 5000 | ||||
| 	before := time.Now() | ||||
| 	logLoop(iterations, false) | ||||
| 	during := time.Now() | ||||
| 	logLoop(iterations, true) | ||||
| 	after := time.Now() | ||||
|  | ||||
| 	elapsedNotReporting := during.Sub(before).Nanoseconds() | ||||
| 	elapsedReporting := after.Sub(during).Nanoseconds() | ||||
|  | ||||
| 	maxDelta := 1 * time.Second | ||||
| 	assert.WithinDuration(t, during, before, maxDelta, | ||||
| 		"%d log calls without caller name lookup takes less than %d second(s) (was %d nanoseconds)", | ||||
| 		iterations, maxDelta.Seconds(), elapsedNotReporting) | ||||
| 	assert.WithinDuration(t, after, during, maxDelta, | ||||
| 		"%d log calls without caller name lookup takes less than %d second(s) (was %d nanoseconds)", | ||||
| 		iterations, maxDelta.Seconds(), elapsedReporting) | ||||
| } | ||||
|  | ||||
| // benchmarks for both with and without caller-function reporting | ||||
| func BenchmarkWithoutCallerTracing(b *testing.B) { | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		logLoop(1000, false) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func BenchmarkWithCallerTracing(b *testing.B) { | ||||
| 	for i := 0; i < b.N; i++ { | ||||
| 		logLoop(1000, true) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestConvertLevelToString(t *testing.T) { | ||||
| 	assert.Equal(t, "trace", TraceLevel.String()) | ||||
| 	assert.Equal(t, "debug", DebugLevel.String()) | ||||
| 	assert.Equal(t, "info", InfoLevel.String()) | ||||
| 	assert.Equal(t, "warning", WarnLevel.String()) | ||||
| 	assert.Equal(t, "error", ErrorLevel.String()) | ||||
| 	assert.Equal(t, "fatal", FatalLevel.String()) | ||||
| 	assert.Equal(t, "panic", PanicLevel.String()) | ||||
| } | ||||
|  | ||||
| func TestParseLevel(t *testing.T) { | ||||
| 	l, err := ParseLevel("panic") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, PanicLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("PANIC") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, PanicLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("fatal") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, FatalLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("FATAL") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, FatalLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("error") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, ErrorLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("ERROR") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, ErrorLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("warn") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, WarnLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("WARN") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, WarnLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("warning") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, WarnLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("WARNING") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, WarnLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("info") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, InfoLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("INFO") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, InfoLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("debug") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, DebugLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("DEBUG") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, DebugLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("trace") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, TraceLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("TRACE") | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, TraceLevel, l) | ||||
|  | ||||
| 	l, err = ParseLevel("invalid") | ||||
| 	assert.Equal(t, "not a valid logrus Level: \"invalid\"", err.Error()) | ||||
| } | ||||
|  | ||||
| func TestUnmarshalText(t *testing.T) { | ||||
| 	var u Level | ||||
| 	for _, level := range AllLevels { | ||||
| 		t.Run(level.String(), func(t *testing.T) { | ||||
| 			assert.NoError(t, u.UnmarshalText([]byte(level.String()))) | ||||
| 			assert.Equal(t, level, u) | ||||
| 		}) | ||||
| 	} | ||||
| 	t.Run("invalid", func(t *testing.T) { | ||||
| 		assert.Error(t, u.UnmarshalText([]byte("invalid"))) | ||||
| 	}) | ||||
| } | ||||
|  | ||||
| func TestGetSetLevelRace(t *testing.T) { | ||||
| 	wg := sync.WaitGroup{} | ||||
| 	for i := 0; i < 100; i++ { | ||||
| 		wg.Add(1) | ||||
| 		go func(i int) { | ||||
| 			defer wg.Done() | ||||
| 			if i%2 == 0 { | ||||
| 				SetLevel(InfoLevel) | ||||
| 			} else { | ||||
| 				GetLevel() | ||||
| 			} | ||||
| 		}(i) | ||||
|  | ||||
| 	} | ||||
| 	wg.Wait() | ||||
| } | ||||
|  | ||||
| func TestLoggingRace(t *testing.T) { | ||||
| 	logger := New() | ||||
|  | ||||
| 	var wg sync.WaitGroup | ||||
| 	wg.Add(100) | ||||
|  | ||||
| 	for i := 0; i < 100; i++ { | ||||
| 		go func() { | ||||
| 			logger.Info("info") | ||||
| 			wg.Done() | ||||
| 		}() | ||||
| 	} | ||||
| 	wg.Wait() | ||||
| } | ||||
|  | ||||
| func TestLoggingRaceWithHooksOnEntry(t *testing.T) { | ||||
| 	logger := New() | ||||
| 	hook := new(ModifyHook) | ||||
| 	logger.AddHook(hook) | ||||
| 	entry := logger.WithField("context", "clue") | ||||
|  | ||||
| 	var wg sync.WaitGroup | ||||
| 	wg.Add(100) | ||||
|  | ||||
| 	for i := 0; i < 100; i++ { | ||||
| 		go func() { | ||||
| 			entry.Info("info") | ||||
| 			wg.Done() | ||||
| 		}() | ||||
| 	} | ||||
| 	wg.Wait() | ||||
| } | ||||
|  | ||||
| func TestReplaceHooks(t *testing.T) { | ||||
| 	old, cur := &TestHook{}, &TestHook{} | ||||
|  | ||||
| 	logger := New() | ||||
| 	logger.SetOutput(ioutil.Discard) | ||||
| 	logger.AddHook(old) | ||||
|  | ||||
| 	hooks := make(LevelHooks) | ||||
| 	hooks.Add(cur) | ||||
| 	replaced := logger.ReplaceHooks(hooks) | ||||
|  | ||||
| 	logger.Info("test") | ||||
|  | ||||
| 	assert.Equal(t, old.Fired, false) | ||||
| 	assert.Equal(t, cur.Fired, true) | ||||
|  | ||||
| 	logger.ReplaceHooks(replaced) | ||||
| 	logger.Info("test") | ||||
| 	assert.Equal(t, old.Fired, true) | ||||
| } | ||||
|  | ||||
| // Compile test | ||||
| func TestLogrusInterfaces(t *testing.T) { | ||||
| 	var buffer bytes.Buffer | ||||
| 	// This verifies FieldLogger and Ext1FieldLogger work as designed. | ||||
| 	// Please don't use them. Use Logger and Entry directly. | ||||
| 	fn := func(xl Ext1FieldLogger) { | ||||
| 		var l FieldLogger = xl | ||||
| 		b := l.WithField("key", "value") | ||||
| 		b.Debug("Test") | ||||
| 	} | ||||
| 	// test logger | ||||
| 	logger := New() | ||||
| 	logger.Out = &buffer | ||||
| 	fn(logger) | ||||
|  | ||||
| 	// test Entry | ||||
| 	e := logger.WithField("another", "value") | ||||
| 	fn(e) | ||||
| } | ||||
|  | ||||
| // Implements io.Writer using channels for synchronization, so we can wait on | ||||
| // the Entry.Writer goroutine to write in a non-racey way. This does assume that | ||||
| // there is a single call to Logger.Out for each message. | ||||
| type channelWriter chan []byte | ||||
|  | ||||
| func (cw channelWriter) Write(p []byte) (int, error) { | ||||
| 	cw <- p | ||||
| 	return len(p), nil | ||||
| } | ||||
|  | ||||
| func TestEntryWriter(t *testing.T) { | ||||
| 	cw := channelWriter(make(chan []byte, 1)) | ||||
| 	log := New() | ||||
| 	log.Out = cw | ||||
| 	log.Formatter = new(JSONFormatter) | ||||
| 	log.WithField("foo", "bar").WriterLevel(WarnLevel).Write([]byte("hello\n")) | ||||
|  | ||||
| 	bs := <-cw | ||||
| 	var fields Fields | ||||
| 	err := json.Unmarshal(bs, &fields) | ||||
| 	assert.Nil(t, err) | ||||
| 	assert.Equal(t, fields["foo"], "bar") | ||||
| 	assert.Equal(t, fields["level"], "warning") | ||||
| } | ||||
|  | ||||
| func TestLogLevelEnabled(t *testing.T) { | ||||
| 	log := New() | ||||
| 	log.SetLevel(PanicLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(TraceLevel)) | ||||
|  | ||||
| 	log.SetLevel(FatalLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(TraceLevel)) | ||||
|  | ||||
| 	log.SetLevel(ErrorLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(TraceLevel)) | ||||
|  | ||||
| 	log.SetLevel(WarnLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(TraceLevel)) | ||||
|  | ||||
| 	log.SetLevel(InfoLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(TraceLevel)) | ||||
|  | ||||
| 	log.SetLevel(DebugLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, false, log.IsLevelEnabled(TraceLevel)) | ||||
|  | ||||
| 	log.SetLevel(TraceLevel) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(PanicLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(FatalLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(ErrorLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(WarnLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(InfoLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(DebugLevel)) | ||||
| 	assert.Equal(t, true, log.IsLevelEnabled(TraceLevel)) | ||||
| } | ||||
|  | ||||
| func TestReportCallerOnTextFormatter(t *testing.T) { | ||||
| 	l := New() | ||||
|  | ||||
| 	l.Formatter.(*TextFormatter).ForceColors = true | ||||
| 	l.Formatter.(*TextFormatter).DisableColors = false | ||||
| 	l.WithFields(Fields{"func": "func", "file": "file"}).Info("test") | ||||
|  | ||||
| 	l.Formatter.(*TextFormatter).ForceColors = false | ||||
| 	l.Formatter.(*TextFormatter).DisableColors = true | ||||
| 	l.WithFields(Fields{"func": "func", "file": "file"}).Info("test") | ||||
| } | ||||
							
								
								
									
										9
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_aix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_aix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,9 @@ | ||||
| // +build !appengine,!js,!windows,aix | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import "io" | ||||
|  | ||||
| func checkIfTerminal(w io.Writer) bool { | ||||
| 	return false | ||||
| } | ||||
							
								
								
									
										11
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_appengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_appengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| // +build appengine | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| ) | ||||
|  | ||||
| func checkIfTerminal(w io.Writer) bool { | ||||
| 	return true | ||||
| } | ||||
							
								
								
									
										11
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_js.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_js.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| // +build js | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| ) | ||||
|  | ||||
| func checkIfTerminal(w io.Writer) bool { | ||||
| 	return false | ||||
| } | ||||
							
								
								
									
										19
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_notappengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_notappengine.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| // +build !appengine,!js,!windows,!aix | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| 	"os" | ||||
|  | ||||
| 	"golang.org/x/crypto/ssh/terminal" | ||||
| ) | ||||
|  | ||||
| func checkIfTerminal(w io.Writer) bool { | ||||
| 	switch v := w.(type) { | ||||
| 	case *os.File: | ||||
| 		return terminal.IsTerminal(int(v.Fd())) | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										20
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								vendor/github.com/sirupsen/logrus/terminal_check_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| // +build !appengine,!js,windows | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"syscall" | ||||
| ) | ||||
|  | ||||
| func checkIfTerminal(w io.Writer) bool { | ||||
| 	switch v := w.(type) { | ||||
| 	case *os.File: | ||||
| 		var mode uint32 | ||||
| 		err := syscall.GetConsoleMode(syscall.Handle(v.Fd()), &mode) | ||||
| 		return err == nil | ||||
| 	default: | ||||
| 		return false | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										8
									
								
								vendor/github.com/sirupsen/logrus/terminal_notwindows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								vendor/github.com/sirupsen/logrus/terminal_notwindows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| // +build !windows | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import "io" | ||||
|  | ||||
| func initTerminal(w io.Writer) { | ||||
| } | ||||
							
								
								
									
										18
									
								
								vendor/github.com/sirupsen/logrus/terminal_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								vendor/github.com/sirupsen/logrus/terminal_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| // +build !appengine,!js,windows | ||||
|  | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"syscall" | ||||
|  | ||||
| 	sequences "github.com/konsorten/go-windows-terminal-sequences" | ||||
| ) | ||||
|  | ||||
| func initTerminal(w io.Writer) { | ||||
| 	switch v := w.(type) { | ||||
| 	case *os.File: | ||||
| 		sequences.EnableVirtualTerminalProcessing(syscall.Handle(v.Fd()), true) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										273
									
								
								vendor/github.com/sirupsen/logrus/text_formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										273
									
								
								vendor/github.com/sirupsen/logrus/text_formatter.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,273 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"sync" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| const ( | ||||
| 	nocolor = 0 | ||||
| 	red     = 31 | ||||
| 	green   = 32 | ||||
| 	yellow  = 33 | ||||
| 	blue    = 36 | ||||
| 	gray    = 37 | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	baseTimestamp time.Time | ||||
| 	emptyFieldMap FieldMap | ||||
| ) | ||||
|  | ||||
| func init() { | ||||
| 	baseTimestamp = time.Now() | ||||
| } | ||||
|  | ||||
| // TextFormatter formats logs into text | ||||
| type TextFormatter struct { | ||||
| 	// Set to true to bypass checking for a TTY before outputting colors. | ||||
| 	ForceColors bool | ||||
|  | ||||
| 	// Force disabling colors. | ||||
| 	DisableColors bool | ||||
|  | ||||
| 	// Override coloring based on CLICOLOR and CLICOLOR_FORCE. - https://bixense.com/clicolors/ | ||||
| 	EnvironmentOverrideColors bool | ||||
|  | ||||
| 	// Disable timestamp logging. useful when output is redirected to logging | ||||
| 	// system that already adds timestamps. | ||||
| 	DisableTimestamp bool | ||||
|  | ||||
| 	// Enable logging the full timestamp when a TTY is attached instead of just | ||||
| 	// the time passed since beginning of execution. | ||||
| 	FullTimestamp bool | ||||
|  | ||||
| 	// TimestampFormat to use for display when a full timestamp is printed | ||||
| 	TimestampFormat string | ||||
|  | ||||
| 	// The fields are sorted by default for a consistent output. For applications | ||||
| 	// that log extremely frequently and don't use the JSON formatter this may not | ||||
| 	// be desired. | ||||
| 	DisableSorting bool | ||||
|  | ||||
| 	// The keys sorting function, when uninitialized it uses sort.Strings. | ||||
| 	SortingFunc func([]string) | ||||
|  | ||||
| 	// Disables the truncation of the level text to 4 characters. | ||||
| 	DisableLevelTruncation bool | ||||
|  | ||||
| 	// QuoteEmptyFields will wrap empty fields in quotes if true | ||||
| 	QuoteEmptyFields bool | ||||
|  | ||||
| 	// Whether the logger's out is to a terminal | ||||
| 	isTerminal bool | ||||
|  | ||||
| 	// FieldMap allows users to customize the names of keys for default fields. | ||||
| 	// As an example: | ||||
| 	// formatter := &TextFormatter{ | ||||
| 	//     FieldMap: FieldMap{ | ||||
| 	//         FieldKeyTime:  "@timestamp", | ||||
| 	//         FieldKeyLevel: "@level", | ||||
| 	//         FieldKeyMsg:   "@message"}} | ||||
| 	FieldMap FieldMap | ||||
|  | ||||
| 	terminalInitOnce sync.Once | ||||
| } | ||||
|  | ||||
| func (f *TextFormatter) init(entry *Entry) { | ||||
| 	if entry.Logger != nil { | ||||
| 		f.isTerminal = checkIfTerminal(entry.Logger.Out) | ||||
|  | ||||
| 		if f.isTerminal { | ||||
| 			initTerminal(entry.Logger.Out) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (f *TextFormatter) isColored() bool { | ||||
| 	isColored := f.ForceColors || (f.isTerminal && (runtime.GOOS != "windows")) | ||||
|  | ||||
| 	if f.EnvironmentOverrideColors { | ||||
| 		if force, ok := os.LookupEnv("CLICOLOR_FORCE"); ok && force != "0" { | ||||
| 			isColored = true | ||||
| 		} else if ok && force == "0" { | ||||
| 			isColored = false | ||||
| 		} else if os.Getenv("CLICOLOR") == "0" { | ||||
| 			isColored = false | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return isColored && !f.DisableColors | ||||
| } | ||||
|  | ||||
| // Format renders a single log entry | ||||
| func (f *TextFormatter) Format(entry *Entry) ([]byte, error) { | ||||
| 	data := make(Fields) | ||||
| 	for k, v := range entry.Data { | ||||
| 		data[k] = v | ||||
| 	} | ||||
| 	prefixFieldClashes(data, f.FieldMap, entry.HasCaller()) | ||||
| 	keys := make([]string, 0, len(data)) | ||||
| 	for k := range data { | ||||
| 		keys = append(keys, k) | ||||
| 	} | ||||
|  | ||||
| 	fixedKeys := make([]string, 0, 4+len(data)) | ||||
| 	if !f.DisableTimestamp { | ||||
| 		fixedKeys = append(fixedKeys, f.FieldMap.resolve(FieldKeyTime)) | ||||
| 	} | ||||
| 	fixedKeys = append(fixedKeys, f.FieldMap.resolve(FieldKeyLevel)) | ||||
| 	if entry.Message != "" { | ||||
| 		fixedKeys = append(fixedKeys, f.FieldMap.resolve(FieldKeyMsg)) | ||||
| 	} | ||||
| 	if entry.err != "" { | ||||
| 		fixedKeys = append(fixedKeys, f.FieldMap.resolve(FieldKeyLogrusError)) | ||||
| 	} | ||||
| 	if entry.HasCaller() { | ||||
| 		fixedKeys = append(fixedKeys, | ||||
| 			f.FieldMap.resolve(FieldKeyFunc), f.FieldMap.resolve(FieldKeyFile)) | ||||
| 	} | ||||
|  | ||||
| 	if !f.DisableSorting { | ||||
| 		if f.SortingFunc == nil { | ||||
| 			sort.Strings(keys) | ||||
| 			fixedKeys = append(fixedKeys, keys...) | ||||
| 		} else { | ||||
| 			if !f.isColored() { | ||||
| 				fixedKeys = append(fixedKeys, keys...) | ||||
| 				f.SortingFunc(fixedKeys) | ||||
| 			} else { | ||||
| 				f.SortingFunc(keys) | ||||
| 			} | ||||
| 		} | ||||
| 	} else { | ||||
| 		fixedKeys = append(fixedKeys, keys...) | ||||
| 	} | ||||
|  | ||||
| 	var b *bytes.Buffer | ||||
| 	if entry.Buffer != nil { | ||||
| 		b = entry.Buffer | ||||
| 	} else { | ||||
| 		b = &bytes.Buffer{} | ||||
| 	} | ||||
|  | ||||
| 	f.terminalInitOnce.Do(func() { f.init(entry) }) | ||||
|  | ||||
| 	timestampFormat := f.TimestampFormat | ||||
| 	if timestampFormat == "" { | ||||
| 		timestampFormat = defaultTimestampFormat | ||||
| 	} | ||||
| 	if f.isColored() { | ||||
| 		f.printColored(b, entry, keys, data, timestampFormat) | ||||
| 	} else { | ||||
| 		for _, key := range fixedKeys { | ||||
| 			var value interface{} | ||||
| 			switch { | ||||
| 			case key == f.FieldMap.resolve(FieldKeyTime): | ||||
| 				value = entry.Time.Format(timestampFormat) | ||||
| 			case key == f.FieldMap.resolve(FieldKeyLevel): | ||||
| 				value = entry.Level.String() | ||||
| 			case key == f.FieldMap.resolve(FieldKeyMsg): | ||||
| 				value = entry.Message | ||||
| 			case key == f.FieldMap.resolve(FieldKeyLogrusError): | ||||
| 				value = entry.err | ||||
| 			case key == f.FieldMap.resolve(FieldKeyFunc) && entry.HasCaller(): | ||||
| 				value = entry.Caller.Function | ||||
| 			case key == f.FieldMap.resolve(FieldKeyFile) && entry.HasCaller(): | ||||
| 				value = fmt.Sprintf("%s:%d", entry.Caller.File, entry.Caller.Line) | ||||
| 			default: | ||||
| 				value = data[key] | ||||
| 			} | ||||
| 			f.appendKeyValue(b, key, value) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	b.WriteByte('\n') | ||||
| 	return b.Bytes(), nil | ||||
| } | ||||
|  | ||||
| func (f *TextFormatter) printColored(b *bytes.Buffer, entry *Entry, keys []string, data Fields, timestampFormat string) { | ||||
| 	var levelColor int | ||||
| 	switch entry.Level { | ||||
| 	case DebugLevel, TraceLevel: | ||||
| 		levelColor = gray | ||||
| 	case WarnLevel: | ||||
| 		levelColor = yellow | ||||
| 	case ErrorLevel, FatalLevel, PanicLevel: | ||||
| 		levelColor = red | ||||
| 	default: | ||||
| 		levelColor = blue | ||||
| 	} | ||||
|  | ||||
| 	levelText := strings.ToUpper(entry.Level.String()) | ||||
| 	if !f.DisableLevelTruncation { | ||||
| 		levelText = levelText[0:4] | ||||
| 	} | ||||
|  | ||||
| 	// Remove a single newline if it already exists in the message to keep | ||||
| 	// the behavior of logrus text_formatter the same as the stdlib log package | ||||
| 	entry.Message = strings.TrimSuffix(entry.Message, "\n") | ||||
|  | ||||
| 	caller := "" | ||||
|  | ||||
| 	if entry.HasCaller() { | ||||
| 		caller = fmt.Sprintf("%s:%d %s()", | ||||
| 			entry.Caller.File, entry.Caller.Line, entry.Caller.Function) | ||||
| 	} | ||||
|  | ||||
| 	if f.DisableTimestamp { | ||||
| 		fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m%s %-44s ", levelColor, levelText, caller, entry.Message) | ||||
| 	} else if !f.FullTimestamp { | ||||
| 		fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m[%04d]%s %-44s ", levelColor, levelText, int(entry.Time.Sub(baseTimestamp)/time.Second), caller, entry.Message) | ||||
| 	} else { | ||||
| 		fmt.Fprintf(b, "\x1b[%dm%s\x1b[0m[%s]%s %-44s ", levelColor, levelText, entry.Time.Format(timestampFormat), caller, entry.Message) | ||||
| 	} | ||||
| 	for _, k := range keys { | ||||
| 		v := data[k] | ||||
| 		fmt.Fprintf(b, " \x1b[%dm%s\x1b[0m=", levelColor, k) | ||||
| 		f.appendValue(b, v) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (f *TextFormatter) needsQuoting(text string) bool { | ||||
| 	if f.QuoteEmptyFields && len(text) == 0 { | ||||
| 		return true | ||||
| 	} | ||||
| 	for _, ch := range text { | ||||
| 		if !((ch >= 'a' && ch <= 'z') || | ||||
| 			(ch >= 'A' && ch <= 'Z') || | ||||
| 			(ch >= '0' && ch <= '9') || | ||||
| 			ch == '-' || ch == '.' || ch == '_' || ch == '/' || ch == '@' || ch == '^' || ch == '+') { | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| func (f *TextFormatter) appendKeyValue(b *bytes.Buffer, key string, value interface{}) { | ||||
| 	if b.Len() > 0 { | ||||
| 		b.WriteByte(' ') | ||||
| 	} | ||||
| 	b.WriteString(key) | ||||
| 	b.WriteByte('=') | ||||
| 	f.appendValue(b, value) | ||||
| } | ||||
|  | ||||
| func (f *TextFormatter) appendValue(b *bytes.Buffer, value interface{}) { | ||||
| 	stringVal, ok := value.(string) | ||||
| 	if !ok { | ||||
| 		stringVal = fmt.Sprint(value) | ||||
| 	} | ||||
|  | ||||
| 	if !f.needsQuoting(stringVal) { | ||||
| 		b.WriteString(stringVal) | ||||
| 	} else { | ||||
| 		b.WriteString(fmt.Sprintf("%q", stringVal)) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										485
									
								
								vendor/github.com/sirupsen/logrus/text_formatter_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										485
									
								
								vendor/github.com/sirupsen/logrus/text_formatter_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,485 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"sort" | ||||
| 	"strings" | ||||
| 	"testing" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/stretchr/testify/assert" | ||||
| 	"github.com/stretchr/testify/require" | ||||
| ) | ||||
|  | ||||
| func TestFormatting(t *testing.T) { | ||||
| 	tf := &TextFormatter{DisableColors: true} | ||||
|  | ||||
| 	testCases := []struct { | ||||
| 		value    string | ||||
| 		expected string | ||||
| 	}{ | ||||
| 		{`foo`, "time=\"0001-01-01T00:00:00Z\" level=panic test=foo\n"}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range testCases { | ||||
| 		b, _ := tf.Format(WithField("test", tc.value)) | ||||
|  | ||||
| 		if string(b) != tc.expected { | ||||
| 			t.Errorf("formatting expected for %q (result was %q instead of %q)", tc.value, string(b), tc.expected) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestQuoting(t *testing.T) { | ||||
| 	tf := &TextFormatter{DisableColors: true} | ||||
|  | ||||
| 	checkQuoting := func(q bool, value interface{}) { | ||||
| 		b, _ := tf.Format(WithField("test", value)) | ||||
| 		idx := bytes.Index(b, ([]byte)("test=")) | ||||
| 		cont := bytes.Contains(b[idx+5:], []byte("\"")) | ||||
| 		if cont != q { | ||||
| 			if q { | ||||
| 				t.Errorf("quoting expected for: %#v", value) | ||||
| 			} else { | ||||
| 				t.Errorf("quoting not expected for: %#v", value) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	checkQuoting(false, "") | ||||
| 	checkQuoting(false, "abcd") | ||||
| 	checkQuoting(false, "v1.0") | ||||
| 	checkQuoting(false, "1234567890") | ||||
| 	checkQuoting(false, "/foobar") | ||||
| 	checkQuoting(false, "foo_bar") | ||||
| 	checkQuoting(false, "foo@bar") | ||||
| 	checkQuoting(false, "foobar^") | ||||
| 	checkQuoting(false, "+/-_^@f.oobar") | ||||
| 	checkQuoting(true, "foobar$") | ||||
| 	checkQuoting(true, "&foobar") | ||||
| 	checkQuoting(true, "x y") | ||||
| 	checkQuoting(true, "x,y") | ||||
| 	checkQuoting(false, errors.New("invalid")) | ||||
| 	checkQuoting(true, errors.New("invalid argument")) | ||||
|  | ||||
| 	// Test for quoting empty fields. | ||||
| 	tf.QuoteEmptyFields = true | ||||
| 	checkQuoting(true, "") | ||||
| 	checkQuoting(false, "abcd") | ||||
| 	checkQuoting(true, errors.New("invalid argument")) | ||||
| } | ||||
|  | ||||
| func TestEscaping(t *testing.T) { | ||||
| 	tf := &TextFormatter{DisableColors: true} | ||||
|  | ||||
| 	testCases := []struct { | ||||
| 		value    string | ||||
| 		expected string | ||||
| 	}{ | ||||
| 		{`ba"r`, `ba\"r`}, | ||||
| 		{`ba'r`, `ba'r`}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range testCases { | ||||
| 		b, _ := tf.Format(WithField("test", tc.value)) | ||||
| 		if !bytes.Contains(b, []byte(tc.expected)) { | ||||
| 			t.Errorf("escaping expected for %q (result was %q instead of %q)", tc.value, string(b), tc.expected) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEscaping_Interface(t *testing.T) { | ||||
| 	tf := &TextFormatter{DisableColors: true} | ||||
|  | ||||
| 	ts := time.Now() | ||||
|  | ||||
| 	testCases := []struct { | ||||
| 		value    interface{} | ||||
| 		expected string | ||||
| 	}{ | ||||
| 		{ts, fmt.Sprintf("\"%s\"", ts.String())}, | ||||
| 		{errors.New("error: something went wrong"), "\"error: something went wrong\""}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tc := range testCases { | ||||
| 		b, _ := tf.Format(WithField("test", tc.value)) | ||||
| 		if !bytes.Contains(b, []byte(tc.expected)) { | ||||
| 			t.Errorf("escaping expected for %q (result was %q instead of %q)", tc.value, string(b), tc.expected) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestTimestampFormat(t *testing.T) { | ||||
| 	checkTimeStr := func(format string) { | ||||
| 		customFormatter := &TextFormatter{DisableColors: true, TimestampFormat: format} | ||||
| 		customStr, _ := customFormatter.Format(WithField("test", "test")) | ||||
| 		timeStart := bytes.Index(customStr, ([]byte)("time=")) | ||||
| 		timeEnd := bytes.Index(customStr, ([]byte)("level=")) | ||||
| 		timeStr := customStr[timeStart+5+len("\"") : timeEnd-1-len("\"")] | ||||
| 		if format == "" { | ||||
| 			format = time.RFC3339 | ||||
| 		} | ||||
| 		_, e := time.Parse(format, (string)(timeStr)) | ||||
| 		if e != nil { | ||||
| 			t.Errorf("time string \"%s\" did not match provided time format \"%s\": %s", timeStr, format, e) | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	checkTimeStr("2006-01-02T15:04:05.000000000Z07:00") | ||||
| 	checkTimeStr("Mon Jan _2 15:04:05 2006") | ||||
| 	checkTimeStr("") | ||||
| } | ||||
|  | ||||
| func TestDisableLevelTruncation(t *testing.T) { | ||||
| 	entry := &Entry{ | ||||
| 		Time:    time.Now(), | ||||
| 		Message: "testing", | ||||
| 	} | ||||
| 	keys := []string{} | ||||
| 	timestampFormat := "Mon Jan 2 15:04:05 -0700 MST 2006" | ||||
| 	checkDisableTruncation := func(disabled bool, level Level) { | ||||
| 		tf := &TextFormatter{DisableLevelTruncation: disabled} | ||||
| 		var b bytes.Buffer | ||||
| 		entry.Level = level | ||||
| 		tf.printColored(&b, entry, keys, nil, timestampFormat) | ||||
| 		logLine := (&b).String() | ||||
| 		if disabled { | ||||
| 			expected := strings.ToUpper(level.String()) | ||||
| 			if !strings.Contains(logLine, expected) { | ||||
| 				t.Errorf("level string expected to be %s when truncation disabled", expected) | ||||
| 			} | ||||
| 		} else { | ||||
| 			expected := strings.ToUpper(level.String()) | ||||
| 			if len(level.String()) > 4 { | ||||
| 				if strings.Contains(logLine, expected) { | ||||
| 					t.Errorf("level string %s expected to be truncated to %s when truncation is enabled", expected, expected[0:4]) | ||||
| 				} | ||||
| 			} else { | ||||
| 				if !strings.Contains(logLine, expected) { | ||||
| 					t.Errorf("level string expected to be %s when truncation is enabled and level string is below truncation threshold", expected) | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	checkDisableTruncation(true, DebugLevel) | ||||
| 	checkDisableTruncation(true, InfoLevel) | ||||
| 	checkDisableTruncation(false, ErrorLevel) | ||||
| 	checkDisableTruncation(false, InfoLevel) | ||||
| } | ||||
|  | ||||
| func TestDisableTimestampWithColoredOutput(t *testing.T) { | ||||
| 	tf := &TextFormatter{DisableTimestamp: true, ForceColors: true} | ||||
|  | ||||
| 	b, _ := tf.Format(WithField("test", "test")) | ||||
| 	if strings.Contains(string(b), "[0000]") { | ||||
| 		t.Error("timestamp not expected when DisableTimestamp is true") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNewlineBehavior(t *testing.T) { | ||||
| 	tf := &TextFormatter{ForceColors: true} | ||||
|  | ||||
| 	// Ensure a single new line is removed as per stdlib log | ||||
| 	e := NewEntry(StandardLogger()) | ||||
| 	e.Message = "test message\n" | ||||
| 	b, _ := tf.Format(e) | ||||
| 	if bytes.Contains(b, []byte("test message\n")) { | ||||
| 		t.Error("first newline at end of Entry.Message resulted in unexpected 2 newlines in output. Expected newline to be removed.") | ||||
| 	} | ||||
|  | ||||
| 	// Ensure a double new line is reduced to a single new line | ||||
| 	e = NewEntry(StandardLogger()) | ||||
| 	e.Message = "test message\n\n" | ||||
| 	b, _ = tf.Format(e) | ||||
| 	if bytes.Contains(b, []byte("test message\n\n")) { | ||||
| 		t.Error("Double newline at end of Entry.Message resulted in unexpected 2 newlines in output. Expected single newline") | ||||
| 	} | ||||
| 	if !bytes.Contains(b, []byte("test message\n")) { | ||||
| 		t.Error("Double newline at end of Entry.Message did not result in a single newline after formatting") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestTextFormatterFieldMap(t *testing.T) { | ||||
| 	formatter := &TextFormatter{ | ||||
| 		DisableColors: true, | ||||
| 		FieldMap: FieldMap{ | ||||
| 			FieldKeyMsg:   "message", | ||||
| 			FieldKeyLevel: "somelevel", | ||||
| 			FieldKeyTime:  "timeywimey", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	entry := &Entry{ | ||||
| 		Message: "oh hi", | ||||
| 		Level:   WarnLevel, | ||||
| 		Time:    time.Date(1981, time.February, 24, 4, 28, 3, 100, time.UTC), | ||||
| 		Data: Fields{ | ||||
| 			"field1":     "f1", | ||||
| 			"message":    "messagefield", | ||||
| 			"somelevel":  "levelfield", | ||||
| 			"timeywimey": "timeywimeyfield", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	b, err := formatter.Format(entry) | ||||
| 	if err != nil { | ||||
| 		t.Fatal("Unable to format entry: ", err) | ||||
| 	} | ||||
|  | ||||
| 	assert.Equal(t, | ||||
| 		`timeywimey="1981-02-24T04:28:03Z" `+ | ||||
| 			`somelevel=warning `+ | ||||
| 			`message="oh hi" `+ | ||||
| 			`field1=f1 `+ | ||||
| 			`fields.message=messagefield `+ | ||||
| 			`fields.somelevel=levelfield `+ | ||||
| 			`fields.timeywimey=timeywimeyfield`+"\n", | ||||
| 		string(b), | ||||
| 		"Formatted output doesn't respect FieldMap") | ||||
| } | ||||
|  | ||||
| func TestTextFormatterIsColored(t *testing.T) { | ||||
| 	params := []struct { | ||||
| 		name               string | ||||
| 		expectedResult     bool | ||||
| 		isTerminal         bool | ||||
| 		disableColor       bool | ||||
| 		forceColor         bool | ||||
| 		envColor           bool | ||||
| 		clicolorIsSet      bool | ||||
| 		clicolorForceIsSet bool | ||||
| 		clicolorVal        string | ||||
| 		clicolorForceVal   string | ||||
| 	}{ | ||||
| 		// Default values | ||||
| 		{ | ||||
| 			name:               "testcase1", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           false, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: false, | ||||
| 		}, | ||||
| 		// Output on terminal | ||||
| 		{ | ||||
| 			name:               "testcase2", | ||||
| 			expectedResult:     true, | ||||
| 			isTerminal:         true, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           false, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: false, | ||||
| 		}, | ||||
| 		// Output on terminal with color disabled | ||||
| 		{ | ||||
| 			name:               "testcase3", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         true, | ||||
| 			disableColor:       true, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           false, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: false, | ||||
| 		}, | ||||
| 		// Output not on terminal with color disabled | ||||
| 		{ | ||||
| 			name:               "testcase4", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       true, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           false, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: false, | ||||
| 		}, | ||||
| 		// Output not on terminal with color forced | ||||
| 		{ | ||||
| 			name:               "testcase5", | ||||
| 			expectedResult:     true, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         true, | ||||
| 			envColor:           false, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: false, | ||||
| 		}, | ||||
| 		// Output on terminal with clicolor set to "0" | ||||
| 		{ | ||||
| 			name:               "testcase6", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         true, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      true, | ||||
| 			clicolorForceIsSet: false, | ||||
| 			clicolorVal:        "0", | ||||
| 		}, | ||||
| 		// Output on terminal with clicolor set to "1" | ||||
| 		{ | ||||
| 			name:               "testcase7", | ||||
| 			expectedResult:     true, | ||||
| 			isTerminal:         true, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      true, | ||||
| 			clicolorForceIsSet: false, | ||||
| 			clicolorVal:        "1", | ||||
| 		}, | ||||
| 		// Output not on terminal with clicolor set to "0" | ||||
| 		{ | ||||
| 			name:               "testcase8", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      true, | ||||
| 			clicolorForceIsSet: false, | ||||
| 			clicolorVal:        "0", | ||||
| 		}, | ||||
| 		// Output not on terminal with clicolor set to "1" | ||||
| 		{ | ||||
| 			name:               "testcase9", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      true, | ||||
| 			clicolorForceIsSet: false, | ||||
| 			clicolorVal:        "1", | ||||
| 		}, | ||||
| 		// Output not on terminal with clicolor set to "1" and force color | ||||
| 		{ | ||||
| 			name:               "testcase10", | ||||
| 			expectedResult:     true, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         true, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      true, | ||||
| 			clicolorForceIsSet: false, | ||||
| 			clicolorVal:        "1", | ||||
| 		}, | ||||
| 		// Output not on terminal with clicolor set to "0" and force color | ||||
| 		{ | ||||
| 			name:               "testcase11", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         true, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      true, | ||||
| 			clicolorForceIsSet: false, | ||||
| 			clicolorVal:        "0", | ||||
| 		}, | ||||
| 		// Output not on terminal with clicolor_force set to "1" | ||||
| 		{ | ||||
| 			name:               "testcase12", | ||||
| 			expectedResult:     true, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: true, | ||||
| 			clicolorForceVal:   "1", | ||||
| 		}, | ||||
| 		// Output not on terminal with clicolor_force set to "0" | ||||
| 		{ | ||||
| 			name:               "testcase13", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         false, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: true, | ||||
| 			clicolorForceVal:   "0", | ||||
| 		}, | ||||
| 		// Output on terminal with clicolor_force set to "0" | ||||
| 		{ | ||||
| 			name:               "testcase14", | ||||
| 			expectedResult:     false, | ||||
| 			isTerminal:         true, | ||||
| 			disableColor:       false, | ||||
| 			forceColor:         false, | ||||
| 			envColor:           true, | ||||
| 			clicolorIsSet:      false, | ||||
| 			clicolorForceIsSet: true, | ||||
| 			clicolorForceVal:   "0", | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	cleanenv := func() { | ||||
| 		os.Unsetenv("CLICOLOR") | ||||
| 		os.Unsetenv("CLICOLOR_FORCE") | ||||
| 	} | ||||
|  | ||||
| 	defer cleanenv() | ||||
|  | ||||
| 	for _, val := range params { | ||||
| 		t.Run("textformatter_"+val.name, func(subT *testing.T) { | ||||
| 			tf := TextFormatter{ | ||||
| 				isTerminal:                val.isTerminal, | ||||
| 				DisableColors:             val.disableColor, | ||||
| 				ForceColors:               val.forceColor, | ||||
| 				EnvironmentOverrideColors: val.envColor, | ||||
| 			} | ||||
| 			cleanenv() | ||||
| 			if val.clicolorIsSet { | ||||
| 				os.Setenv("CLICOLOR", val.clicolorVal) | ||||
| 			} | ||||
| 			if val.clicolorForceIsSet { | ||||
| 				os.Setenv("CLICOLOR_FORCE", val.clicolorForceVal) | ||||
| 			} | ||||
| 			res := tf.isColored() | ||||
| 			if runtime.GOOS == "windows" && !tf.ForceColors && !val.clicolorForceIsSet { | ||||
| 				assert.Equal(subT, false, res) | ||||
| 			} else { | ||||
| 				assert.Equal(subT, val.expectedResult, res) | ||||
| 			} | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestCustomSorting(t *testing.T) { | ||||
| 	formatter := &TextFormatter{ | ||||
| 		DisableColors: true, | ||||
| 		SortingFunc: func(keys []string) { | ||||
| 			sort.Slice(keys, func(i, j int) bool { | ||||
| 				if keys[j] == "prefix" { | ||||
| 					return false | ||||
| 				} | ||||
| 				if keys[i] == "prefix" { | ||||
| 					return true | ||||
| 				} | ||||
| 				return strings.Compare(keys[i], keys[j]) == -1 | ||||
| 			}) | ||||
| 		}, | ||||
| 	} | ||||
|  | ||||
| 	entry := &Entry{ | ||||
| 		Message: "Testing custom sort function", | ||||
| 		Time:    time.Now(), | ||||
| 		Level:   InfoLevel, | ||||
| 		Data: Fields{ | ||||
| 			"test":      "testvalue", | ||||
| 			"prefix":    "the application prefix", | ||||
| 			"blablabla": "blablabla", | ||||
| 		}, | ||||
| 	} | ||||
| 	b, err := formatter.Format(entry) | ||||
| 	require.NoError(t, err) | ||||
| 	require.True(t, strings.HasPrefix(string(b), "prefix="), "format output is %q", string(b)) | ||||
| } | ||||
							
								
								
									
										64
									
								
								vendor/github.com/sirupsen/logrus/writer.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								vendor/github.com/sirupsen/logrus/writer.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| package logrus | ||||
|  | ||||
| import ( | ||||
| 	"bufio" | ||||
| 	"io" | ||||
| 	"runtime" | ||||
| ) | ||||
|  | ||||
| func (logger *Logger) Writer() *io.PipeWriter { | ||||
| 	return logger.WriterLevel(InfoLevel) | ||||
| } | ||||
|  | ||||
| func (logger *Logger) WriterLevel(level Level) *io.PipeWriter { | ||||
| 	return NewEntry(logger).WriterLevel(level) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) Writer() *io.PipeWriter { | ||||
| 	return entry.WriterLevel(InfoLevel) | ||||
| } | ||||
|  | ||||
| func (entry *Entry) WriterLevel(level Level) *io.PipeWriter { | ||||
| 	reader, writer := io.Pipe() | ||||
|  | ||||
| 	var printFunc func(args ...interface{}) | ||||
|  | ||||
| 	switch level { | ||||
| 	case TraceLevel: | ||||
| 		printFunc = entry.Trace | ||||
| 	case DebugLevel: | ||||
| 		printFunc = entry.Debug | ||||
| 	case InfoLevel: | ||||
| 		printFunc = entry.Info | ||||
| 	case WarnLevel: | ||||
| 		printFunc = entry.Warn | ||||
| 	case ErrorLevel: | ||||
| 		printFunc = entry.Error | ||||
| 	case FatalLevel: | ||||
| 		printFunc = entry.Fatal | ||||
| 	case PanicLevel: | ||||
| 		printFunc = entry.Panic | ||||
| 	default: | ||||
| 		printFunc = entry.Print | ||||
| 	} | ||||
|  | ||||
| 	go entry.writerScanner(reader, printFunc) | ||||
| 	runtime.SetFinalizer(writer, writerFinalizer) | ||||
|  | ||||
| 	return writer | ||||
| } | ||||
|  | ||||
| func (entry *Entry) writerScanner(reader *io.PipeReader, printFunc func(args ...interface{})) { | ||||
| 	scanner := bufio.NewScanner(reader) | ||||
| 	for scanner.Scan() { | ||||
| 		printFunc(scanner.Text()) | ||||
| 	} | ||||
| 	if err := scanner.Err(); err != nil { | ||||
| 		entry.Errorf("Error while reading from Writer: %s", err) | ||||
| 	} | ||||
| 	reader.Close() | ||||
| } | ||||
|  | ||||
| func writerFinalizer(writer *io.PipeWriter) { | ||||
| 	writer.Close() | ||||
| } | ||||
							
								
								
									
										28
									
								
								vendor/github.com/stretchr/testify/require/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								vendor/github.com/stretchr/testify/require/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| // Package require implements the same assertions as the `assert` package but | ||||
| // stops test execution when a test fails. | ||||
| // | ||||
| // Example Usage | ||||
| // | ||||
| // The following is a complete example using require in a standard test function: | ||||
| //    import ( | ||||
| //      "testing" | ||||
| //      "github.com/stretchr/testify/require" | ||||
| //    ) | ||||
| // | ||||
| //    func TestSomething(t *testing.T) { | ||||
| // | ||||
| //      var a string = "Hello" | ||||
| //      var b string = "Hello" | ||||
| // | ||||
| //      require.Equal(t, a, b, "The two words should be the same.") | ||||
| // | ||||
| //    } | ||||
| // | ||||
| // Assertions | ||||
| // | ||||
| // The `require` package have same global functions as in the `assert` package, | ||||
| // but instead of returning a boolean result they call `t.FailNow()`. | ||||
| // | ||||
| // Every assertion function also takes an optional string message as the final argument, | ||||
| // allowing custom error messages to be appended to the message the assertion method outputs. | ||||
| package require | ||||
							
								
								
									
										16
									
								
								vendor/github.com/stretchr/testify/require/forward_requirements.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								vendor/github.com/stretchr/testify/require/forward_requirements.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| package require | ||||
|  | ||||
| // Assertions provides assertion methods around the | ||||
| // TestingT interface. | ||||
| type Assertions struct { | ||||
| 	t TestingT | ||||
| } | ||||
|  | ||||
| // New makes a new Assertions object for the specified TestingT. | ||||
| func New(t TestingT) *Assertions { | ||||
| 	return &Assertions{ | ||||
| 		t: t, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| //go:generate go run ../_codegen/main.go -output-package=require -template=require_forward.go.tmpl -include-format-funcs | ||||
							
								
								
									
										385
									
								
								vendor/github.com/stretchr/testify/require/forward_requirements_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										385
									
								
								vendor/github.com/stretchr/testify/require/forward_requirements_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,385 @@ | ||||
| package require | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| func TestImplementsWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
|  | ||||
| 	require.Implements((*AssertionTesterInterface)(nil), new(AssertionTesterConformingObject)) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Implements((*AssertionTesterInterface)(nil), new(AssertionTesterNonConformingObject)) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestIsTypeWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.IsType(new(AssertionTesterConformingObject), new(AssertionTesterConformingObject)) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.IsType(new(AssertionTesterConformingObject), new(AssertionTesterNonConformingObject)) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEqualWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Equal(1, 1) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Equal(1, 2) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotEqualWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NotEqual(1, 2) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NotEqual(2, 2) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestExactlyWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
|  | ||||
| 	a := float32(1) | ||||
| 	b := float32(1) | ||||
| 	c := float64(1) | ||||
|  | ||||
| 	require.Exactly(a, b) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Exactly(a, c) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotNilWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NotNil(t, new(AssertionTesterConformingObject)) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NotNil(nil) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNilWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Nil(nil) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Nil(new(AssertionTesterConformingObject)) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestTrueWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.True(true) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.True(false) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFalseWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.False(false) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.False(true) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestContainsWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Contains("Hello World", "Hello") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Contains("Hello World", "Salut") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotContainsWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NotContains("Hello World", "Hello!") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NotContains("Hello World", "Hello") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPanicsWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Panics(func() { | ||||
| 		panic("Panic!") | ||||
| 	}) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Panics(func() {}) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotPanicsWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NotPanics(func() {}) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NotPanics(func() { | ||||
| 		panic("Panic!") | ||||
| 	}) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNoErrorWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NoError(nil) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NoError(errors.New("some error")) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Error(errors.New("some error")) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Error(nil) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEqualErrorWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.EqualError(errors.New("some error"), "some error") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.EqualError(errors.New("some error"), "Not some error") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEmptyWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Empty("") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Empty("x") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotEmptyWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NotEmpty("x") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NotEmpty("") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestWithinDurationWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	a := time.Now() | ||||
| 	b := a.Add(10 * time.Second) | ||||
|  | ||||
| 	require.WithinDuration(a, b, 15*time.Second) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.WithinDuration(a, b, 5*time.Second) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestInDeltaWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.InDelta(1.001, 1, 0.01) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.InDelta(1, 2, 0.5) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestZeroWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.Zero(0) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.Zero(1) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotZeroWrapper(t *testing.T) { | ||||
| 	require := New(t) | ||||
| 	require.NotZero(1) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
| 	mockRequire.NotZero(0) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_EqualSONString(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`{"hello": "world", "foo": "bar"}`, `{"hello": "world", "foo": "bar"}`) | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_EquivalentButNotEqual(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_HashOfArraysAndHashes(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq("{\r\n\t\"numeric\": 1.5,\r\n\t\"array\": [{\"foo\": \"bar\"}, 1, \"string\", [\"nested\", \"array\", 5.5]],\r\n\t\"hash\": {\"nested\": \"hash\", \"nested_slice\": [\"this\", \"is\", \"nested\"]},\r\n\t\"string\": \"foo\"\r\n}", | ||||
| 		"{\r\n\t\"numeric\": 1.5,\r\n\t\"hash\": {\"nested\": \"hash\", \"nested_slice\": [\"this\", \"is\", \"nested\"]},\r\n\t\"string\": \"foo\",\r\n\t\"array\": [{\"foo\": \"bar\"}, 1, \"string\", [\"nested\", \"array\", 5.5]]\r\n}") | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_Array(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`["foo", {"hello": "world", "nested": "hash"}]`, `["foo", {"nested": "hash", "hello": "world"}]`) | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_HashAndArrayNotEquivalent(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`["foo", {"hello": "world", "nested": "hash"}]`, `{"foo": "bar", {"nested": "hash", "hello": "world"}}`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_HashesNotEquivalent(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`{"foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_ActualIsNotJSON(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`{"foo": "bar"}`, "Not JSON") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_ExpectedIsNotJSON(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq("Not JSON", `{"foo": "bar", "hello": "world"}`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_ExpectedAndActualNotJSON(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq("Not JSON", "Not JSON") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEqWrapper_ArraysOfDifferentOrder(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	mockRequire := New(mockT) | ||||
|  | ||||
| 	mockRequire.JSONEq(`["foo", {"hello": "world", "nested": "hash"}]`, `[{ "hello": "world", "nested": "hash"}, "foo"]`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										1227
									
								
								vendor/github.com/stretchr/testify/require/require.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1227
									
								
								vendor/github.com/stretchr/testify/require/require.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										6
									
								
								vendor/github.com/stretchr/testify/require/require.go.tmpl
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								vendor/github.com/stretchr/testify/require/require.go.tmpl
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | ||||
| {{.Comment}} | ||||
| func {{.DocInfo.Name}}(t TestingT, {{.Params}}) { | ||||
| 	if assert.{{.DocInfo.Name}}(t, {{.ForwardedParams}}) { return } | ||||
| 	if h, ok := t.(tHelper); ok { h.Helper() } | ||||
| 	t.FailNow() | ||||
| } | ||||
							
								
								
									
										957
									
								
								vendor/github.com/stretchr/testify/require/require_forward.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										957
									
								
								vendor/github.com/stretchr/testify/require/require_forward.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,957 @@ | ||||
| /* | ||||
| * CODE GENERATED AUTOMATICALLY WITH github.com/stretchr/testify/_codegen | ||||
| * THIS FILE MUST NOT BE EDITED BY HAND | ||||
|  */ | ||||
|  | ||||
| package require | ||||
|  | ||||
| import ( | ||||
| 	assert "github.com/stretchr/testify/assert" | ||||
| 	http "net/http" | ||||
| 	url "net/url" | ||||
| 	time "time" | ||||
| ) | ||||
|  | ||||
| // Condition uses a Comparison to assert a complex condition. | ||||
| func (a *Assertions) Condition(comp assert.Comparison, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Condition(a.t, comp, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Conditionf uses a Comparison to assert a complex condition. | ||||
| func (a *Assertions) Conditionf(comp assert.Comparison, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Conditionf(a.t, comp, msg, args...) | ||||
| } | ||||
|  | ||||
| // Contains asserts that the specified string, list(array, slice...) or map contains the | ||||
| // specified substring or element. | ||||
| // | ||||
| //    a.Contains("Hello World", "World") | ||||
| //    a.Contains(["Hello", "World"], "World") | ||||
| //    a.Contains({"Hello": "World"}, "Hello") | ||||
| func (a *Assertions) Contains(s interface{}, contains interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Contains(a.t, s, contains, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Containsf asserts that the specified string, list(array, slice...) or map contains the | ||||
| // specified substring or element. | ||||
| // | ||||
| //    a.Containsf("Hello World", "World", "error message %s", "formatted") | ||||
| //    a.Containsf(["Hello", "World"], "World", "error message %s", "formatted") | ||||
| //    a.Containsf({"Hello": "World"}, "Hello", "error message %s", "formatted") | ||||
| func (a *Assertions) Containsf(s interface{}, contains interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Containsf(a.t, s, contains, msg, args...) | ||||
| } | ||||
|  | ||||
| // DirExists checks whether a directory exists in the given path. It also fails if the path is a file rather a directory or there is an error checking whether it exists. | ||||
| func (a *Assertions) DirExists(path string, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	DirExists(a.t, path, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // DirExistsf checks whether a directory exists in the given path. It also fails if the path is a file rather a directory or there is an error checking whether it exists. | ||||
| func (a *Assertions) DirExistsf(path string, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	DirExistsf(a.t, path, msg, args...) | ||||
| } | ||||
|  | ||||
| // ElementsMatch asserts that the specified listA(array, slice...) is equal to specified | ||||
| // listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, | ||||
| // the number of appearances of each of them in both lists should match. | ||||
| // | ||||
| // a.ElementsMatch([1, 3, 2, 3], [1, 3, 3, 2]) | ||||
| func (a *Assertions) ElementsMatch(listA interface{}, listB interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	ElementsMatch(a.t, listA, listB, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // ElementsMatchf asserts that the specified listA(array, slice...) is equal to specified | ||||
| // listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, | ||||
| // the number of appearances of each of them in both lists should match. | ||||
| // | ||||
| // a.ElementsMatchf([1, 3, 2, 3], [1, 3, 3, 2], "error message %s", "formatted") | ||||
| func (a *Assertions) ElementsMatchf(listA interface{}, listB interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	ElementsMatchf(a.t, listA, listB, msg, args...) | ||||
| } | ||||
|  | ||||
| // Empty asserts that the specified object is empty.  I.e. nil, "", false, 0 or either | ||||
| // a slice or a channel with len == 0. | ||||
| // | ||||
| //  a.Empty(obj) | ||||
| func (a *Assertions) Empty(object interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Empty(a.t, object, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Emptyf asserts that the specified object is empty.  I.e. nil, "", false, 0 or either | ||||
| // a slice or a channel with len == 0. | ||||
| // | ||||
| //  a.Emptyf(obj, "error message %s", "formatted") | ||||
| func (a *Assertions) Emptyf(object interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Emptyf(a.t, object, msg, args...) | ||||
| } | ||||
|  | ||||
| // Equal asserts that two objects are equal. | ||||
| // | ||||
| //    a.Equal(123, 123) | ||||
| // | ||||
| // Pointer variable equality is determined based on the equality of the | ||||
| // referenced values (as opposed to the memory addresses). Function equality | ||||
| // cannot be determined and will always fail. | ||||
| func (a *Assertions) Equal(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Equal(a.t, expected, actual, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // EqualError asserts that a function returned an error (i.e. not `nil`) | ||||
| // and that it is equal to the provided error. | ||||
| // | ||||
| //   actualObj, err := SomeFunction() | ||||
| //   a.EqualError(err,  expectedErrorString) | ||||
| func (a *Assertions) EqualError(theError error, errString string, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	EqualError(a.t, theError, errString, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // EqualErrorf asserts that a function returned an error (i.e. not `nil`) | ||||
| // and that it is equal to the provided error. | ||||
| // | ||||
| //   actualObj, err := SomeFunction() | ||||
| //   a.EqualErrorf(err,  expectedErrorString, "error message %s", "formatted") | ||||
| func (a *Assertions) EqualErrorf(theError error, errString string, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	EqualErrorf(a.t, theError, errString, msg, args...) | ||||
| } | ||||
|  | ||||
| // EqualValues asserts that two objects are equal or convertable to the same types | ||||
| // and equal. | ||||
| // | ||||
| //    a.EqualValues(uint32(123), int32(123)) | ||||
| func (a *Assertions) EqualValues(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	EqualValues(a.t, expected, actual, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // EqualValuesf asserts that two objects are equal or convertable to the same types | ||||
| // and equal. | ||||
| // | ||||
| //    a.EqualValuesf(uint32(123, "error message %s", "formatted"), int32(123)) | ||||
| func (a *Assertions) EqualValuesf(expected interface{}, actual interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	EqualValuesf(a.t, expected, actual, msg, args...) | ||||
| } | ||||
|  | ||||
| // Equalf asserts that two objects are equal. | ||||
| // | ||||
| //    a.Equalf(123, 123, "error message %s", "formatted") | ||||
| // | ||||
| // Pointer variable equality is determined based on the equality of the | ||||
| // referenced values (as opposed to the memory addresses). Function equality | ||||
| // cannot be determined and will always fail. | ||||
| func (a *Assertions) Equalf(expected interface{}, actual interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Equalf(a.t, expected, actual, msg, args...) | ||||
| } | ||||
|  | ||||
| // Error asserts that a function returned an error (i.e. not `nil`). | ||||
| // | ||||
| //   actualObj, err := SomeFunction() | ||||
| //   if a.Error(err) { | ||||
| // 	   assert.Equal(t, expectedError, err) | ||||
| //   } | ||||
| func (a *Assertions) Error(err error, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Error(a.t, err, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Errorf asserts that a function returned an error (i.e. not `nil`). | ||||
| // | ||||
| //   actualObj, err := SomeFunction() | ||||
| //   if a.Errorf(err, "error message %s", "formatted") { | ||||
| // 	   assert.Equal(t, expectedErrorf, err) | ||||
| //   } | ||||
| func (a *Assertions) Errorf(err error, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Errorf(a.t, err, msg, args...) | ||||
| } | ||||
|  | ||||
| // Exactly asserts that two objects are equal in value and type. | ||||
| // | ||||
| //    a.Exactly(int32(123), int64(123)) | ||||
| func (a *Assertions) Exactly(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Exactly(a.t, expected, actual, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Exactlyf asserts that two objects are equal in value and type. | ||||
| // | ||||
| //    a.Exactlyf(int32(123, "error message %s", "formatted"), int64(123)) | ||||
| func (a *Assertions) Exactlyf(expected interface{}, actual interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Exactlyf(a.t, expected, actual, msg, args...) | ||||
| } | ||||
|  | ||||
| // Fail reports a failure through | ||||
| func (a *Assertions) Fail(failureMessage string, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Fail(a.t, failureMessage, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // FailNow fails test | ||||
| func (a *Assertions) FailNow(failureMessage string, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	FailNow(a.t, failureMessage, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // FailNowf fails test | ||||
| func (a *Assertions) FailNowf(failureMessage string, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	FailNowf(a.t, failureMessage, msg, args...) | ||||
| } | ||||
|  | ||||
| // Failf reports a failure through | ||||
| func (a *Assertions) Failf(failureMessage string, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Failf(a.t, failureMessage, msg, args...) | ||||
| } | ||||
|  | ||||
| // False asserts that the specified value is false. | ||||
| // | ||||
| //    a.False(myBool) | ||||
| func (a *Assertions) False(value bool, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	False(a.t, value, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Falsef asserts that the specified value is false. | ||||
| // | ||||
| //    a.Falsef(myBool, "error message %s", "formatted") | ||||
| func (a *Assertions) Falsef(value bool, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Falsef(a.t, value, msg, args...) | ||||
| } | ||||
|  | ||||
| // FileExists checks whether a file exists in the given path. It also fails if the path points to a directory or there is an error when trying to check the file. | ||||
| func (a *Assertions) FileExists(path string, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	FileExists(a.t, path, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // FileExistsf checks whether a file exists in the given path. It also fails if the path points to a directory or there is an error when trying to check the file. | ||||
| func (a *Assertions) FileExistsf(path string, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	FileExistsf(a.t, path, msg, args...) | ||||
| } | ||||
|  | ||||
| // HTTPBodyContains asserts that a specified handler returns a | ||||
| // body that contains a string. | ||||
| // | ||||
| //  a.HTTPBodyContains(myHandler, "www.google.com", nil, "I'm Feeling Lucky") | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPBodyContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPBodyContains(a.t, handler, method, url, values, str, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // HTTPBodyContainsf asserts that a specified handler returns a | ||||
| // body that contains a string. | ||||
| // | ||||
| //  a.HTTPBodyContainsf(myHandler, "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPBodyContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPBodyContainsf(a.t, handler, method, url, values, str, msg, args...) | ||||
| } | ||||
|  | ||||
| // HTTPBodyNotContains asserts that a specified handler returns a | ||||
| // body that does not contain a string. | ||||
| // | ||||
| //  a.HTTPBodyNotContains(myHandler, "www.google.com", nil, "I'm Feeling Lucky") | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPBodyNotContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPBodyNotContains(a.t, handler, method, url, values, str, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // HTTPBodyNotContainsf asserts that a specified handler returns a | ||||
| // body that does not contain a string. | ||||
| // | ||||
| //  a.HTTPBodyNotContainsf(myHandler, "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPBodyNotContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPBodyNotContainsf(a.t, handler, method, url, values, str, msg, args...) | ||||
| } | ||||
|  | ||||
| // HTTPError asserts that a specified handler returns an error status code. | ||||
| // | ||||
| //  a.HTTPError(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPError(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPError(a.t, handler, method, url, values, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // HTTPErrorf asserts that a specified handler returns an error status code. | ||||
| // | ||||
| //  a.HTTPErrorf(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} | ||||
| // | ||||
| // Returns whether the assertion was successful (true, "error message %s", "formatted") or not (false). | ||||
| func (a *Assertions) HTTPErrorf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPErrorf(a.t, handler, method, url, values, msg, args...) | ||||
| } | ||||
|  | ||||
| // HTTPRedirect asserts that a specified handler returns a redirect status code. | ||||
| // | ||||
| //  a.HTTPRedirect(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPRedirect(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPRedirect(a.t, handler, method, url, values, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // HTTPRedirectf asserts that a specified handler returns a redirect status code. | ||||
| // | ||||
| //  a.HTTPRedirectf(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} | ||||
| // | ||||
| // Returns whether the assertion was successful (true, "error message %s", "formatted") or not (false). | ||||
| func (a *Assertions) HTTPRedirectf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPRedirectf(a.t, handler, method, url, values, msg, args...) | ||||
| } | ||||
|  | ||||
| // HTTPSuccess asserts that a specified handler returns a success status code. | ||||
| // | ||||
| //  a.HTTPSuccess(myHandler, "POST", "http://www.google.com", nil) | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPSuccess(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPSuccess(a.t, handler, method, url, values, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // HTTPSuccessf asserts that a specified handler returns a success status code. | ||||
| // | ||||
| //  a.HTTPSuccessf(myHandler, "POST", "http://www.google.com", nil, "error message %s", "formatted") | ||||
| // | ||||
| // Returns whether the assertion was successful (true) or not (false). | ||||
| func (a *Assertions) HTTPSuccessf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	HTTPSuccessf(a.t, handler, method, url, values, msg, args...) | ||||
| } | ||||
|  | ||||
| // Implements asserts that an object is implemented by the specified interface. | ||||
| // | ||||
| //    a.Implements((*MyInterface)(nil), new(MyObject)) | ||||
| func (a *Assertions) Implements(interfaceObject interface{}, object interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Implements(a.t, interfaceObject, object, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Implementsf asserts that an object is implemented by the specified interface. | ||||
| // | ||||
| //    a.Implementsf((*MyInterface, "error message %s", "formatted")(nil), new(MyObject)) | ||||
| func (a *Assertions) Implementsf(interfaceObject interface{}, object interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Implementsf(a.t, interfaceObject, object, msg, args...) | ||||
| } | ||||
|  | ||||
| // InDelta asserts that the two numerals are within delta of each other. | ||||
| // | ||||
| // 	 a.InDelta(math.Pi, (22 / 7.0), 0.01) | ||||
| func (a *Assertions) InDelta(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InDelta(a.t, expected, actual, delta, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // InDeltaMapValues is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. | ||||
| func (a *Assertions) InDeltaMapValues(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InDeltaMapValues(a.t, expected, actual, delta, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // InDeltaMapValuesf is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. | ||||
| func (a *Assertions) InDeltaMapValuesf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InDeltaMapValuesf(a.t, expected, actual, delta, msg, args...) | ||||
| } | ||||
|  | ||||
| // InDeltaSlice is the same as InDelta, except it compares two slices. | ||||
| func (a *Assertions) InDeltaSlice(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InDeltaSlice(a.t, expected, actual, delta, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // InDeltaSlicef is the same as InDelta, except it compares two slices. | ||||
| func (a *Assertions) InDeltaSlicef(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InDeltaSlicef(a.t, expected, actual, delta, msg, args...) | ||||
| } | ||||
|  | ||||
| // InDeltaf asserts that the two numerals are within delta of each other. | ||||
| // | ||||
| // 	 a.InDeltaf(math.Pi, (22 / 7.0, "error message %s", "formatted"), 0.01) | ||||
| func (a *Assertions) InDeltaf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InDeltaf(a.t, expected, actual, delta, msg, args...) | ||||
| } | ||||
|  | ||||
| // InEpsilon asserts that expected and actual have a relative error less than epsilon | ||||
| func (a *Assertions) InEpsilon(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InEpsilon(a.t, expected, actual, epsilon, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // InEpsilonSlice is the same as InEpsilon, except it compares each value from two slices. | ||||
| func (a *Assertions) InEpsilonSlice(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InEpsilonSlice(a.t, expected, actual, epsilon, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // InEpsilonSlicef is the same as InEpsilon, except it compares each value from two slices. | ||||
| func (a *Assertions) InEpsilonSlicef(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InEpsilonSlicef(a.t, expected, actual, epsilon, msg, args...) | ||||
| } | ||||
|  | ||||
| // InEpsilonf asserts that expected and actual have a relative error less than epsilon | ||||
| func (a *Assertions) InEpsilonf(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	InEpsilonf(a.t, expected, actual, epsilon, msg, args...) | ||||
| } | ||||
|  | ||||
| // IsType asserts that the specified objects are of the same type. | ||||
| func (a *Assertions) IsType(expectedType interface{}, object interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	IsType(a.t, expectedType, object, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // IsTypef asserts that the specified objects are of the same type. | ||||
| func (a *Assertions) IsTypef(expectedType interface{}, object interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	IsTypef(a.t, expectedType, object, msg, args...) | ||||
| } | ||||
|  | ||||
| // JSONEq asserts that two JSON strings are equivalent. | ||||
| // | ||||
| //  a.JSONEq(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) | ||||
| func (a *Assertions) JSONEq(expected string, actual string, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	JSONEq(a.t, expected, actual, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // JSONEqf asserts that two JSON strings are equivalent. | ||||
| // | ||||
| //  a.JSONEqf(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`, "error message %s", "formatted") | ||||
| func (a *Assertions) JSONEqf(expected string, actual string, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	JSONEqf(a.t, expected, actual, msg, args...) | ||||
| } | ||||
|  | ||||
| // Len asserts that the specified object has specific length. | ||||
| // Len also fails if the object has a type that len() not accept. | ||||
| // | ||||
| //    a.Len(mySlice, 3) | ||||
| func (a *Assertions) Len(object interface{}, length int, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Len(a.t, object, length, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Lenf asserts that the specified object has specific length. | ||||
| // Lenf also fails if the object has a type that len() not accept. | ||||
| // | ||||
| //    a.Lenf(mySlice, 3, "error message %s", "formatted") | ||||
| func (a *Assertions) Lenf(object interface{}, length int, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Lenf(a.t, object, length, msg, args...) | ||||
| } | ||||
|  | ||||
| // Nil asserts that the specified object is nil. | ||||
| // | ||||
| //    a.Nil(err) | ||||
| func (a *Assertions) Nil(object interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Nil(a.t, object, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Nilf asserts that the specified object is nil. | ||||
| // | ||||
| //    a.Nilf(err, "error message %s", "formatted") | ||||
| func (a *Assertions) Nilf(object interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Nilf(a.t, object, msg, args...) | ||||
| } | ||||
|  | ||||
| // NoError asserts that a function returned no error (i.e. `nil`). | ||||
| // | ||||
| //   actualObj, err := SomeFunction() | ||||
| //   if a.NoError(err) { | ||||
| // 	   assert.Equal(t, expectedObj, actualObj) | ||||
| //   } | ||||
| func (a *Assertions) NoError(err error, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NoError(a.t, err, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NoErrorf asserts that a function returned no error (i.e. `nil`). | ||||
| // | ||||
| //   actualObj, err := SomeFunction() | ||||
| //   if a.NoErrorf(err, "error message %s", "formatted") { | ||||
| // 	   assert.Equal(t, expectedObj, actualObj) | ||||
| //   } | ||||
| func (a *Assertions) NoErrorf(err error, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NoErrorf(a.t, err, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotContains asserts that the specified string, list(array, slice...) or map does NOT contain the | ||||
| // specified substring or element. | ||||
| // | ||||
| //    a.NotContains("Hello World", "Earth") | ||||
| //    a.NotContains(["Hello", "World"], "Earth") | ||||
| //    a.NotContains({"Hello": "World"}, "Earth") | ||||
| func (a *Assertions) NotContains(s interface{}, contains interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotContains(a.t, s, contains, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotContainsf asserts that the specified string, list(array, slice...) or map does NOT contain the | ||||
| // specified substring or element. | ||||
| // | ||||
| //    a.NotContainsf("Hello World", "Earth", "error message %s", "formatted") | ||||
| //    a.NotContainsf(["Hello", "World"], "Earth", "error message %s", "formatted") | ||||
| //    a.NotContainsf({"Hello": "World"}, "Earth", "error message %s", "formatted") | ||||
| func (a *Assertions) NotContainsf(s interface{}, contains interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotContainsf(a.t, s, contains, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotEmpty asserts that the specified object is NOT empty.  I.e. not nil, "", false, 0 or either | ||||
| // a slice or a channel with len == 0. | ||||
| // | ||||
| //  if a.NotEmpty(obj) { | ||||
| //    assert.Equal(t, "two", obj[1]) | ||||
| //  } | ||||
| func (a *Assertions) NotEmpty(object interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotEmpty(a.t, object, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotEmptyf asserts that the specified object is NOT empty.  I.e. not nil, "", false, 0 or either | ||||
| // a slice or a channel with len == 0. | ||||
| // | ||||
| //  if a.NotEmptyf(obj, "error message %s", "formatted") { | ||||
| //    assert.Equal(t, "two", obj[1]) | ||||
| //  } | ||||
| func (a *Assertions) NotEmptyf(object interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotEmptyf(a.t, object, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotEqual asserts that the specified values are NOT equal. | ||||
| // | ||||
| //    a.NotEqual(obj1, obj2) | ||||
| // | ||||
| // Pointer variable equality is determined based on the equality of the | ||||
| // referenced values (as opposed to the memory addresses). | ||||
| func (a *Assertions) NotEqual(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotEqual(a.t, expected, actual, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotEqualf asserts that the specified values are NOT equal. | ||||
| // | ||||
| //    a.NotEqualf(obj1, obj2, "error message %s", "formatted") | ||||
| // | ||||
| // Pointer variable equality is determined based on the equality of the | ||||
| // referenced values (as opposed to the memory addresses). | ||||
| func (a *Assertions) NotEqualf(expected interface{}, actual interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotEqualf(a.t, expected, actual, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotNil asserts that the specified object is not nil. | ||||
| // | ||||
| //    a.NotNil(err) | ||||
| func (a *Assertions) NotNil(object interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotNil(a.t, object, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotNilf asserts that the specified object is not nil. | ||||
| // | ||||
| //    a.NotNilf(err, "error message %s", "formatted") | ||||
| func (a *Assertions) NotNilf(object interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotNilf(a.t, object, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotPanics asserts that the code inside the specified PanicTestFunc does NOT panic. | ||||
| // | ||||
| //   a.NotPanics(func(){ RemainCalm() }) | ||||
| func (a *Assertions) NotPanics(f assert.PanicTestFunc, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotPanics(a.t, f, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotPanicsf asserts that the code inside the specified PanicTestFunc does NOT panic. | ||||
| // | ||||
| //   a.NotPanicsf(func(){ RemainCalm() }, "error message %s", "formatted") | ||||
| func (a *Assertions) NotPanicsf(f assert.PanicTestFunc, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotPanicsf(a.t, f, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotRegexp asserts that a specified regexp does not match a string. | ||||
| // | ||||
| //  a.NotRegexp(regexp.MustCompile("starts"), "it's starting") | ||||
| //  a.NotRegexp("^start", "it's not starting") | ||||
| func (a *Assertions) NotRegexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotRegexp(a.t, rx, str, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotRegexpf asserts that a specified regexp does not match a string. | ||||
| // | ||||
| //  a.NotRegexpf(regexp.MustCompile("starts", "error message %s", "formatted"), "it's starting") | ||||
| //  a.NotRegexpf("^start", "it's not starting", "error message %s", "formatted") | ||||
| func (a *Assertions) NotRegexpf(rx interface{}, str interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotRegexpf(a.t, rx, str, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotSubset asserts that the specified list(array, slice...) contains not all | ||||
| // elements given in the specified subset(array, slice...). | ||||
| // | ||||
| //    a.NotSubset([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]") | ||||
| func (a *Assertions) NotSubset(list interface{}, subset interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotSubset(a.t, list, subset, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotSubsetf asserts that the specified list(array, slice...) contains not all | ||||
| // elements given in the specified subset(array, slice...). | ||||
| // | ||||
| //    a.NotSubsetf([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]", "error message %s", "formatted") | ||||
| func (a *Assertions) NotSubsetf(list interface{}, subset interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotSubsetf(a.t, list, subset, msg, args...) | ||||
| } | ||||
|  | ||||
| // NotZero asserts that i is not the zero value for its type. | ||||
| func (a *Assertions) NotZero(i interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotZero(a.t, i, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // NotZerof asserts that i is not the zero value for its type. | ||||
| func (a *Assertions) NotZerof(i interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	NotZerof(a.t, i, msg, args...) | ||||
| } | ||||
|  | ||||
| // Panics asserts that the code inside the specified PanicTestFunc panics. | ||||
| // | ||||
| //   a.Panics(func(){ GoCrazy() }) | ||||
| func (a *Assertions) Panics(f assert.PanicTestFunc, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Panics(a.t, f, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // PanicsWithValue asserts that the code inside the specified PanicTestFunc panics, and that | ||||
| // the recovered panic value equals the expected panic value. | ||||
| // | ||||
| //   a.PanicsWithValue("crazy error", func(){ GoCrazy() }) | ||||
| func (a *Assertions) PanicsWithValue(expected interface{}, f assert.PanicTestFunc, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	PanicsWithValue(a.t, expected, f, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // PanicsWithValuef asserts that the code inside the specified PanicTestFunc panics, and that | ||||
| // the recovered panic value equals the expected panic value. | ||||
| // | ||||
| //   a.PanicsWithValuef("crazy error", func(){ GoCrazy() }, "error message %s", "formatted") | ||||
| func (a *Assertions) PanicsWithValuef(expected interface{}, f assert.PanicTestFunc, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	PanicsWithValuef(a.t, expected, f, msg, args...) | ||||
| } | ||||
|  | ||||
| // Panicsf asserts that the code inside the specified PanicTestFunc panics. | ||||
| // | ||||
| //   a.Panicsf(func(){ GoCrazy() }, "error message %s", "formatted") | ||||
| func (a *Assertions) Panicsf(f assert.PanicTestFunc, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Panicsf(a.t, f, msg, args...) | ||||
| } | ||||
|  | ||||
| // Regexp asserts that a specified regexp matches a string. | ||||
| // | ||||
| //  a.Regexp(regexp.MustCompile("start"), "it's starting") | ||||
| //  a.Regexp("start...$", "it's not starting") | ||||
| func (a *Assertions) Regexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Regexp(a.t, rx, str, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Regexpf asserts that a specified regexp matches a string. | ||||
| // | ||||
| //  a.Regexpf(regexp.MustCompile("start", "error message %s", "formatted"), "it's starting") | ||||
| //  a.Regexpf("start...$", "it's not starting", "error message %s", "formatted") | ||||
| func (a *Assertions) Regexpf(rx interface{}, str interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Regexpf(a.t, rx, str, msg, args...) | ||||
| } | ||||
|  | ||||
| // Subset asserts that the specified list(array, slice...) contains all | ||||
| // elements given in the specified subset(array, slice...). | ||||
| // | ||||
| //    a.Subset([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]") | ||||
| func (a *Assertions) Subset(list interface{}, subset interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Subset(a.t, list, subset, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Subsetf asserts that the specified list(array, slice...) contains all | ||||
| // elements given in the specified subset(array, slice...). | ||||
| // | ||||
| //    a.Subsetf([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]", "error message %s", "formatted") | ||||
| func (a *Assertions) Subsetf(list interface{}, subset interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Subsetf(a.t, list, subset, msg, args...) | ||||
| } | ||||
|  | ||||
| // True asserts that the specified value is true. | ||||
| // | ||||
| //    a.True(myBool) | ||||
| func (a *Assertions) True(value bool, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	True(a.t, value, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Truef asserts that the specified value is true. | ||||
| // | ||||
| //    a.Truef(myBool, "error message %s", "formatted") | ||||
| func (a *Assertions) Truef(value bool, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Truef(a.t, value, msg, args...) | ||||
| } | ||||
|  | ||||
| // WithinDuration asserts that the two times are within duration delta of each other. | ||||
| // | ||||
| //   a.WithinDuration(time.Now(), time.Now(), 10*time.Second) | ||||
| func (a *Assertions) WithinDuration(expected time.Time, actual time.Time, delta time.Duration, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	WithinDuration(a.t, expected, actual, delta, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // WithinDurationf asserts that the two times are within duration delta of each other. | ||||
| // | ||||
| //   a.WithinDurationf(time.Now(), time.Now(), 10*time.Second, "error message %s", "formatted") | ||||
| func (a *Assertions) WithinDurationf(expected time.Time, actual time.Time, delta time.Duration, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	WithinDurationf(a.t, expected, actual, delta, msg, args...) | ||||
| } | ||||
|  | ||||
| // Zero asserts that i is the zero value for its type. | ||||
| func (a *Assertions) Zero(i interface{}, msgAndArgs ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Zero(a.t, i, msgAndArgs...) | ||||
| } | ||||
|  | ||||
| // Zerof asserts that i is the zero value for its type. | ||||
| func (a *Assertions) Zerof(i interface{}, msg string, args ...interface{}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { | ||||
| 		h.Helper() | ||||
| 	} | ||||
| 	Zerof(a.t, i, msg, args...) | ||||
| } | ||||
							
								
								
									
										5
									
								
								vendor/github.com/stretchr/testify/require/require_forward.go.tmpl
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								vendor/github.com/stretchr/testify/require/require_forward.go.tmpl
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| {{.CommentWithoutT "a"}} | ||||
| func (a *Assertions) {{.DocInfo.Name}}({{.Params}}) { | ||||
| 	if h, ok := a.t.(tHelper); ok { h.Helper() } | ||||
| 	{{.DocInfo.Name}}(a.t, {{.ForwardedParams}}) | ||||
| } | ||||
							
								
								
									
										29
									
								
								vendor/github.com/stretchr/testify/require/requirements.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/github.com/stretchr/testify/require/requirements.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| package require | ||||
|  | ||||
| // TestingT is an interface wrapper around *testing.T | ||||
| type TestingT interface { | ||||
| 	Errorf(format string, args ...interface{}) | ||||
| 	FailNow() | ||||
| } | ||||
|  | ||||
| type tHelper interface { | ||||
| 	Helper() | ||||
| } | ||||
|  | ||||
| // ComparisonAssertionFunc is a common function prototype when comparing two values.  Can be useful | ||||
| // for table driven tests. | ||||
| type ComparisonAssertionFunc func(TestingT, interface{}, interface{}, ...interface{}) | ||||
|  | ||||
| // ValueAssertionFunc is a common function prototype when validating a single value.  Can be useful | ||||
| // for table driven tests. | ||||
| type ValueAssertionFunc func(TestingT, interface{}, ...interface{}) | ||||
|  | ||||
| // BoolAssertionFunc is a common function prototype when validating a bool value.  Can be useful | ||||
| // for table driven tests. | ||||
| type BoolAssertionFunc func(TestingT, bool, ...interface{}) | ||||
|  | ||||
| // ValuesAssertionFunc is a common function prototype when validating an error value.  Can be useful | ||||
| // for table driven tests. | ||||
| type ErrorAssertionFunc func(TestingT, error, ...interface{}) | ||||
|  | ||||
| //go:generate go run ../_codegen/main.go -output-package=require -template=require.go.tmpl -include-format-funcs | ||||
							
								
								
									
										566
									
								
								vendor/github.com/stretchr/testify/require/requirements_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										566
									
								
								vendor/github.com/stretchr/testify/require/requirements_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,566 @@ | ||||
| package require | ||||
|  | ||||
| import ( | ||||
| 	"encoding/json" | ||||
| 	"errors" | ||||
| 	"testing" | ||||
| 	"time" | ||||
| ) | ||||
|  | ||||
| // AssertionTesterInterface defines an interface to be used for testing assertion methods | ||||
| type AssertionTesterInterface interface { | ||||
| 	TestMethod() | ||||
| } | ||||
|  | ||||
| // AssertionTesterConformingObject is an object that conforms to the AssertionTesterInterface interface | ||||
| type AssertionTesterConformingObject struct { | ||||
| } | ||||
|  | ||||
| func (a *AssertionTesterConformingObject) TestMethod() { | ||||
| } | ||||
|  | ||||
| // AssertionTesterNonConformingObject is an object that does not conform to the AssertionTesterInterface interface | ||||
| type AssertionTesterNonConformingObject struct { | ||||
| } | ||||
|  | ||||
| type MockT struct { | ||||
| 	Failed bool | ||||
| } | ||||
|  | ||||
| func (t *MockT) FailNow() { | ||||
| 	t.Failed = true | ||||
| } | ||||
|  | ||||
| func (t *MockT) Errorf(format string, args ...interface{}) { | ||||
| 	_, _ = format, args | ||||
| } | ||||
|  | ||||
| func TestImplements(t *testing.T) { | ||||
|  | ||||
| 	Implements(t, (*AssertionTesterInterface)(nil), new(AssertionTesterConformingObject)) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Implements(mockT, (*AssertionTesterInterface)(nil), new(AssertionTesterNonConformingObject)) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestIsType(t *testing.T) { | ||||
|  | ||||
| 	IsType(t, new(AssertionTesterConformingObject), new(AssertionTesterConformingObject)) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	IsType(mockT, new(AssertionTesterConformingObject), new(AssertionTesterNonConformingObject)) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEqual(t *testing.T) { | ||||
|  | ||||
| 	Equal(t, 1, 1) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Equal(mockT, 1, 2) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
|  | ||||
| } | ||||
|  | ||||
| func TestNotEqual(t *testing.T) { | ||||
|  | ||||
| 	NotEqual(t, 1, 2) | ||||
| 	mockT := new(MockT) | ||||
| 	NotEqual(mockT, 2, 2) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestExactly(t *testing.T) { | ||||
|  | ||||
| 	a := float32(1) | ||||
| 	b := float32(1) | ||||
| 	c := float64(1) | ||||
|  | ||||
| 	Exactly(t, a, b) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Exactly(mockT, a, c) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotNil(t *testing.T) { | ||||
|  | ||||
| 	NotNil(t, new(AssertionTesterConformingObject)) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	NotNil(mockT, nil) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNil(t *testing.T) { | ||||
|  | ||||
| 	Nil(t, nil) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Nil(mockT, new(AssertionTesterConformingObject)) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestTrue(t *testing.T) { | ||||
|  | ||||
| 	True(t, true) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	True(mockT, false) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestFalse(t *testing.T) { | ||||
|  | ||||
| 	False(t, false) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	False(mockT, true) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestContains(t *testing.T) { | ||||
|  | ||||
| 	Contains(t, "Hello World", "Hello") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Contains(mockT, "Hello World", "Salut") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotContains(t *testing.T) { | ||||
|  | ||||
| 	NotContains(t, "Hello World", "Hello!") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	NotContains(mockT, "Hello World", "Hello") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPanics(t *testing.T) { | ||||
|  | ||||
| 	Panics(t, func() { | ||||
| 		panic("Panic!") | ||||
| 	}) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Panics(mockT, func() {}) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotPanics(t *testing.T) { | ||||
|  | ||||
| 	NotPanics(t, func() {}) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	NotPanics(mockT, func() { | ||||
| 		panic("Panic!") | ||||
| 	}) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNoError(t *testing.T) { | ||||
|  | ||||
| 	NoError(t, nil) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	NoError(mockT, errors.New("some error")) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestError(t *testing.T) { | ||||
|  | ||||
| 	Error(t, errors.New("some error")) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Error(mockT, nil) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEqualError(t *testing.T) { | ||||
|  | ||||
| 	EqualError(t, errors.New("some error"), "some error") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	EqualError(mockT, errors.New("some error"), "Not some error") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestEmpty(t *testing.T) { | ||||
|  | ||||
| 	Empty(t, "") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Empty(mockT, "x") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotEmpty(t *testing.T) { | ||||
|  | ||||
| 	NotEmpty(t, "x") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	NotEmpty(mockT, "") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestWithinDuration(t *testing.T) { | ||||
|  | ||||
| 	a := time.Now() | ||||
| 	b := a.Add(10 * time.Second) | ||||
|  | ||||
| 	WithinDuration(t, a, b, 15*time.Second) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	WithinDuration(mockT, a, b, 5*time.Second) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestInDelta(t *testing.T) { | ||||
|  | ||||
| 	InDelta(t, 1.001, 1, 0.01) | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	InDelta(mockT, 1, 2, 0.5) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestZero(t *testing.T) { | ||||
|  | ||||
| 	Zero(t, "") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	Zero(mockT, "x") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestNotZero(t *testing.T) { | ||||
|  | ||||
| 	NotZero(t, "x") | ||||
|  | ||||
| 	mockT := new(MockT) | ||||
| 	NotZero(mockT, "") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_EqualSONString(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `{"hello": "world", "foo": "bar"}`, `{"hello": "world", "foo": "bar"}`) | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_EquivalentButNotEqual(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_HashOfArraysAndHashes(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, "{\r\n\t\"numeric\": 1.5,\r\n\t\"array\": [{\"foo\": \"bar\"}, 1, \"string\", [\"nested\", \"array\", 5.5]],\r\n\t\"hash\": {\"nested\": \"hash\", \"nested_slice\": [\"this\", \"is\", \"nested\"]},\r\n\t\"string\": \"foo\"\r\n}", | ||||
| 		"{\r\n\t\"numeric\": 1.5,\r\n\t\"hash\": {\"nested\": \"hash\", \"nested_slice\": [\"this\", \"is\", \"nested\"]},\r\n\t\"string\": \"foo\",\r\n\t\"array\": [{\"foo\": \"bar\"}, 1, \"string\", [\"nested\", \"array\", 5.5]]\r\n}") | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_Array(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `["foo", {"hello": "world", "nested": "hash"}]`, `["foo", {"nested": "hash", "hello": "world"}]`) | ||||
| 	if mockT.Failed { | ||||
| 		t.Error("Check should pass") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_HashAndArrayNotEquivalent(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `["foo", {"hello": "world", "nested": "hash"}]`, `{"foo": "bar", {"nested": "hash", "hello": "world"}}`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_HashesNotEquivalent(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `{"foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_ActualIsNotJSON(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `{"foo": "bar"}`, "Not JSON") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_ExpectedIsNotJSON(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, "Not JSON", `{"foo": "bar", "hello": "world"}`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_ExpectedAndActualNotJSON(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, "Not JSON", "Not JSON") | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestJSONEq_ArraysOfDifferentOrder(t *testing.T) { | ||||
| 	mockT := new(MockT) | ||||
| 	JSONEq(mockT, `["foo", {"hello": "world", "nested": "hash"}]`, `[{ "hello": "world", "nested": "hash"}, "foo"]`) | ||||
| 	if !mockT.Failed { | ||||
| 		t.Error("Check should fail") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ExampleComparisonAssertionFunc() { | ||||
| 	t := &testing.T{} // provided by test | ||||
|  | ||||
| 	adder := func(x, y int) int { | ||||
| 		return x + y | ||||
| 	} | ||||
|  | ||||
| 	type args struct { | ||||
| 		x int | ||||
| 		y int | ||||
| 	} | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		args      args | ||||
| 		expect    int | ||||
| 		assertion ComparisonAssertionFunc | ||||
| 	}{ | ||||
| 		{"2+2=4", args{2, 2}, 4, Equal}, | ||||
| 		{"2+2!=5", args{2, 2}, 5, NotEqual}, | ||||
| 		{"2+3==5", args{2, 3}, 5, Exactly}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, tt.expect, adder(tt.args.x, tt.args.y)) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestComparisonAssertionFunc(t *testing.T) { | ||||
| 	type iface interface { | ||||
| 		Name() string | ||||
| 	} | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		expect    interface{} | ||||
| 		got       interface{} | ||||
| 		assertion ComparisonAssertionFunc | ||||
| 	}{ | ||||
| 		{"implements", (*iface)(nil), t, Implements}, | ||||
| 		{"isType", (*testing.T)(nil), t, IsType}, | ||||
| 		{"equal", t, t, Equal}, | ||||
| 		{"equalValues", t, t, EqualValues}, | ||||
| 		{"exactly", t, t, Exactly}, | ||||
| 		{"notEqual", t, nil, NotEqual}, | ||||
| 		{"notContains", []int{1, 2, 3}, 4, NotContains}, | ||||
| 		{"subset", []int{1, 2, 3, 4}, []int{2, 3}, Subset}, | ||||
| 		{"notSubset", []int{1, 2, 3, 4}, []int{0, 3}, NotSubset}, | ||||
| 		{"elementsMatch", []byte("abc"), []byte("bac"), ElementsMatch}, | ||||
| 		{"regexp", "^t.*y$", "testify", Regexp}, | ||||
| 		{"notRegexp", "^t.*y$", "Testify", NotRegexp}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, tt.expect, tt.got) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ExampleValueAssertionFunc() { | ||||
| 	t := &testing.T{} // provided by test | ||||
|  | ||||
| 	dumbParse := func(input string) interface{} { | ||||
| 		var x interface{} | ||||
| 		json.Unmarshal([]byte(input), &x) | ||||
| 		return x | ||||
| 	} | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		arg       string | ||||
| 		assertion ValueAssertionFunc | ||||
| 	}{ | ||||
| 		{"true is not nil", "true", NotNil}, | ||||
| 		{"empty string is nil", "", Nil}, | ||||
| 		{"zero is not nil", "0", NotNil}, | ||||
| 		{"zero is zero", "0", Zero}, | ||||
| 		{"false is zero", "false", Zero}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, dumbParse(tt.arg)) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestValueAssertionFunc(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		value     interface{} | ||||
| 		assertion ValueAssertionFunc | ||||
| 	}{ | ||||
| 		{"notNil", true, NotNil}, | ||||
| 		{"nil", nil, Nil}, | ||||
| 		{"empty", []int{}, Empty}, | ||||
| 		{"notEmpty", []int{1}, NotEmpty}, | ||||
| 		{"zero", false, Zero}, | ||||
| 		{"notZero", 42, NotZero}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, tt.value) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ExampleBoolAssertionFunc() { | ||||
| 	t := &testing.T{} // provided by test | ||||
|  | ||||
| 	isOkay := func(x int) bool { | ||||
| 		return x >= 42 | ||||
| 	} | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		arg       int | ||||
| 		assertion BoolAssertionFunc | ||||
| 	}{ | ||||
| 		{"-1 is bad", -1, False}, | ||||
| 		{"42 is good", 42, True}, | ||||
| 		{"41 is bad", 41, False}, | ||||
| 		{"45 is cool", 45, True}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, isOkay(tt.arg)) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestBoolAssertionFunc(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		value     bool | ||||
| 		assertion BoolAssertionFunc | ||||
| 	}{ | ||||
| 		{"true", true, True}, | ||||
| 		{"false", false, False}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, tt.value) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func ExampleErrorAssertionFunc() { | ||||
| 	t := &testing.T{} // provided by test | ||||
|  | ||||
| 	dumbParseNum := func(input string, v interface{}) error { | ||||
| 		return json.Unmarshal([]byte(input), v) | ||||
| 	} | ||||
|  | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		arg       string | ||||
| 		assertion ErrorAssertionFunc | ||||
| 	}{ | ||||
| 		{"1.2 is number", "1.2", NoError}, | ||||
| 		{"1.2.3 not number", "1.2.3", Error}, | ||||
| 		{"true is not number", "true", Error}, | ||||
| 		{"3 is number", "3", NoError}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			var x float64 | ||||
| 			tt.assertion(t, dumbParseNum(tt.arg, &x)) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestErrorAssertionFunc(t *testing.T) { | ||||
| 	tests := []struct { | ||||
| 		name      string | ||||
| 		err       error | ||||
| 		assertion ErrorAssertionFunc | ||||
| 	}{ | ||||
| 		{"noError", nil, NoError}, | ||||
| 		{"error", errors.New("whoops"), Error}, | ||||
| 	} | ||||
|  | ||||
| 	for _, tt := range tests { | ||||
| 		t.Run(tt.name, func(t *testing.T) { | ||||
| 			tt.assertion(t, tt.err) | ||||
| 		}) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										27
									
								
								vendor/golang.org/x/crypto/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								vendor/golang.org/x/crypto/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| Copyright (c) 2009 The Go Authors. All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are | ||||
| met: | ||||
|  | ||||
|    * Redistributions of source code must retain the above copyright | ||||
| notice, this list of conditions and the following disclaimer. | ||||
|    * Redistributions in binary form must reproduce the above | ||||
| copyright notice, this list of conditions and the following disclaimer | ||||
| in the documentation and/or other materials provided with the | ||||
| distribution. | ||||
|    * Neither the name of Google Inc. nor the names of its | ||||
| contributors may be used to endorse or promote products derived from | ||||
| this software without specific prior written permission. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | ||||
| "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | ||||
| LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | ||||
| A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | ||||
| OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | ||||
| SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | ||||
| LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | ||||
| DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | ||||
| THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||||
| (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
							
								
								
									
										22
									
								
								vendor/golang.org/x/crypto/PATENTS
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								vendor/golang.org/x/crypto/PATENTS
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| Additional IP Rights Grant (Patents) | ||||
|  | ||||
| "This implementation" means the copyrightable works distributed by | ||||
| Google as part of the Go project. | ||||
|  | ||||
| Google hereby grants to You a perpetual, worldwide, non-exclusive, | ||||
| no-charge, royalty-free, irrevocable (except as stated in this section) | ||||
| patent license to make, have made, use, offer to sell, sell, import, | ||||
| transfer and otherwise run, modify and propagate the contents of this | ||||
| implementation of Go, where such license applies only to those patent | ||||
| claims, both currently owned or controlled by Google and acquired in | ||||
| the future, licensable by Google that are necessarily infringed by this | ||||
| implementation of Go.  This grant does not include claims that would be | ||||
| infringed only as a consequence of further modification of this | ||||
| implementation.  If you or your agent or exclusive licensee institute or | ||||
| order or agree to the institution of patent litigation against any | ||||
| entity (including a cross-claim or counterclaim in a lawsuit) alleging | ||||
| that this implementation of Go or any code incorporated within this | ||||
| implementation of Go constitutes direct or contributory patent | ||||
| infringement, or inducement of patent infringement, then any patent | ||||
| rights granted to you under this License for this implementation of Go | ||||
| shall terminate as of the date such litigation is filed. | ||||
							
								
								
									
										951
									
								
								vendor/golang.org/x/crypto/ssh/terminal/terminal.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										951
									
								
								vendor/golang.org/x/crypto/ssh/terminal/terminal.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,951 @@ | ||||
| // Copyright 2011 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package terminal | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"io" | ||||
| 	"sync" | ||||
| 	"unicode/utf8" | ||||
| ) | ||||
|  | ||||
| // EscapeCodes contains escape sequences that can be written to the terminal in | ||||
| // order to achieve different styles of text. | ||||
| type EscapeCodes struct { | ||||
| 	// Foreground colors | ||||
| 	Black, Red, Green, Yellow, Blue, Magenta, Cyan, White []byte | ||||
|  | ||||
| 	// Reset all attributes | ||||
| 	Reset []byte | ||||
| } | ||||
|  | ||||
| var vt100EscapeCodes = EscapeCodes{ | ||||
| 	Black:   []byte{keyEscape, '[', '3', '0', 'm'}, | ||||
| 	Red:     []byte{keyEscape, '[', '3', '1', 'm'}, | ||||
| 	Green:   []byte{keyEscape, '[', '3', '2', 'm'}, | ||||
| 	Yellow:  []byte{keyEscape, '[', '3', '3', 'm'}, | ||||
| 	Blue:    []byte{keyEscape, '[', '3', '4', 'm'}, | ||||
| 	Magenta: []byte{keyEscape, '[', '3', '5', 'm'}, | ||||
| 	Cyan:    []byte{keyEscape, '[', '3', '6', 'm'}, | ||||
| 	White:   []byte{keyEscape, '[', '3', '7', 'm'}, | ||||
|  | ||||
| 	Reset: []byte{keyEscape, '[', '0', 'm'}, | ||||
| } | ||||
|  | ||||
| // Terminal contains the state for running a VT100 terminal that is capable of | ||||
| // reading lines of input. | ||||
| type Terminal struct { | ||||
| 	// AutoCompleteCallback, if non-null, is called for each keypress with | ||||
| 	// the full input line and the current position of the cursor (in | ||||
| 	// bytes, as an index into |line|). If it returns ok=false, the key | ||||
| 	// press is processed normally. Otherwise it returns a replacement line | ||||
| 	// and the new cursor position. | ||||
| 	AutoCompleteCallback func(line string, pos int, key rune) (newLine string, newPos int, ok bool) | ||||
|  | ||||
| 	// Escape contains a pointer to the escape codes for this terminal. | ||||
| 	// It's always a valid pointer, although the escape codes themselves | ||||
| 	// may be empty if the terminal doesn't support them. | ||||
| 	Escape *EscapeCodes | ||||
|  | ||||
| 	// lock protects the terminal and the state in this object from | ||||
| 	// concurrent processing of a key press and a Write() call. | ||||
| 	lock sync.Mutex | ||||
|  | ||||
| 	c      io.ReadWriter | ||||
| 	prompt []rune | ||||
|  | ||||
| 	// line is the current line being entered. | ||||
| 	line []rune | ||||
| 	// pos is the logical position of the cursor in line | ||||
| 	pos int | ||||
| 	// echo is true if local echo is enabled | ||||
| 	echo bool | ||||
| 	// pasteActive is true iff there is a bracketed paste operation in | ||||
| 	// progress. | ||||
| 	pasteActive bool | ||||
|  | ||||
| 	// cursorX contains the current X value of the cursor where the left | ||||
| 	// edge is 0. cursorY contains the row number where the first row of | ||||
| 	// the current line is 0. | ||||
| 	cursorX, cursorY int | ||||
| 	// maxLine is the greatest value of cursorY so far. | ||||
| 	maxLine int | ||||
|  | ||||
| 	termWidth, termHeight int | ||||
|  | ||||
| 	// outBuf contains the terminal data to be sent. | ||||
| 	outBuf []byte | ||||
| 	// remainder contains the remainder of any partial key sequences after | ||||
| 	// a read. It aliases into inBuf. | ||||
| 	remainder []byte | ||||
| 	inBuf     [256]byte | ||||
|  | ||||
| 	// history contains previously entered commands so that they can be | ||||
| 	// accessed with the up and down keys. | ||||
| 	history stRingBuffer | ||||
| 	// historyIndex stores the currently accessed history entry, where zero | ||||
| 	// means the immediately previous entry. | ||||
| 	historyIndex int | ||||
| 	// When navigating up and down the history it's possible to return to | ||||
| 	// the incomplete, initial line. That value is stored in | ||||
| 	// historyPending. | ||||
| 	historyPending string | ||||
| } | ||||
|  | ||||
| // NewTerminal runs a VT100 terminal on the given ReadWriter. If the ReadWriter is | ||||
| // a local terminal, that terminal must first have been put into raw mode. | ||||
| // prompt is a string that is written at the start of each input line (i.e. | ||||
| // "> "). | ||||
| func NewTerminal(c io.ReadWriter, prompt string) *Terminal { | ||||
| 	return &Terminal{ | ||||
| 		Escape:       &vt100EscapeCodes, | ||||
| 		c:            c, | ||||
| 		prompt:       []rune(prompt), | ||||
| 		termWidth:    80, | ||||
| 		termHeight:   24, | ||||
| 		echo:         true, | ||||
| 		historyIndex: -1, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| const ( | ||||
| 	keyCtrlD     = 4 | ||||
| 	keyCtrlU     = 21 | ||||
| 	keyEnter     = '\r' | ||||
| 	keyEscape    = 27 | ||||
| 	keyBackspace = 127 | ||||
| 	keyUnknown   = 0xd800 /* UTF-16 surrogate area */ + iota | ||||
| 	keyUp | ||||
| 	keyDown | ||||
| 	keyLeft | ||||
| 	keyRight | ||||
| 	keyAltLeft | ||||
| 	keyAltRight | ||||
| 	keyHome | ||||
| 	keyEnd | ||||
| 	keyDeleteWord | ||||
| 	keyDeleteLine | ||||
| 	keyClearScreen | ||||
| 	keyPasteStart | ||||
| 	keyPasteEnd | ||||
| ) | ||||
|  | ||||
| var ( | ||||
| 	crlf       = []byte{'\r', '\n'} | ||||
| 	pasteStart = []byte{keyEscape, '[', '2', '0', '0', '~'} | ||||
| 	pasteEnd   = []byte{keyEscape, '[', '2', '0', '1', '~'} | ||||
| ) | ||||
|  | ||||
| // bytesToKey tries to parse a key sequence from b. If successful, it returns | ||||
| // the key and the remainder of the input. Otherwise it returns utf8.RuneError. | ||||
| func bytesToKey(b []byte, pasteActive bool) (rune, []byte) { | ||||
| 	if len(b) == 0 { | ||||
| 		return utf8.RuneError, nil | ||||
| 	} | ||||
|  | ||||
| 	if !pasteActive { | ||||
| 		switch b[0] { | ||||
| 		case 1: // ^A | ||||
| 			return keyHome, b[1:] | ||||
| 		case 5: // ^E | ||||
| 			return keyEnd, b[1:] | ||||
| 		case 8: // ^H | ||||
| 			return keyBackspace, b[1:] | ||||
| 		case 11: // ^K | ||||
| 			return keyDeleteLine, b[1:] | ||||
| 		case 12: // ^L | ||||
| 			return keyClearScreen, b[1:] | ||||
| 		case 23: // ^W | ||||
| 			return keyDeleteWord, b[1:] | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if b[0] != keyEscape { | ||||
| 		if !utf8.FullRune(b) { | ||||
| 			return utf8.RuneError, b | ||||
| 		} | ||||
| 		r, l := utf8.DecodeRune(b) | ||||
| 		return r, b[l:] | ||||
| 	} | ||||
|  | ||||
| 	if !pasteActive && len(b) >= 3 && b[0] == keyEscape && b[1] == '[' { | ||||
| 		switch b[2] { | ||||
| 		case 'A': | ||||
| 			return keyUp, b[3:] | ||||
| 		case 'B': | ||||
| 			return keyDown, b[3:] | ||||
| 		case 'C': | ||||
| 			return keyRight, b[3:] | ||||
| 		case 'D': | ||||
| 			return keyLeft, b[3:] | ||||
| 		case 'H': | ||||
| 			return keyHome, b[3:] | ||||
| 		case 'F': | ||||
| 			return keyEnd, b[3:] | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if !pasteActive && len(b) >= 6 && b[0] == keyEscape && b[1] == '[' && b[2] == '1' && b[3] == ';' && b[4] == '3' { | ||||
| 		switch b[5] { | ||||
| 		case 'C': | ||||
| 			return keyAltRight, b[6:] | ||||
| 		case 'D': | ||||
| 			return keyAltLeft, b[6:] | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	if !pasteActive && len(b) >= 6 && bytes.Equal(b[:6], pasteStart) { | ||||
| 		return keyPasteStart, b[6:] | ||||
| 	} | ||||
|  | ||||
| 	if pasteActive && len(b) >= 6 && bytes.Equal(b[:6], pasteEnd) { | ||||
| 		return keyPasteEnd, b[6:] | ||||
| 	} | ||||
|  | ||||
| 	// If we get here then we have a key that we don't recognise, or a | ||||
| 	// partial sequence. It's not clear how one should find the end of a | ||||
| 	// sequence without knowing them all, but it seems that [a-zA-Z~] only | ||||
| 	// appears at the end of a sequence. | ||||
| 	for i, c := range b[0:] { | ||||
| 		if c >= 'a' && c <= 'z' || c >= 'A' && c <= 'Z' || c == '~' { | ||||
| 			return keyUnknown, b[i+1:] | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return utf8.RuneError, b | ||||
| } | ||||
|  | ||||
| // queue appends data to the end of t.outBuf | ||||
| func (t *Terminal) queue(data []rune) { | ||||
| 	t.outBuf = append(t.outBuf, []byte(string(data))...) | ||||
| } | ||||
|  | ||||
| var eraseUnderCursor = []rune{' ', keyEscape, '[', 'D'} | ||||
| var space = []rune{' '} | ||||
|  | ||||
| func isPrintable(key rune) bool { | ||||
| 	isInSurrogateArea := key >= 0xd800 && key <= 0xdbff | ||||
| 	return key >= 32 && !isInSurrogateArea | ||||
| } | ||||
|  | ||||
| // moveCursorToPos appends data to t.outBuf which will move the cursor to the | ||||
| // given, logical position in the text. | ||||
| func (t *Terminal) moveCursorToPos(pos int) { | ||||
| 	if !t.echo { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	x := visualLength(t.prompt) + pos | ||||
| 	y := x / t.termWidth | ||||
| 	x = x % t.termWidth | ||||
|  | ||||
| 	up := 0 | ||||
| 	if y < t.cursorY { | ||||
| 		up = t.cursorY - y | ||||
| 	} | ||||
|  | ||||
| 	down := 0 | ||||
| 	if y > t.cursorY { | ||||
| 		down = y - t.cursorY | ||||
| 	} | ||||
|  | ||||
| 	left := 0 | ||||
| 	if x < t.cursorX { | ||||
| 		left = t.cursorX - x | ||||
| 	} | ||||
|  | ||||
| 	right := 0 | ||||
| 	if x > t.cursorX { | ||||
| 		right = x - t.cursorX | ||||
| 	} | ||||
|  | ||||
| 	t.cursorX = x | ||||
| 	t.cursorY = y | ||||
| 	t.move(up, down, left, right) | ||||
| } | ||||
|  | ||||
| func (t *Terminal) move(up, down, left, right int) { | ||||
| 	movement := make([]rune, 3*(up+down+left+right)) | ||||
| 	m := movement | ||||
| 	for i := 0; i < up; i++ { | ||||
| 		m[0] = keyEscape | ||||
| 		m[1] = '[' | ||||
| 		m[2] = 'A' | ||||
| 		m = m[3:] | ||||
| 	} | ||||
| 	for i := 0; i < down; i++ { | ||||
| 		m[0] = keyEscape | ||||
| 		m[1] = '[' | ||||
| 		m[2] = 'B' | ||||
| 		m = m[3:] | ||||
| 	} | ||||
| 	for i := 0; i < left; i++ { | ||||
| 		m[0] = keyEscape | ||||
| 		m[1] = '[' | ||||
| 		m[2] = 'D' | ||||
| 		m = m[3:] | ||||
| 	} | ||||
| 	for i := 0; i < right; i++ { | ||||
| 		m[0] = keyEscape | ||||
| 		m[1] = '[' | ||||
| 		m[2] = 'C' | ||||
| 		m = m[3:] | ||||
| 	} | ||||
|  | ||||
| 	t.queue(movement) | ||||
| } | ||||
|  | ||||
| func (t *Terminal) clearLineToRight() { | ||||
| 	op := []rune{keyEscape, '[', 'K'} | ||||
| 	t.queue(op) | ||||
| } | ||||
|  | ||||
| const maxLineLength = 4096 | ||||
|  | ||||
| func (t *Terminal) setLine(newLine []rune, newPos int) { | ||||
| 	if t.echo { | ||||
| 		t.moveCursorToPos(0) | ||||
| 		t.writeLine(newLine) | ||||
| 		for i := len(newLine); i < len(t.line); i++ { | ||||
| 			t.writeLine(space) | ||||
| 		} | ||||
| 		t.moveCursorToPos(newPos) | ||||
| 	} | ||||
| 	t.line = newLine | ||||
| 	t.pos = newPos | ||||
| } | ||||
|  | ||||
| func (t *Terminal) advanceCursor(places int) { | ||||
| 	t.cursorX += places | ||||
| 	t.cursorY += t.cursorX / t.termWidth | ||||
| 	if t.cursorY > t.maxLine { | ||||
| 		t.maxLine = t.cursorY | ||||
| 	} | ||||
| 	t.cursorX = t.cursorX % t.termWidth | ||||
|  | ||||
| 	if places > 0 && t.cursorX == 0 { | ||||
| 		// Normally terminals will advance the current position | ||||
| 		// when writing a character. But that doesn't happen | ||||
| 		// for the last character in a line. However, when | ||||
| 		// writing a character (except a new line) that causes | ||||
| 		// a line wrap, the position will be advanced two | ||||
| 		// places. | ||||
| 		// | ||||
| 		// So, if we are stopping at the end of a line, we | ||||
| 		// need to write a newline so that our cursor can be | ||||
| 		// advanced to the next line. | ||||
| 		t.outBuf = append(t.outBuf, '\r', '\n') | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func (t *Terminal) eraseNPreviousChars(n int) { | ||||
| 	if n == 0 { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	if t.pos < n { | ||||
| 		n = t.pos | ||||
| 	} | ||||
| 	t.pos -= n | ||||
| 	t.moveCursorToPos(t.pos) | ||||
|  | ||||
| 	copy(t.line[t.pos:], t.line[n+t.pos:]) | ||||
| 	t.line = t.line[:len(t.line)-n] | ||||
| 	if t.echo { | ||||
| 		t.writeLine(t.line[t.pos:]) | ||||
| 		for i := 0; i < n; i++ { | ||||
| 			t.queue(space) | ||||
| 		} | ||||
| 		t.advanceCursor(n) | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // countToLeftWord returns then number of characters from the cursor to the | ||||
| // start of the previous word. | ||||
| func (t *Terminal) countToLeftWord() int { | ||||
| 	if t.pos == 0 { | ||||
| 		return 0 | ||||
| 	} | ||||
|  | ||||
| 	pos := t.pos - 1 | ||||
| 	for pos > 0 { | ||||
| 		if t.line[pos] != ' ' { | ||||
| 			break | ||||
| 		} | ||||
| 		pos-- | ||||
| 	} | ||||
| 	for pos > 0 { | ||||
| 		if t.line[pos] == ' ' { | ||||
| 			pos++ | ||||
| 			break | ||||
| 		} | ||||
| 		pos-- | ||||
| 	} | ||||
|  | ||||
| 	return t.pos - pos | ||||
| } | ||||
|  | ||||
| // countToRightWord returns then number of characters from the cursor to the | ||||
| // start of the next word. | ||||
| func (t *Terminal) countToRightWord() int { | ||||
| 	pos := t.pos | ||||
| 	for pos < len(t.line) { | ||||
| 		if t.line[pos] == ' ' { | ||||
| 			break | ||||
| 		} | ||||
| 		pos++ | ||||
| 	} | ||||
| 	for pos < len(t.line) { | ||||
| 		if t.line[pos] != ' ' { | ||||
| 			break | ||||
| 		} | ||||
| 		pos++ | ||||
| 	} | ||||
| 	return pos - t.pos | ||||
| } | ||||
|  | ||||
| // visualLength returns the number of visible glyphs in s. | ||||
| func visualLength(runes []rune) int { | ||||
| 	inEscapeSeq := false | ||||
| 	length := 0 | ||||
|  | ||||
| 	for _, r := range runes { | ||||
| 		switch { | ||||
| 		case inEscapeSeq: | ||||
| 			if (r >= 'a' && r <= 'z') || (r >= 'A' && r <= 'Z') { | ||||
| 				inEscapeSeq = false | ||||
| 			} | ||||
| 		case r == '\x1b': | ||||
| 			inEscapeSeq = true | ||||
| 		default: | ||||
| 			length++ | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return length | ||||
| } | ||||
|  | ||||
| // handleKey processes the given key and, optionally, returns a line of text | ||||
| // that the user has entered. | ||||
| func (t *Terminal) handleKey(key rune) (line string, ok bool) { | ||||
| 	if t.pasteActive && key != keyEnter { | ||||
| 		t.addKeyToLine(key) | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	switch key { | ||||
| 	case keyBackspace: | ||||
| 		if t.pos == 0 { | ||||
| 			return | ||||
| 		} | ||||
| 		t.eraseNPreviousChars(1) | ||||
| 	case keyAltLeft: | ||||
| 		// move left by a word. | ||||
| 		t.pos -= t.countToLeftWord() | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyAltRight: | ||||
| 		// move right by a word. | ||||
| 		t.pos += t.countToRightWord() | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyLeft: | ||||
| 		if t.pos == 0 { | ||||
| 			return | ||||
| 		} | ||||
| 		t.pos-- | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyRight: | ||||
| 		if t.pos == len(t.line) { | ||||
| 			return | ||||
| 		} | ||||
| 		t.pos++ | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyHome: | ||||
| 		if t.pos == 0 { | ||||
| 			return | ||||
| 		} | ||||
| 		t.pos = 0 | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyEnd: | ||||
| 		if t.pos == len(t.line) { | ||||
| 			return | ||||
| 		} | ||||
| 		t.pos = len(t.line) | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyUp: | ||||
| 		entry, ok := t.history.NthPreviousEntry(t.historyIndex + 1) | ||||
| 		if !ok { | ||||
| 			return "", false | ||||
| 		} | ||||
| 		if t.historyIndex == -1 { | ||||
| 			t.historyPending = string(t.line) | ||||
| 		} | ||||
| 		t.historyIndex++ | ||||
| 		runes := []rune(entry) | ||||
| 		t.setLine(runes, len(runes)) | ||||
| 	case keyDown: | ||||
| 		switch t.historyIndex { | ||||
| 		case -1: | ||||
| 			return | ||||
| 		case 0: | ||||
| 			runes := []rune(t.historyPending) | ||||
| 			t.setLine(runes, len(runes)) | ||||
| 			t.historyIndex-- | ||||
| 		default: | ||||
| 			entry, ok := t.history.NthPreviousEntry(t.historyIndex - 1) | ||||
| 			if ok { | ||||
| 				t.historyIndex-- | ||||
| 				runes := []rune(entry) | ||||
| 				t.setLine(runes, len(runes)) | ||||
| 			} | ||||
| 		} | ||||
| 	case keyEnter: | ||||
| 		t.moveCursorToPos(len(t.line)) | ||||
| 		t.queue([]rune("\r\n")) | ||||
| 		line = string(t.line) | ||||
| 		ok = true | ||||
| 		t.line = t.line[:0] | ||||
| 		t.pos = 0 | ||||
| 		t.cursorX = 0 | ||||
| 		t.cursorY = 0 | ||||
| 		t.maxLine = 0 | ||||
| 	case keyDeleteWord: | ||||
| 		// Delete zero or more spaces and then one or more characters. | ||||
| 		t.eraseNPreviousChars(t.countToLeftWord()) | ||||
| 	case keyDeleteLine: | ||||
| 		// Delete everything from the current cursor position to the | ||||
| 		// end of line. | ||||
| 		for i := t.pos; i < len(t.line); i++ { | ||||
| 			t.queue(space) | ||||
| 			t.advanceCursor(1) | ||||
| 		} | ||||
| 		t.line = t.line[:t.pos] | ||||
| 		t.moveCursorToPos(t.pos) | ||||
| 	case keyCtrlD: | ||||
| 		// Erase the character under the current position. | ||||
| 		// The EOF case when the line is empty is handled in | ||||
| 		// readLine(). | ||||
| 		if t.pos < len(t.line) { | ||||
| 			t.pos++ | ||||
| 			t.eraseNPreviousChars(1) | ||||
| 		} | ||||
| 	case keyCtrlU: | ||||
| 		t.eraseNPreviousChars(t.pos) | ||||
| 	case keyClearScreen: | ||||
| 		// Erases the screen and moves the cursor to the home position. | ||||
| 		t.queue([]rune("\x1b[2J\x1b[H")) | ||||
| 		t.queue(t.prompt) | ||||
| 		t.cursorX, t.cursorY = 0, 0 | ||||
| 		t.advanceCursor(visualLength(t.prompt)) | ||||
| 		t.setLine(t.line, t.pos) | ||||
| 	default: | ||||
| 		if t.AutoCompleteCallback != nil { | ||||
| 			prefix := string(t.line[:t.pos]) | ||||
| 			suffix := string(t.line[t.pos:]) | ||||
|  | ||||
| 			t.lock.Unlock() | ||||
| 			newLine, newPos, completeOk := t.AutoCompleteCallback(prefix+suffix, len(prefix), key) | ||||
| 			t.lock.Lock() | ||||
|  | ||||
| 			if completeOk { | ||||
| 				t.setLine([]rune(newLine), utf8.RuneCount([]byte(newLine)[:newPos])) | ||||
| 				return | ||||
| 			} | ||||
| 		} | ||||
| 		if !isPrintable(key) { | ||||
| 			return | ||||
| 		} | ||||
| 		if len(t.line) == maxLineLength { | ||||
| 			return | ||||
| 		} | ||||
| 		t.addKeyToLine(key) | ||||
| 	} | ||||
| 	return | ||||
| } | ||||
|  | ||||
| // addKeyToLine inserts the given key at the current position in the current | ||||
| // line. | ||||
| func (t *Terminal) addKeyToLine(key rune) { | ||||
| 	if len(t.line) == cap(t.line) { | ||||
| 		newLine := make([]rune, len(t.line), 2*(1+len(t.line))) | ||||
| 		copy(newLine, t.line) | ||||
| 		t.line = newLine | ||||
| 	} | ||||
| 	t.line = t.line[:len(t.line)+1] | ||||
| 	copy(t.line[t.pos+1:], t.line[t.pos:]) | ||||
| 	t.line[t.pos] = key | ||||
| 	if t.echo { | ||||
| 		t.writeLine(t.line[t.pos:]) | ||||
| 	} | ||||
| 	t.pos++ | ||||
| 	t.moveCursorToPos(t.pos) | ||||
| } | ||||
|  | ||||
| func (t *Terminal) writeLine(line []rune) { | ||||
| 	for len(line) != 0 { | ||||
| 		remainingOnLine := t.termWidth - t.cursorX | ||||
| 		todo := len(line) | ||||
| 		if todo > remainingOnLine { | ||||
| 			todo = remainingOnLine | ||||
| 		} | ||||
| 		t.queue(line[:todo]) | ||||
| 		t.advanceCursor(visualLength(line[:todo])) | ||||
| 		line = line[todo:] | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // writeWithCRLF writes buf to w but replaces all occurrences of \n with \r\n. | ||||
| func writeWithCRLF(w io.Writer, buf []byte) (n int, err error) { | ||||
| 	for len(buf) > 0 { | ||||
| 		i := bytes.IndexByte(buf, '\n') | ||||
| 		todo := len(buf) | ||||
| 		if i >= 0 { | ||||
| 			todo = i | ||||
| 		} | ||||
|  | ||||
| 		var nn int | ||||
| 		nn, err = w.Write(buf[:todo]) | ||||
| 		n += nn | ||||
| 		if err != nil { | ||||
| 			return n, err | ||||
| 		} | ||||
| 		buf = buf[todo:] | ||||
|  | ||||
| 		if i >= 0 { | ||||
| 			if _, err = w.Write(crlf); err != nil { | ||||
| 				return n, err | ||||
| 			} | ||||
| 			n++ | ||||
| 			buf = buf[1:] | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return n, nil | ||||
| } | ||||
|  | ||||
| func (t *Terminal) Write(buf []byte) (n int, err error) { | ||||
| 	t.lock.Lock() | ||||
| 	defer t.lock.Unlock() | ||||
|  | ||||
| 	if t.cursorX == 0 && t.cursorY == 0 { | ||||
| 		// This is the easy case: there's nothing on the screen that we | ||||
| 		// have to move out of the way. | ||||
| 		return writeWithCRLF(t.c, buf) | ||||
| 	} | ||||
|  | ||||
| 	// We have a prompt and possibly user input on the screen. We | ||||
| 	// have to clear it first. | ||||
| 	t.move(0 /* up */, 0 /* down */, t.cursorX /* left */, 0 /* right */) | ||||
| 	t.cursorX = 0 | ||||
| 	t.clearLineToRight() | ||||
|  | ||||
| 	for t.cursorY > 0 { | ||||
| 		t.move(1 /* up */, 0, 0, 0) | ||||
| 		t.cursorY-- | ||||
| 		t.clearLineToRight() | ||||
| 	} | ||||
|  | ||||
| 	if _, err = t.c.Write(t.outBuf); err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	t.outBuf = t.outBuf[:0] | ||||
|  | ||||
| 	if n, err = writeWithCRLF(t.c, buf); err != nil { | ||||
| 		return | ||||
| 	} | ||||
|  | ||||
| 	t.writeLine(t.prompt) | ||||
| 	if t.echo { | ||||
| 		t.writeLine(t.line) | ||||
| 	} | ||||
|  | ||||
| 	t.moveCursorToPos(t.pos) | ||||
|  | ||||
| 	if _, err = t.c.Write(t.outBuf); err != nil { | ||||
| 		return | ||||
| 	} | ||||
| 	t.outBuf = t.outBuf[:0] | ||||
| 	return | ||||
| } | ||||
|  | ||||
| // ReadPassword temporarily changes the prompt and reads a password, without | ||||
| // echo, from the terminal. | ||||
| func (t *Terminal) ReadPassword(prompt string) (line string, err error) { | ||||
| 	t.lock.Lock() | ||||
| 	defer t.lock.Unlock() | ||||
|  | ||||
| 	oldPrompt := t.prompt | ||||
| 	t.prompt = []rune(prompt) | ||||
| 	t.echo = false | ||||
|  | ||||
| 	line, err = t.readLine() | ||||
|  | ||||
| 	t.prompt = oldPrompt | ||||
| 	t.echo = true | ||||
|  | ||||
| 	return | ||||
| } | ||||
|  | ||||
| // ReadLine returns a line of input from the terminal. | ||||
| func (t *Terminal) ReadLine() (line string, err error) { | ||||
| 	t.lock.Lock() | ||||
| 	defer t.lock.Unlock() | ||||
|  | ||||
| 	return t.readLine() | ||||
| } | ||||
|  | ||||
| func (t *Terminal) readLine() (line string, err error) { | ||||
| 	// t.lock must be held at this point | ||||
|  | ||||
| 	if t.cursorX == 0 && t.cursorY == 0 { | ||||
| 		t.writeLine(t.prompt) | ||||
| 		t.c.Write(t.outBuf) | ||||
| 		t.outBuf = t.outBuf[:0] | ||||
| 	} | ||||
|  | ||||
| 	lineIsPasted := t.pasteActive | ||||
|  | ||||
| 	for { | ||||
| 		rest := t.remainder | ||||
| 		lineOk := false | ||||
| 		for !lineOk { | ||||
| 			var key rune | ||||
| 			key, rest = bytesToKey(rest, t.pasteActive) | ||||
| 			if key == utf8.RuneError { | ||||
| 				break | ||||
| 			} | ||||
| 			if !t.pasteActive { | ||||
| 				if key == keyCtrlD { | ||||
| 					if len(t.line) == 0 { | ||||
| 						return "", io.EOF | ||||
| 					} | ||||
| 				} | ||||
| 				if key == keyPasteStart { | ||||
| 					t.pasteActive = true | ||||
| 					if len(t.line) == 0 { | ||||
| 						lineIsPasted = true | ||||
| 					} | ||||
| 					continue | ||||
| 				} | ||||
| 			} else if key == keyPasteEnd { | ||||
| 				t.pasteActive = false | ||||
| 				continue | ||||
| 			} | ||||
| 			if !t.pasteActive { | ||||
| 				lineIsPasted = false | ||||
| 			} | ||||
| 			line, lineOk = t.handleKey(key) | ||||
| 		} | ||||
| 		if len(rest) > 0 { | ||||
| 			n := copy(t.inBuf[:], rest) | ||||
| 			t.remainder = t.inBuf[:n] | ||||
| 		} else { | ||||
| 			t.remainder = nil | ||||
| 		} | ||||
| 		t.c.Write(t.outBuf) | ||||
| 		t.outBuf = t.outBuf[:0] | ||||
| 		if lineOk { | ||||
| 			if t.echo { | ||||
| 				t.historyIndex = -1 | ||||
| 				t.history.Add(line) | ||||
| 			} | ||||
| 			if lineIsPasted { | ||||
| 				err = ErrPasteIndicator | ||||
| 			} | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		// t.remainder is a slice at the beginning of t.inBuf | ||||
| 		// containing a partial key sequence | ||||
| 		readBuf := t.inBuf[len(t.remainder):] | ||||
| 		var n int | ||||
|  | ||||
| 		t.lock.Unlock() | ||||
| 		n, err = t.c.Read(readBuf) | ||||
| 		t.lock.Lock() | ||||
|  | ||||
| 		if err != nil { | ||||
| 			return | ||||
| 		} | ||||
|  | ||||
| 		t.remainder = t.inBuf[:n+len(t.remainder)] | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // SetPrompt sets the prompt to be used when reading subsequent lines. | ||||
| func (t *Terminal) SetPrompt(prompt string) { | ||||
| 	t.lock.Lock() | ||||
| 	defer t.lock.Unlock() | ||||
|  | ||||
| 	t.prompt = []rune(prompt) | ||||
| } | ||||
|  | ||||
| func (t *Terminal) clearAndRepaintLinePlusNPrevious(numPrevLines int) { | ||||
| 	// Move cursor to column zero at the start of the line. | ||||
| 	t.move(t.cursorY, 0, t.cursorX, 0) | ||||
| 	t.cursorX, t.cursorY = 0, 0 | ||||
| 	t.clearLineToRight() | ||||
| 	for t.cursorY < numPrevLines { | ||||
| 		// Move down a line | ||||
| 		t.move(0, 1, 0, 0) | ||||
| 		t.cursorY++ | ||||
| 		t.clearLineToRight() | ||||
| 	} | ||||
| 	// Move back to beginning. | ||||
| 	t.move(t.cursorY, 0, 0, 0) | ||||
| 	t.cursorX, t.cursorY = 0, 0 | ||||
|  | ||||
| 	t.queue(t.prompt) | ||||
| 	t.advanceCursor(visualLength(t.prompt)) | ||||
| 	t.writeLine(t.line) | ||||
| 	t.moveCursorToPos(t.pos) | ||||
| } | ||||
|  | ||||
| func (t *Terminal) SetSize(width, height int) error { | ||||
| 	t.lock.Lock() | ||||
| 	defer t.lock.Unlock() | ||||
|  | ||||
| 	if width == 0 { | ||||
| 		width = 1 | ||||
| 	} | ||||
|  | ||||
| 	oldWidth := t.termWidth | ||||
| 	t.termWidth, t.termHeight = width, height | ||||
|  | ||||
| 	switch { | ||||
| 	case width == oldWidth: | ||||
| 		// If the width didn't change then nothing else needs to be | ||||
| 		// done. | ||||
| 		return nil | ||||
| 	case len(t.line) == 0 && t.cursorX == 0 && t.cursorY == 0: | ||||
| 		// If there is nothing on current line and no prompt printed, | ||||
| 		// just do nothing | ||||
| 		return nil | ||||
| 	case width < oldWidth: | ||||
| 		// Some terminals (e.g. xterm) will truncate lines that were | ||||
| 		// too long when shinking. Others, (e.g. gnome-terminal) will | ||||
| 		// attempt to wrap them. For the former, repainting t.maxLine | ||||
| 		// works great, but that behaviour goes badly wrong in the case | ||||
| 		// of the latter because they have doubled every full line. | ||||
|  | ||||
| 		// We assume that we are working on a terminal that wraps lines | ||||
| 		// and adjust the cursor position based on every previous line | ||||
| 		// wrapping and turning into two. This causes the prompt on | ||||
| 		// xterms to move upwards, which isn't great, but it avoids a | ||||
| 		// huge mess with gnome-terminal. | ||||
| 		if t.cursorX >= t.termWidth { | ||||
| 			t.cursorX = t.termWidth - 1 | ||||
| 		} | ||||
| 		t.cursorY *= 2 | ||||
| 		t.clearAndRepaintLinePlusNPrevious(t.maxLine * 2) | ||||
| 	case width > oldWidth: | ||||
| 		// If the terminal expands then our position calculations will | ||||
| 		// be wrong in the future because we think the cursor is | ||||
| 		// |t.pos| chars into the string, but there will be a gap at | ||||
| 		// the end of any wrapped line. | ||||
| 		// | ||||
| 		// But the position will actually be correct until we move, so | ||||
| 		// we can move back to the beginning and repaint everything. | ||||
| 		t.clearAndRepaintLinePlusNPrevious(t.maxLine) | ||||
| 	} | ||||
|  | ||||
| 	_, err := t.c.Write(t.outBuf) | ||||
| 	t.outBuf = t.outBuf[:0] | ||||
| 	return err | ||||
| } | ||||
|  | ||||
| type pasteIndicatorError struct{} | ||||
|  | ||||
| func (pasteIndicatorError) Error() string { | ||||
| 	return "terminal: ErrPasteIndicator not correctly handled" | ||||
| } | ||||
|  | ||||
| // ErrPasteIndicator may be returned from ReadLine as the error, in addition | ||||
| // to valid line data. It indicates that bracketed paste mode is enabled and | ||||
| // that the returned line consists only of pasted data. Programs may wish to | ||||
| // interpret pasted data more literally than typed data. | ||||
| var ErrPasteIndicator = pasteIndicatorError{} | ||||
|  | ||||
| // SetBracketedPasteMode requests that the terminal bracket paste operations | ||||
| // with markers. Not all terminals support this but, if it is supported, then | ||||
| // enabling this mode will stop any autocomplete callback from running due to | ||||
| // pastes. Additionally, any lines that are completely pasted will be returned | ||||
| // from ReadLine with the error set to ErrPasteIndicator. | ||||
| func (t *Terminal) SetBracketedPasteMode(on bool) { | ||||
| 	if on { | ||||
| 		io.WriteString(t.c, "\x1b[?2004h") | ||||
| 	} else { | ||||
| 		io.WriteString(t.c, "\x1b[?2004l") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // stRingBuffer is a ring buffer of strings. | ||||
| type stRingBuffer struct { | ||||
| 	// entries contains max elements. | ||||
| 	entries []string | ||||
| 	max     int | ||||
| 	// head contains the index of the element most recently added to the ring. | ||||
| 	head int | ||||
| 	// size contains the number of elements in the ring. | ||||
| 	size int | ||||
| } | ||||
|  | ||||
| func (s *stRingBuffer) Add(a string) { | ||||
| 	if s.entries == nil { | ||||
| 		const defaultNumEntries = 100 | ||||
| 		s.entries = make([]string, defaultNumEntries) | ||||
| 		s.max = defaultNumEntries | ||||
| 	} | ||||
|  | ||||
| 	s.head = (s.head + 1) % s.max | ||||
| 	s.entries[s.head] = a | ||||
| 	if s.size < s.max { | ||||
| 		s.size++ | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // NthPreviousEntry returns the value passed to the nth previous call to Add. | ||||
| // If n is zero then the immediately prior value is returned, if one, then the | ||||
| // next most recent, and so on. If such an element doesn't exist then ok is | ||||
| // false. | ||||
| func (s *stRingBuffer) NthPreviousEntry(n int) (value string, ok bool) { | ||||
| 	if n >= s.size { | ||||
| 		return "", false | ||||
| 	} | ||||
| 	index := s.head - n | ||||
| 	if index < 0 { | ||||
| 		index += s.max | ||||
| 	} | ||||
| 	return s.entries[index], true | ||||
| } | ||||
|  | ||||
| // readPasswordLine reads from reader until it finds \n or io.EOF. | ||||
| // The slice returned does not include the \n. | ||||
| // readPasswordLine also ignores any \r it finds. | ||||
| func readPasswordLine(reader io.Reader) ([]byte, error) { | ||||
| 	var buf [1]byte | ||||
| 	var ret []byte | ||||
|  | ||||
| 	for { | ||||
| 		n, err := reader.Read(buf[:]) | ||||
| 		if n > 0 { | ||||
| 			switch buf[0] { | ||||
| 			case '\n': | ||||
| 				return ret, nil | ||||
| 			case '\r': | ||||
| 				// remove \r from passwords on Windows | ||||
| 			default: | ||||
| 				ret = append(ret, buf[0]) | ||||
| 			} | ||||
| 			continue | ||||
| 		} | ||||
| 		if err != nil { | ||||
| 			if err == io.EOF && len(ret) > 0 { | ||||
| 				return ret, nil | ||||
| 			} | ||||
| 			return ret, err | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										358
									
								
								vendor/golang.org/x/crypto/ssh/terminal/terminal_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										358
									
								
								vendor/golang.org/x/crypto/ssh/terminal/terminal_test.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,358 @@ | ||||
| // Copyright 2011 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build aix darwin dragonfly freebsd linux,!appengine netbsd openbsd windows plan9 solaris | ||||
|  | ||||
| package terminal | ||||
|  | ||||
| import ( | ||||
| 	"bytes" | ||||
| 	"io" | ||||
| 	"os" | ||||
| 	"runtime" | ||||
| 	"testing" | ||||
| ) | ||||
|  | ||||
| type MockTerminal struct { | ||||
| 	toSend       []byte | ||||
| 	bytesPerRead int | ||||
| 	received     []byte | ||||
| } | ||||
|  | ||||
| func (c *MockTerminal) Read(data []byte) (n int, err error) { | ||||
| 	n = len(data) | ||||
| 	if n == 0 { | ||||
| 		return | ||||
| 	} | ||||
| 	if n > len(c.toSend) { | ||||
| 		n = len(c.toSend) | ||||
| 	} | ||||
| 	if n == 0 { | ||||
| 		return 0, io.EOF | ||||
| 	} | ||||
| 	if c.bytesPerRead > 0 && n > c.bytesPerRead { | ||||
| 		n = c.bytesPerRead | ||||
| 	} | ||||
| 	copy(data, c.toSend[:n]) | ||||
| 	c.toSend = c.toSend[n:] | ||||
| 	return | ||||
| } | ||||
|  | ||||
| func (c *MockTerminal) Write(data []byte) (n int, err error) { | ||||
| 	c.received = append(c.received, data...) | ||||
| 	return len(data), nil | ||||
| } | ||||
|  | ||||
| func TestClose(t *testing.T) { | ||||
| 	c := &MockTerminal{} | ||||
| 	ss := NewTerminal(c, "> ") | ||||
| 	line, err := ss.ReadLine() | ||||
| 	if line != "" { | ||||
| 		t.Errorf("Expected empty line but got: %s", line) | ||||
| 	} | ||||
| 	if err != io.EOF { | ||||
| 		t.Errorf("Error should have been EOF but got: %s", err) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| var keyPressTests = []struct { | ||||
| 	in             string | ||||
| 	line           string | ||||
| 	err            error | ||||
| 	throwAwayLines int | ||||
| }{ | ||||
| 	{ | ||||
| 		err: io.EOF, | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "\r", | ||||
| 		line: "", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "foo\r", | ||||
| 		line: "foo", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a\x1b[Cb\r", // right | ||||
| 		line: "ab", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a\x1b[Db\r", // left | ||||
| 		line: "ba", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a\177b\r", // backspace | ||||
| 		line: "b", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in: "\x1b[A\r", // up | ||||
| 	}, | ||||
| 	{ | ||||
| 		in: "\x1b[B\r", // down | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "line\x1b[A\x1b[B\r", // up then down | ||||
| 		line: "line", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:             "line1\rline2\x1b[A\r", // recall previous line. | ||||
| 		line:           "line1", | ||||
| 		throwAwayLines: 1, | ||||
| 	}, | ||||
| 	{ | ||||
| 		// recall two previous lines and append. | ||||
| 		in:             "line1\rline2\rline3\x1b[A\x1b[Axxx\r", | ||||
| 		line:           "line1xxx", | ||||
| 		throwAwayLines: 2, | ||||
| 	}, | ||||
| 	{ | ||||
| 		// Ctrl-A to move to beginning of line followed by ^K to kill | ||||
| 		// line. | ||||
| 		in:   "a b \001\013\r", | ||||
| 		line: "", | ||||
| 	}, | ||||
| 	{ | ||||
| 		// Ctrl-A to move to beginning of line, Ctrl-E to move to end, | ||||
| 		// finally ^K to kill nothing. | ||||
| 		in:   "a b \001\005\013\r", | ||||
| 		line: "a b ", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "\027\r", | ||||
| 		line: "", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a\027\r", | ||||
| 		line: "", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a \027\r", | ||||
| 		line: "", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a b\027\r", | ||||
| 		line: "a ", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a b \027\r", | ||||
| 		line: "a ", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "one two thr\x1b[D\027\r", | ||||
| 		line: "one two r", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "\013\r", | ||||
| 		line: "", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "a\013\r", | ||||
| 		line: "a", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "ab\x1b[D\013\r", | ||||
| 		line: "a", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:   "Ξεσκεπάζω\r", | ||||
| 		line: "Ξεσκεπάζω", | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:             "£\r\x1b[A\177\r", // non-ASCII char, enter, up, backspace. | ||||
| 		line:           "", | ||||
| 		throwAwayLines: 1, | ||||
| 	}, | ||||
| 	{ | ||||
| 		in:             "£\r££\x1b[A\x1b[B\177\r", // non-ASCII char, enter, 2x non-ASCII, up, down, backspace, enter. | ||||
| 		line:           "£", | ||||
| 		throwAwayLines: 1, | ||||
| 	}, | ||||
| 	{ | ||||
| 		// Ctrl-D at the end of the line should be ignored. | ||||
| 		in:   "a\004\r", | ||||
| 		line: "a", | ||||
| 	}, | ||||
| 	{ | ||||
| 		// a, b, left, Ctrl-D should erase the b. | ||||
| 		in:   "ab\x1b[D\004\r", | ||||
| 		line: "a", | ||||
| 	}, | ||||
| 	{ | ||||
| 		// a, b, c, d, left, left, ^U should erase to the beginning of | ||||
| 		// the line. | ||||
| 		in:   "abcd\x1b[D\x1b[D\025\r", | ||||
| 		line: "cd", | ||||
| 	}, | ||||
| 	{ | ||||
| 		// Bracketed paste mode: control sequences should be returned | ||||
| 		// verbatim in paste mode. | ||||
| 		in:   "abc\x1b[200~de\177f\x1b[201~\177\r", | ||||
| 		line: "abcde\177", | ||||
| 	}, | ||||
| 	{ | ||||
| 		// Enter in bracketed paste mode should still work. | ||||
| 		in:             "abc\x1b[200~d\refg\x1b[201~h\r", | ||||
| 		line:           "efgh", | ||||
| 		throwAwayLines: 1, | ||||
| 	}, | ||||
| 	{ | ||||
| 		// Lines consisting entirely of pasted data should be indicated as such. | ||||
| 		in:   "\x1b[200~a\r", | ||||
| 		line: "a", | ||||
| 		err:  ErrPasteIndicator, | ||||
| 	}, | ||||
| } | ||||
|  | ||||
| func TestKeyPresses(t *testing.T) { | ||||
| 	for i, test := range keyPressTests { | ||||
| 		for j := 1; j < len(test.in); j++ { | ||||
| 			c := &MockTerminal{ | ||||
| 				toSend:       []byte(test.in), | ||||
| 				bytesPerRead: j, | ||||
| 			} | ||||
| 			ss := NewTerminal(c, "> ") | ||||
| 			for k := 0; k < test.throwAwayLines; k++ { | ||||
| 				_, err := ss.ReadLine() | ||||
| 				if err != nil { | ||||
| 					t.Errorf("Throwaway line %d from test %d resulted in error: %s", k, i, err) | ||||
| 				} | ||||
| 			} | ||||
| 			line, err := ss.ReadLine() | ||||
| 			if line != test.line { | ||||
| 				t.Errorf("Line resulting from test %d (%d bytes per read) was '%s', expected '%s'", i, j, line, test.line) | ||||
| 				break | ||||
| 			} | ||||
| 			if err != test.err { | ||||
| 				t.Errorf("Error resulting from test %d (%d bytes per read) was '%v', expected '%v'", i, j, err, test.err) | ||||
| 				break | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestPasswordNotSaved(t *testing.T) { | ||||
| 	c := &MockTerminal{ | ||||
| 		toSend:       []byte("password\r\x1b[A\r"), | ||||
| 		bytesPerRead: 1, | ||||
| 	} | ||||
| 	ss := NewTerminal(c, "> ") | ||||
| 	pw, _ := ss.ReadPassword("> ") | ||||
| 	if pw != "password" { | ||||
| 		t.Fatalf("failed to read password, got %s", pw) | ||||
| 	} | ||||
| 	line, _ := ss.ReadLine() | ||||
| 	if len(line) > 0 { | ||||
| 		t.Fatalf("password was saved in history") | ||||
| 	} | ||||
| } | ||||
|  | ||||
| var setSizeTests = []struct { | ||||
| 	width, height int | ||||
| }{ | ||||
| 	{40, 13}, | ||||
| 	{80, 24}, | ||||
| 	{132, 43}, | ||||
| } | ||||
|  | ||||
| func TestTerminalSetSize(t *testing.T) { | ||||
| 	for _, setSize := range setSizeTests { | ||||
| 		c := &MockTerminal{ | ||||
| 			toSend:       []byte("password\r\x1b[A\r"), | ||||
| 			bytesPerRead: 1, | ||||
| 		} | ||||
| 		ss := NewTerminal(c, "> ") | ||||
| 		ss.SetSize(setSize.width, setSize.height) | ||||
| 		pw, _ := ss.ReadPassword("Password: ") | ||||
| 		if pw != "password" { | ||||
| 			t.Fatalf("failed to read password, got %s", pw) | ||||
| 		} | ||||
| 		if string(c.received) != "Password: \r\n" { | ||||
| 			t.Errorf("failed to set the temporary prompt expected %q, got %q", "Password: ", c.received) | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestReadPasswordLineEnd(t *testing.T) { | ||||
| 	var tests = []struct { | ||||
| 		input string | ||||
| 		want  string | ||||
| 	}{ | ||||
| 		{"\n", ""}, | ||||
| 		{"\r\n", ""}, | ||||
| 		{"test\r\n", "test"}, | ||||
| 		{"testtesttesttes\n", "testtesttesttes"}, | ||||
| 		{"testtesttesttes\r\n", "testtesttesttes"}, | ||||
| 		{"testtesttesttesttest\n", "testtesttesttesttest"}, | ||||
| 		{"testtesttesttesttest\r\n", "testtesttesttesttest"}, | ||||
| 	} | ||||
| 	for _, test := range tests { | ||||
| 		buf := new(bytes.Buffer) | ||||
| 		if _, err := buf.WriteString(test.input); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
|  | ||||
| 		have, err := readPasswordLine(buf) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("readPasswordLine(%q) failed: %v", test.input, err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if string(have) != test.want { | ||||
| 			t.Errorf("readPasswordLine(%q) returns %q, but %q is expected", test.input, string(have), test.want) | ||||
| 			continue | ||||
| 		} | ||||
|  | ||||
| 		if _, err = buf.WriteString(test.input); err != nil { | ||||
| 			t.Fatal(err) | ||||
| 		} | ||||
| 		have, err = readPasswordLine(buf) | ||||
| 		if err != nil { | ||||
| 			t.Errorf("readPasswordLine(%q) failed: %v", test.input, err) | ||||
| 			continue | ||||
| 		} | ||||
| 		if string(have) != test.want { | ||||
| 			t.Errorf("readPasswordLine(%q) returns %q, but %q is expected", test.input, string(have), test.want) | ||||
| 			continue | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestMakeRawState(t *testing.T) { | ||||
| 	fd := int(os.Stdout.Fd()) | ||||
| 	if !IsTerminal(fd) { | ||||
| 		t.Skip("stdout is not a terminal; skipping test") | ||||
| 	} | ||||
|  | ||||
| 	st, err := GetState(fd) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("failed to get terminal state from GetState: %s", err) | ||||
| 	} | ||||
|  | ||||
| 	if runtime.GOOS == "darwin" && (runtime.GOARCH == "arm" || runtime.GOARCH == "arm64") { | ||||
| 		t.Skip("MakeRaw not allowed on iOS; skipping test") | ||||
| 	} | ||||
|  | ||||
| 	defer Restore(fd, st) | ||||
| 	raw, err := MakeRaw(fd) | ||||
| 	if err != nil { | ||||
| 		t.Fatalf("failed to get terminal state from MakeRaw: %s", err) | ||||
| 	} | ||||
|  | ||||
| 	if *st != *raw { | ||||
| 		t.Errorf("states do not match; was %v, expected %v", raw, st) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func TestOutputNewlines(t *testing.T) { | ||||
| 	// \n should be changed to \r\n in terminal output. | ||||
| 	buf := new(bytes.Buffer) | ||||
| 	term := NewTerminal(buf, ">") | ||||
|  | ||||
| 	term.Write([]byte("1\n2\n")) | ||||
| 	output := string(buf.Bytes()) | ||||
| 	const expected = "1\r\n2\r\n" | ||||
|  | ||||
| 	if output != expected { | ||||
| 		t.Errorf("incorrect output: was %q, expected %q", output, expected) | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										114
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | ||||
| // Copyright 2011 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build aix darwin dragonfly freebsd linux,!appengine netbsd openbsd | ||||
|  | ||||
| // Package terminal provides support functions for dealing with terminals, as | ||||
| // commonly found on UNIX systems. | ||||
| // | ||||
| // Putting a terminal into raw mode is the most common requirement: | ||||
| // | ||||
| // 	oldState, err := terminal.MakeRaw(0) | ||||
| // 	if err != nil { | ||||
| // 	        panic(err) | ||||
| // 	} | ||||
| // 	defer terminal.Restore(0, oldState) | ||||
| package terminal // import "golang.org/x/crypto/ssh/terminal" | ||||
|  | ||||
| import ( | ||||
| 	"golang.org/x/sys/unix" | ||||
| ) | ||||
|  | ||||
| // State contains the state of a terminal. | ||||
| type State struct { | ||||
| 	termios unix.Termios | ||||
| } | ||||
|  | ||||
| // IsTerminal returns whether the given file descriptor is a terminal. | ||||
| func IsTerminal(fd int) bool { | ||||
| 	_, err := unix.IoctlGetTermios(fd, ioctlReadTermios) | ||||
| 	return err == nil | ||||
| } | ||||
|  | ||||
| // MakeRaw put the terminal connected to the given file descriptor into raw | ||||
| // mode and returns the previous state of the terminal so that it can be | ||||
| // restored. | ||||
| func MakeRaw(fd int) (*State, error) { | ||||
| 	termios, err := unix.IoctlGetTermios(fd, ioctlReadTermios) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	oldState := State{termios: *termios} | ||||
|  | ||||
| 	// This attempts to replicate the behaviour documented for cfmakeraw in | ||||
| 	// the termios(3) manpage. | ||||
| 	termios.Iflag &^= unix.IGNBRK | unix.BRKINT | unix.PARMRK | unix.ISTRIP | unix.INLCR | unix.IGNCR | unix.ICRNL | unix.IXON | ||||
| 	termios.Oflag &^= unix.OPOST | ||||
| 	termios.Lflag &^= unix.ECHO | unix.ECHONL | unix.ICANON | unix.ISIG | unix.IEXTEN | ||||
| 	termios.Cflag &^= unix.CSIZE | unix.PARENB | ||||
| 	termios.Cflag |= unix.CS8 | ||||
| 	termios.Cc[unix.VMIN] = 1 | ||||
| 	termios.Cc[unix.VTIME] = 0 | ||||
| 	if err := unix.IoctlSetTermios(fd, ioctlWriteTermios, termios); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &oldState, nil | ||||
| } | ||||
|  | ||||
| // GetState returns the current state of a terminal which may be useful to | ||||
| // restore the terminal after a signal. | ||||
| func GetState(fd int) (*State, error) { | ||||
| 	termios, err := unix.IoctlGetTermios(fd, ioctlReadTermios) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &State{termios: *termios}, nil | ||||
| } | ||||
|  | ||||
| // Restore restores the terminal connected to the given file descriptor to a | ||||
| // previous state. | ||||
| func Restore(fd int, state *State) error { | ||||
| 	return unix.IoctlSetTermios(fd, ioctlWriteTermios, &state.termios) | ||||
| } | ||||
|  | ||||
| // GetSize returns the dimensions of the given terminal. | ||||
| func GetSize(fd int) (width, height int, err error) { | ||||
| 	ws, err := unix.IoctlGetWinsize(fd, unix.TIOCGWINSZ) | ||||
| 	if err != nil { | ||||
| 		return -1, -1, err | ||||
| 	} | ||||
| 	return int(ws.Col), int(ws.Row), nil | ||||
| } | ||||
|  | ||||
| // passwordReader is an io.Reader that reads from a specific file descriptor. | ||||
| type passwordReader int | ||||
|  | ||||
| func (r passwordReader) Read(buf []byte) (int, error) { | ||||
| 	return unix.Read(int(r), buf) | ||||
| } | ||||
|  | ||||
| // ReadPassword reads a line of input from a terminal without local echo.  This | ||||
| // is commonly used for inputting passwords and other sensitive data. The slice | ||||
| // returned does not include the \n. | ||||
| func ReadPassword(fd int) ([]byte, error) { | ||||
| 	termios, err := unix.IoctlGetTermios(fd, ioctlReadTermios) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	newState := *termios | ||||
| 	newState.Lflag &^= unix.ECHO | ||||
| 	newState.Lflag |= unix.ICANON | unix.ISIG | ||||
| 	newState.Iflag |= unix.ICRNL | ||||
| 	if err := unix.IoctlSetTermios(fd, ioctlWriteTermios, &newState); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	defer unix.IoctlSetTermios(fd, ioctlWriteTermios, termios) | ||||
|  | ||||
| 	return readPasswordLine(passwordReader(fd)) | ||||
| } | ||||
							
								
								
									
										12
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_aix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_aix.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| // Copyright 2018 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build aix | ||||
|  | ||||
| package terminal | ||||
|  | ||||
| import "golang.org/x/sys/unix" | ||||
|  | ||||
| const ioctlReadTermios = unix.TCGETS | ||||
| const ioctlWriteTermios = unix.TCSETS | ||||
							
								
								
									
										12
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_bsd.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_bsd.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| // Copyright 2013 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build darwin dragonfly freebsd netbsd openbsd | ||||
|  | ||||
| package terminal | ||||
|  | ||||
| import "golang.org/x/sys/unix" | ||||
|  | ||||
| const ioctlReadTermios = unix.TIOCGETA | ||||
| const ioctlWriteTermios = unix.TIOCSETA | ||||
							
								
								
									
										10
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_linux.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_linux.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| // Copyright 2013 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| package terminal | ||||
|  | ||||
| import "golang.org/x/sys/unix" | ||||
|  | ||||
| const ioctlReadTermios = unix.TCGETS | ||||
| const ioctlWriteTermios = unix.TCSETS | ||||
							
								
								
									
										58
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_plan9.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_plan9.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | ||||
| // Copyright 2016 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // Package terminal provides support functions for dealing with terminals, as | ||||
| // commonly found on UNIX systems. | ||||
| // | ||||
| // Putting a terminal into raw mode is the most common requirement: | ||||
| // | ||||
| // 	oldState, err := terminal.MakeRaw(0) | ||||
| // 	if err != nil { | ||||
| // 	        panic(err) | ||||
| // 	} | ||||
| // 	defer terminal.Restore(0, oldState) | ||||
| package terminal | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"runtime" | ||||
| ) | ||||
|  | ||||
| type State struct{} | ||||
|  | ||||
| // IsTerminal returns whether the given file descriptor is a terminal. | ||||
| func IsTerminal(fd int) bool { | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| // MakeRaw put the terminal connected to the given file descriptor into raw | ||||
| // mode and returns the previous state of the terminal so that it can be | ||||
| // restored. | ||||
| func MakeRaw(fd int) (*State, error) { | ||||
| 	return nil, fmt.Errorf("terminal: MakeRaw not implemented on %s/%s", runtime.GOOS, runtime.GOARCH) | ||||
| } | ||||
|  | ||||
| // GetState returns the current state of a terminal which may be useful to | ||||
| // restore the terminal after a signal. | ||||
| func GetState(fd int) (*State, error) { | ||||
| 	return nil, fmt.Errorf("terminal: GetState not implemented on %s/%s", runtime.GOOS, runtime.GOARCH) | ||||
| } | ||||
|  | ||||
| // Restore restores the terminal connected to the given file descriptor to a | ||||
| // previous state. | ||||
| func Restore(fd int, state *State) error { | ||||
| 	return fmt.Errorf("terminal: Restore not implemented on %s/%s", runtime.GOOS, runtime.GOARCH) | ||||
| } | ||||
|  | ||||
| // GetSize returns the dimensions of the given terminal. | ||||
| func GetSize(fd int) (width, height int, err error) { | ||||
| 	return 0, 0, fmt.Errorf("terminal: GetSize not implemented on %s/%s", runtime.GOOS, runtime.GOARCH) | ||||
| } | ||||
|  | ||||
| // ReadPassword reads a line of input from a terminal without local echo.  This | ||||
| // is commonly used for inputting passwords and other sensitive data. The slice | ||||
| // returned does not include the \n. | ||||
| func ReadPassword(fd int) ([]byte, error) { | ||||
| 	return nil, fmt.Errorf("terminal: ReadPassword not implemented on %s/%s", runtime.GOOS, runtime.GOARCH) | ||||
| } | ||||
							
								
								
									
										124
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_solaris.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_solaris.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| // Copyright 2015 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build solaris | ||||
|  | ||||
| package terminal // import "golang.org/x/crypto/ssh/terminal" | ||||
|  | ||||
| import ( | ||||
| 	"golang.org/x/sys/unix" | ||||
| 	"io" | ||||
| 	"syscall" | ||||
| ) | ||||
|  | ||||
| // State contains the state of a terminal. | ||||
| type State struct { | ||||
| 	termios unix.Termios | ||||
| } | ||||
|  | ||||
| // IsTerminal returns whether the given file descriptor is a terminal. | ||||
| func IsTerminal(fd int) bool { | ||||
| 	_, err := unix.IoctlGetTermio(fd, unix.TCGETA) | ||||
| 	return err == nil | ||||
| } | ||||
|  | ||||
| // ReadPassword reads a line of input from a terminal without local echo.  This | ||||
| // is commonly used for inputting passwords and other sensitive data. The slice | ||||
| // returned does not include the \n. | ||||
| func ReadPassword(fd int) ([]byte, error) { | ||||
| 	// see also: http://src.illumos.org/source/xref/illumos-gate/usr/src/lib/libast/common/uwin/getpass.c | ||||
| 	val, err := unix.IoctlGetTermios(fd, unix.TCGETS) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	oldState := *val | ||||
|  | ||||
| 	newState := oldState | ||||
| 	newState.Lflag &^= syscall.ECHO | ||||
| 	newState.Lflag |= syscall.ICANON | syscall.ISIG | ||||
| 	newState.Iflag |= syscall.ICRNL | ||||
| 	err = unix.IoctlSetTermios(fd, unix.TCSETS, &newState) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	defer unix.IoctlSetTermios(fd, unix.TCSETS, &oldState) | ||||
|  | ||||
| 	var buf [16]byte | ||||
| 	var ret []byte | ||||
| 	for { | ||||
| 		n, err := syscall.Read(fd, buf[:]) | ||||
| 		if err != nil { | ||||
| 			return nil, err | ||||
| 		} | ||||
| 		if n == 0 { | ||||
| 			if len(ret) == 0 { | ||||
| 				return nil, io.EOF | ||||
| 			} | ||||
| 			break | ||||
| 		} | ||||
| 		if buf[n-1] == '\n' { | ||||
| 			n-- | ||||
| 		} | ||||
| 		ret = append(ret, buf[:n]...) | ||||
| 		if n < len(buf) { | ||||
| 			break | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return ret, nil | ||||
| } | ||||
|  | ||||
| // MakeRaw puts the terminal connected to the given file descriptor into raw | ||||
| // mode and returns the previous state of the terminal so that it can be | ||||
| // restored. | ||||
| // see http://cr.illumos.org/~webrev/andy_js/1060/ | ||||
| func MakeRaw(fd int) (*State, error) { | ||||
| 	termios, err := unix.IoctlGetTermios(fd, unix.TCGETS) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	oldState := State{termios: *termios} | ||||
|  | ||||
| 	termios.Iflag &^= unix.IGNBRK | unix.BRKINT | unix.PARMRK | unix.ISTRIP | unix.INLCR | unix.IGNCR | unix.ICRNL | unix.IXON | ||||
| 	termios.Oflag &^= unix.OPOST | ||||
| 	termios.Lflag &^= unix.ECHO | unix.ECHONL | unix.ICANON | unix.ISIG | unix.IEXTEN | ||||
| 	termios.Cflag &^= unix.CSIZE | unix.PARENB | ||||
| 	termios.Cflag |= unix.CS8 | ||||
| 	termios.Cc[unix.VMIN] = 1 | ||||
| 	termios.Cc[unix.VTIME] = 0 | ||||
|  | ||||
| 	if err := unix.IoctlSetTermios(fd, unix.TCSETS, termios); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &oldState, nil | ||||
| } | ||||
|  | ||||
| // Restore restores the terminal connected to the given file descriptor to a | ||||
| // previous state. | ||||
| func Restore(fd int, oldState *State) error { | ||||
| 	return unix.IoctlSetTermios(fd, unix.TCSETS, &oldState.termios) | ||||
| } | ||||
|  | ||||
| // GetState returns the current state of a terminal which may be useful to | ||||
| // restore the terminal after a signal. | ||||
| func GetState(fd int) (*State, error) { | ||||
| 	termios, err := unix.IoctlGetTermios(fd, unix.TCGETS) | ||||
| 	if err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &State{termios: *termios}, nil | ||||
| } | ||||
|  | ||||
| // GetSize returns the dimensions of the given terminal. | ||||
| func GetSize(fd int) (width, height int, err error) { | ||||
| 	ws, err := unix.IoctlGetWinsize(fd, unix.TIOCGWINSZ) | ||||
| 	if err != nil { | ||||
| 		return 0, 0, err | ||||
| 	} | ||||
| 	return int(ws.Col), int(ws.Row), nil | ||||
| } | ||||
							
								
								
									
										103
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										103
									
								
								vendor/golang.org/x/crypto/ssh/terminal/util_windows.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,103 @@ | ||||
| // Copyright 2011 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build windows | ||||
|  | ||||
| // Package terminal provides support functions for dealing with terminals, as | ||||
| // commonly found on UNIX systems. | ||||
| // | ||||
| // Putting a terminal into raw mode is the most common requirement: | ||||
| // | ||||
| // 	oldState, err := terminal.MakeRaw(0) | ||||
| // 	if err != nil { | ||||
| // 	        panic(err) | ||||
| // 	} | ||||
| // 	defer terminal.Restore(0, oldState) | ||||
| package terminal | ||||
|  | ||||
| import ( | ||||
| 	"os" | ||||
|  | ||||
| 	"golang.org/x/sys/windows" | ||||
| ) | ||||
|  | ||||
| type State struct { | ||||
| 	mode uint32 | ||||
| } | ||||
|  | ||||
| // IsTerminal returns whether the given file descriptor is a terminal. | ||||
| func IsTerminal(fd int) bool { | ||||
| 	var st uint32 | ||||
| 	err := windows.GetConsoleMode(windows.Handle(fd), &st) | ||||
| 	return err == nil | ||||
| } | ||||
|  | ||||
| // MakeRaw put the terminal connected to the given file descriptor into raw | ||||
| // mode and returns the previous state of the terminal so that it can be | ||||
| // restored. | ||||
| func MakeRaw(fd int) (*State, error) { | ||||
| 	var st uint32 | ||||
| 	if err := windows.GetConsoleMode(windows.Handle(fd), &st); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	raw := st &^ (windows.ENABLE_ECHO_INPUT | windows.ENABLE_PROCESSED_INPUT | windows.ENABLE_LINE_INPUT | windows.ENABLE_PROCESSED_OUTPUT) | ||||
| 	if err := windows.SetConsoleMode(windows.Handle(fd), raw); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &State{st}, nil | ||||
| } | ||||
|  | ||||
| // GetState returns the current state of a terminal which may be useful to | ||||
| // restore the terminal after a signal. | ||||
| func GetState(fd int) (*State, error) { | ||||
| 	var st uint32 | ||||
| 	if err := windows.GetConsoleMode(windows.Handle(fd), &st); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	return &State{st}, nil | ||||
| } | ||||
|  | ||||
| // Restore restores the terminal connected to the given file descriptor to a | ||||
| // previous state. | ||||
| func Restore(fd int, state *State) error { | ||||
| 	return windows.SetConsoleMode(windows.Handle(fd), state.mode) | ||||
| } | ||||
|  | ||||
| // GetSize returns the dimensions of the given terminal. | ||||
| func GetSize(fd int) (width, height int, err error) { | ||||
| 	var info windows.ConsoleScreenBufferInfo | ||||
| 	if err := windows.GetConsoleScreenBufferInfo(windows.Handle(fd), &info); err != nil { | ||||
| 		return 0, 0, err | ||||
| 	} | ||||
| 	return int(info.Size.X), int(info.Size.Y), nil | ||||
| } | ||||
|  | ||||
| // ReadPassword reads a line of input from a terminal without local echo.  This | ||||
| // is commonly used for inputting passwords and other sensitive data. The slice | ||||
| // returned does not include the \n. | ||||
| func ReadPassword(fd int) ([]byte, error) { | ||||
| 	var st uint32 | ||||
| 	if err := windows.GetConsoleMode(windows.Handle(fd), &st); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| 	old := st | ||||
|  | ||||
| 	st &^= (windows.ENABLE_ECHO_INPUT) | ||||
| 	st |= (windows.ENABLE_PROCESSED_INPUT | windows.ENABLE_LINE_INPUT | windows.ENABLE_PROCESSED_OUTPUT) | ||||
| 	if err := windows.SetConsoleMode(windows.Handle(fd), st); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	defer windows.SetConsoleMode(windows.Handle(fd), old) | ||||
|  | ||||
| 	var h windows.Handle | ||||
| 	p, _ := windows.GetCurrentProcess() | ||||
| 	if err := windows.DuplicateHandle(p, windows.Handle(fd), p, &h, 0, false, windows.DUPLICATE_SAME_ACCESS); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	f := os.NewFile(uintptr(h), "stdin") | ||||
| 	defer f.Close() | ||||
| 	return readPasswordLine(f) | ||||
| } | ||||
							
								
								
									
										27
									
								
								vendor/golang.org/x/sys/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								vendor/golang.org/x/sys/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| Copyright (c) 2009 The Go Authors. All rights reserved. | ||||
|  | ||||
| Redistribution and use in source and binary forms, with or without | ||||
| modification, are permitted provided that the following conditions are | ||||
| met: | ||||
|  | ||||
|    * Redistributions of source code must retain the above copyright | ||||
| notice, this list of conditions and the following disclaimer. | ||||
|    * Redistributions in binary form must reproduce the above | ||||
| copyright notice, this list of conditions and the following disclaimer | ||||
| in the documentation and/or other materials provided with the | ||||
| distribution. | ||||
|    * Neither the name of Google Inc. nor the names of its | ||||
| contributors may be used to endorse or promote products derived from | ||||
| this software without specific prior written permission. | ||||
|  | ||||
| THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | ||||
| "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | ||||
| LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | ||||
| A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | ||||
| OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | ||||
| SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT | ||||
| LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | ||||
| DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | ||||
| THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||||
| (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | ||||
| OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||||
							
								
								
									
										22
									
								
								vendor/golang.org/x/sys/PATENTS
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								vendor/golang.org/x/sys/PATENTS
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| Additional IP Rights Grant (Patents) | ||||
|  | ||||
| "This implementation" means the copyrightable works distributed by | ||||
| Google as part of the Go project. | ||||
|  | ||||
| Google hereby grants to You a perpetual, worldwide, non-exclusive, | ||||
| no-charge, royalty-free, irrevocable (except as stated in this section) | ||||
| patent license to make, have made, use, offer to sell, sell, import, | ||||
| transfer and otherwise run, modify and propagate the contents of this | ||||
| implementation of Go, where such license applies only to those patent | ||||
| claims, both currently owned or controlled by Google and acquired in | ||||
| the future, licensable by Google that are necessarily infringed by this | ||||
| implementation of Go.  This grant does not include claims that would be | ||||
| infringed only as a consequence of further modification of this | ||||
| implementation.  If you or your agent or exclusive licensee institute or | ||||
| order or agree to the institution of patent litigation against any | ||||
| entity (including a cross-claim or counterclaim in a lawsuit) alleging | ||||
| that this implementation of Go or any code incorporated within this | ||||
| implementation of Go constitutes direct or contributory patent | ||||
| infringement, or inducement of patent infringement, then any patent | ||||
| rights granted to you under this License for this implementation of Go | ||||
| shall terminate as of the date such litigation is filed. | ||||
							
								
								
									
										173
									
								
								vendor/golang.org/x/sys/unix/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								vendor/golang.org/x/sys/unix/README.md
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,173 @@ | ||||
| # Building `sys/unix` | ||||
|  | ||||
| The sys/unix package provides access to the raw system call interface of the | ||||
| underlying operating system. See: https://godoc.org/golang.org/x/sys/unix | ||||
|  | ||||
| Porting Go to a new architecture/OS combination or adding syscalls, types, or | ||||
| constants to an existing architecture/OS pair requires some manual effort; | ||||
| however, there are tools that automate much of the process. | ||||
|  | ||||
| ## Build Systems | ||||
|  | ||||
| There are currently two ways we generate the necessary files. We are currently | ||||
| migrating the build system to use containers so the builds are reproducible. | ||||
| This is being done on an OS-by-OS basis. Please update this documentation as | ||||
| components of the build system change. | ||||
|  | ||||
| ### Old Build System (currently for `GOOS != "Linux" || GOARCH == "sparc64"`) | ||||
|  | ||||
| The old build system generates the Go files based on the C header files | ||||
| present on your system. This means that files | ||||
| for a given GOOS/GOARCH pair must be generated on a system with that OS and | ||||
| architecture. This also means that the generated code can differ from system | ||||
| to system, based on differences in the header files. | ||||
|  | ||||
| To avoid this, if you are using the old build system, only generate the Go | ||||
| files on an installation with unmodified header files. It is also important to | ||||
| keep track of which version of the OS the files were generated from (ex. | ||||
| Darwin 14 vs Darwin 15). This makes it easier to track the progress of changes | ||||
| and have each OS upgrade correspond to a single change. | ||||
|  | ||||
| To build the files for your current OS and architecture, make sure GOOS and | ||||
| GOARCH are set correctly and run `mkall.sh`. This will generate the files for | ||||
| your specific system. Running `mkall.sh -n` shows the commands that will be run. | ||||
|  | ||||
| Requirements: bash, perl, go | ||||
|  | ||||
| ### New Build System (currently for `GOOS == "Linux" && GOARCH != "sparc64"`) | ||||
|  | ||||
| The new build system uses a Docker container to generate the go files directly | ||||
| from source checkouts of the kernel and various system libraries. This means | ||||
| that on any platform that supports Docker, all the files using the new build | ||||
| system can be generated at once, and generated files will not change based on | ||||
| what the person running the scripts has installed on their computer. | ||||
|  | ||||
| The OS specific files for the new build system are located in the `${GOOS}` | ||||
| directory, and the build is coordinated by the `${GOOS}/mkall.go` program. When | ||||
| the kernel or system library updates, modify the Dockerfile at | ||||
| `${GOOS}/Dockerfile` to checkout the new release of the source. | ||||
|  | ||||
| To build all the files under the new build system, you must be on an amd64/Linux | ||||
| system and have your GOOS and GOARCH set accordingly. Running `mkall.sh` will | ||||
| then generate all of the files for all of the GOOS/GOARCH pairs in the new build | ||||
| system. Running `mkall.sh -n` shows the commands that will be run. | ||||
|  | ||||
| Requirements: bash, perl, go, docker | ||||
|  | ||||
| ## Component files | ||||
|  | ||||
| This section describes the various files used in the code generation process. | ||||
| It also contains instructions on how to modify these files to add a new | ||||
| architecture/OS or to add additional syscalls, types, or constants. Note that | ||||
| if you are using the new build system, the scripts cannot be called normally. | ||||
| They must be called from within the docker container. | ||||
|  | ||||
| ### asm files | ||||
|  | ||||
| The hand-written assembly file at `asm_${GOOS}_${GOARCH}.s` implements system | ||||
| call dispatch. There are three entry points: | ||||
| ``` | ||||
|   func Syscall(trap, a1, a2, a3 uintptr) (r1, r2, err uintptr) | ||||
|   func Syscall6(trap, a1, a2, a3, a4, a5, a6 uintptr) (r1, r2, err uintptr) | ||||
|   func RawSyscall(trap, a1, a2, a3 uintptr) (r1, r2, err uintptr) | ||||
| ``` | ||||
| The first and second are the standard ones; they differ only in how many | ||||
| arguments can be passed to the kernel. The third is for low-level use by the | ||||
| ForkExec wrapper. Unlike the first two, it does not call into the scheduler to | ||||
| let it know that a system call is running. | ||||
|  | ||||
| When porting Go to an new architecture/OS, this file must be implemented for | ||||
| each GOOS/GOARCH pair. | ||||
|  | ||||
| ### mksysnum | ||||
|  | ||||
| Mksysnum is a script located at `${GOOS}/mksysnum.pl` (or `mksysnum_${GOOS}.pl` | ||||
| for the old system). This script takes in a list of header files containing the | ||||
| syscall number declarations and parses them to produce the corresponding list of | ||||
| Go numeric constants. See `zsysnum_${GOOS}_${GOARCH}.go` for the generated | ||||
| constants. | ||||
|  | ||||
| Adding new syscall numbers is mostly done by running the build on a sufficiently | ||||
| new installation of the target OS (or updating the source checkouts for the | ||||
| new build system). However, depending on the OS, you make need to update the | ||||
| parsing in mksysnum. | ||||
|  | ||||
| ### mksyscall.pl | ||||
|  | ||||
| The `syscall.go`, `syscall_${GOOS}.go`, `syscall_${GOOS}_${GOARCH}.go` are | ||||
| hand-written Go files which implement system calls (for unix, the specific OS, | ||||
| or the specific OS/Architecture pair respectively) that need special handling | ||||
| and list `//sys` comments giving prototypes for ones that can be generated. | ||||
|  | ||||
| The mksyscall.pl script takes the `//sys` and `//sysnb` comments and converts | ||||
| them into syscalls. This requires the name of the prototype in the comment to | ||||
| match a syscall number in the `zsysnum_${GOOS}_${GOARCH}.go` file. The function | ||||
| prototype can be exported (capitalized) or not. | ||||
|  | ||||
| Adding a new syscall often just requires adding a new `//sys` function prototype | ||||
| with the desired arguments and a capitalized name so it is exported. However, if | ||||
| you want the interface to the syscall to be different, often one will make an | ||||
| unexported `//sys` prototype, an then write a custom wrapper in | ||||
| `syscall_${GOOS}.go`. | ||||
|  | ||||
| ### types files | ||||
|  | ||||
| For each OS, there is a hand-written Go file at `${GOOS}/types.go` (or | ||||
| `types_${GOOS}.go` on the old system). This file includes standard C headers and | ||||
| creates Go type aliases to the corresponding C types. The file is then fed | ||||
| through godef to get the Go compatible definitions. Finally, the generated code | ||||
| is fed though mkpost.go to format the code correctly and remove any hidden or | ||||
| private identifiers. This cleaned-up code is written to | ||||
| `ztypes_${GOOS}_${GOARCH}.go`. | ||||
|  | ||||
| The hardest part about preparing this file is figuring out which headers to | ||||
| include and which symbols need to be `#define`d to get the actual data | ||||
| structures that pass through to the kernel system calls. Some C libraries | ||||
| preset alternate versions for binary compatibility and translate them on the | ||||
| way in and out of system calls, but there is almost always a `#define` that can | ||||
| get the real ones. | ||||
| See `types_darwin.go` and `linux/types.go` for examples. | ||||
|  | ||||
| To add a new type, add in the necessary include statement at the top of the | ||||
| file (if it is not already there) and add in a type alias line. Note that if | ||||
| your type is significantly different on different architectures, you may need | ||||
| some `#if/#elif` macros in your include statements. | ||||
|  | ||||
| ### mkerrors.sh | ||||
|  | ||||
| This script is used to generate the system's various constants. This doesn't | ||||
| just include the error numbers and error strings, but also the signal numbers | ||||
| an a wide variety of miscellaneous constants. The constants come from the list | ||||
| of include files in the `includes_${uname}` variable. A regex then picks out | ||||
| the desired `#define` statements, and generates the corresponding Go constants. | ||||
| The error numbers and strings are generated from `#include <errno.h>`, and the | ||||
| signal numbers and strings are generated from `#include <signal.h>`. All of | ||||
| these constants are written to `zerrors_${GOOS}_${GOARCH}.go` via a C program, | ||||
| `_errors.c`, which prints out all the constants. | ||||
|  | ||||
| To add a constant, add the header that includes it to the appropriate variable. | ||||
| Then, edit the regex (if necessary) to match the desired constant. Avoid making | ||||
| the regex too broad to avoid matching unintended constants. | ||||
|  | ||||
|  | ||||
| ## Generated files | ||||
|  | ||||
| ### `zerror_${GOOS}_${GOARCH}.go` | ||||
|  | ||||
| A file containing all of the system's generated error numbers, error strings, | ||||
| signal numbers, and constants. Generated by `mkerrors.sh` (see above). | ||||
|  | ||||
| ### `zsyscall_${GOOS}_${GOARCH}.go` | ||||
|  | ||||
| A file containing all the generated syscalls for a specific GOOS and GOARCH. | ||||
| Generated by `mksyscall.pl` (see above). | ||||
|  | ||||
| ### `zsysnum_${GOOS}_${GOARCH}.go` | ||||
|  | ||||
| A list of numeric constants for all the syscall number of the specific GOOS | ||||
| and GOARCH. Generated by mksysnum (see above). | ||||
|  | ||||
| ### `ztypes_${GOOS}_${GOARCH}.go` | ||||
|  | ||||
| A file containing Go types for passing into (or returning from) syscalls. | ||||
| Generated by godefs and the types file (see above). | ||||
							
								
								
									
										124
									
								
								vendor/golang.org/x/sys/unix/affinity_linux.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								vendor/golang.org/x/sys/unix/affinity_linux.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| // Copyright 2018 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // CPU affinity functions | ||||
|  | ||||
| package unix | ||||
|  | ||||
| import ( | ||||
| 	"unsafe" | ||||
| ) | ||||
|  | ||||
| const cpuSetSize = _CPU_SETSIZE / _NCPUBITS | ||||
|  | ||||
| // CPUSet represents a CPU affinity mask. | ||||
| type CPUSet [cpuSetSize]cpuMask | ||||
|  | ||||
| func schedAffinity(trap uintptr, pid int, set *CPUSet) error { | ||||
| 	_, _, e := RawSyscall(trap, uintptr(pid), uintptr(unsafe.Sizeof(*set)), uintptr(unsafe.Pointer(set))) | ||||
| 	if e != 0 { | ||||
| 		return errnoErr(e) | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // SchedGetaffinity gets the CPU affinity mask of the thread specified by pid. | ||||
| // If pid is 0 the calling thread is used. | ||||
| func SchedGetaffinity(pid int, set *CPUSet) error { | ||||
| 	return schedAffinity(SYS_SCHED_GETAFFINITY, pid, set) | ||||
| } | ||||
|  | ||||
| // SchedSetaffinity sets the CPU affinity mask of the thread specified by pid. | ||||
| // If pid is 0 the calling thread is used. | ||||
| func SchedSetaffinity(pid int, set *CPUSet) error { | ||||
| 	return schedAffinity(SYS_SCHED_SETAFFINITY, pid, set) | ||||
| } | ||||
|  | ||||
| // Zero clears the set s, so that it contains no CPUs. | ||||
| func (s *CPUSet) Zero() { | ||||
| 	for i := range s { | ||||
| 		s[i] = 0 | ||||
| 	} | ||||
| } | ||||
|  | ||||
| func cpuBitsIndex(cpu int) int { | ||||
| 	return cpu / _NCPUBITS | ||||
| } | ||||
|  | ||||
| func cpuBitsMask(cpu int) cpuMask { | ||||
| 	return cpuMask(1 << (uint(cpu) % _NCPUBITS)) | ||||
| } | ||||
|  | ||||
| // Set adds cpu to the set s. | ||||
| func (s *CPUSet) Set(cpu int) { | ||||
| 	i := cpuBitsIndex(cpu) | ||||
| 	if i < len(s) { | ||||
| 		s[i] |= cpuBitsMask(cpu) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Clear removes cpu from the set s. | ||||
| func (s *CPUSet) Clear(cpu int) { | ||||
| 	i := cpuBitsIndex(cpu) | ||||
| 	if i < len(s) { | ||||
| 		s[i] &^= cpuBitsMask(cpu) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // IsSet reports whether cpu is in the set s. | ||||
| func (s *CPUSet) IsSet(cpu int) bool { | ||||
| 	i := cpuBitsIndex(cpu) | ||||
| 	if i < len(s) { | ||||
| 		return s[i]&cpuBitsMask(cpu) != 0 | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| // Count returns the number of CPUs in the set s. | ||||
| func (s *CPUSet) Count() int { | ||||
| 	c := 0 | ||||
| 	for _, b := range s { | ||||
| 		c += onesCount64(uint64(b)) | ||||
| 	} | ||||
| 	return c | ||||
| } | ||||
|  | ||||
| // onesCount64 is a copy of Go 1.9's math/bits.OnesCount64. | ||||
| // Once this package can require Go 1.9, we can delete this | ||||
| // and update the caller to use bits.OnesCount64. | ||||
| func onesCount64(x uint64) int { | ||||
| 	const m0 = 0x5555555555555555 // 01010101 ... | ||||
| 	const m1 = 0x3333333333333333 // 00110011 ... | ||||
| 	const m2 = 0x0f0f0f0f0f0f0f0f // 00001111 ... | ||||
| 	const m3 = 0x00ff00ff00ff00ff // etc. | ||||
| 	const m4 = 0x0000ffff0000ffff | ||||
|  | ||||
| 	// Implementation: Parallel summing of adjacent bits. | ||||
| 	// See "Hacker's Delight", Chap. 5: Counting Bits. | ||||
| 	// The following pattern shows the general approach: | ||||
| 	// | ||||
| 	//   x = x>>1&(m0&m) + x&(m0&m) | ||||
| 	//   x = x>>2&(m1&m) + x&(m1&m) | ||||
| 	//   x = x>>4&(m2&m) + x&(m2&m) | ||||
| 	//   x = x>>8&(m3&m) + x&(m3&m) | ||||
| 	//   x = x>>16&(m4&m) + x&(m4&m) | ||||
| 	//   x = x>>32&(m5&m) + x&(m5&m) | ||||
| 	//   return int(x) | ||||
| 	// | ||||
| 	// Masking (& operations) can be left away when there's no | ||||
| 	// danger that a field's sum will carry over into the next | ||||
| 	// field: Since the result cannot be > 64, 8 bits is enough | ||||
| 	// and we can ignore the masks for the shifts by 8 and up. | ||||
| 	// Per "Hacker's Delight", the first line can be simplified | ||||
| 	// more, but it saves at best one instruction, so we leave | ||||
| 	// it alone for clarity. | ||||
| 	const m = 1<<64 - 1 | ||||
| 	x = x>>1&(m0&m) + x&(m0&m) | ||||
| 	x = x>>2&(m1&m) + x&(m1&m) | ||||
| 	x = (x>>4 + x) & (m2 & m) | ||||
| 	x += x >> 8 | ||||
| 	x += x >> 16 | ||||
| 	x += x >> 32 | ||||
| 	return int(x) & (1<<7 - 1) | ||||
| } | ||||
							
								
								
									
										14
									
								
								vendor/golang.org/x/sys/unix/aliases.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								vendor/golang.org/x/sys/unix/aliases.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| // Copyright 2018 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build aix darwin dragonfly freebsd linux netbsd openbsd solaris | ||||
| // +build go1.9 | ||||
|  | ||||
| package unix | ||||
|  | ||||
| import "syscall" | ||||
|  | ||||
| type Signal = syscall.Signal | ||||
| type Errno = syscall.Errno | ||||
| type SysProcAttr = syscall.SysProcAttr | ||||
							
								
								
									
										17
									
								
								vendor/golang.org/x/sys/unix/asm_aix_ppc64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								vendor/golang.org/x/sys/unix/asm_aix_ppc64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | ||||
| // Copyright 2018 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System calls for ppc64, AIX are implemented in runtime/syscall_aix.go | ||||
| // | ||||
|  | ||||
| TEXT ·syscall6(SB),NOSPLIT,$0-88 | ||||
| 	JMP	syscall·syscall6(SB) | ||||
|  | ||||
| TEXT ·rawSyscall6(SB),NOSPLIT,$0-88 | ||||
| 	JMP	syscall·rawSyscall6(SB) | ||||
							
								
								
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_386.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_386.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for 386, Darwin | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-52 | ||||
| 	JMP	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for AMD64, Darwin | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-104 | ||||
| 	JMP	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT	·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										30
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_arm.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_arm.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| // Copyright 2015 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
| // +build arm,darwin | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for ARM, Darwin | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	B	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-52 | ||||
| 	B	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT	·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	B	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										30
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_arm64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								vendor/golang.org/x/sys/unix/asm_darwin_arm64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| // Copyright 2015 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
| // +build arm64,darwin | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for AMD64, Darwin | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	B	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	B	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-104 | ||||
| 	B	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT	·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	B	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	B	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_dragonfly_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_dragonfly_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for AMD64, DragonFly | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-104 | ||||
| 	JMP	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_freebsd_386.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_freebsd_386.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for 386, FreeBSD | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-52 | ||||
| 	JMP	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_freebsd_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_freebsd_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for AMD64, FreeBSD | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-104 | ||||
| 	JMP	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_freebsd_arm.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								vendor/golang.org/x/sys/unix/asm_freebsd_arm.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| // Copyright 2012 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System call support for ARM, FreeBSD | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT	·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT	·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	B	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT	·Syscall9(SB),NOSPLIT,$0-52 | ||||
| 	B	syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT	·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT	·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	B	syscall·RawSyscall6(SB) | ||||
							
								
								
									
										65
									
								
								vendor/golang.org/x/sys/unix/asm_linux_386.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								vendor/golang.org/x/sys/unix/asm_linux_386.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System calls for 386, Linux | ||||
| // | ||||
|  | ||||
| // See ../runtime/sys_linux_386.s for the reason why we always use int 0x80 | ||||
| // instead of the glibc-specific "CALL 0x10(GS)". | ||||
| #define INVOKE_SYSCALL	INT	$0x80 | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT ·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT ·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT ·SyscallNoError(SB),NOSPLIT,$0-24 | ||||
| 	CALL	runtime·entersyscall(SB) | ||||
| 	MOVL	trap+0(FP), AX  // syscall entry | ||||
| 	MOVL	a1+4(FP), BX | ||||
| 	MOVL	a2+8(FP), CX | ||||
| 	MOVL	a3+12(FP), DX | ||||
| 	MOVL	$0, SI | ||||
| 	MOVL	$0, DI | ||||
| 	INVOKE_SYSCALL | ||||
| 	MOVL	AX, r1+16(FP) | ||||
| 	MOVL	DX, r2+20(FP) | ||||
| 	CALL	runtime·exitsyscall(SB) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT ·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
|  | ||||
| TEXT ·RawSyscallNoError(SB),NOSPLIT,$0-24 | ||||
| 	MOVL	trap+0(FP), AX  // syscall entry | ||||
| 	MOVL	a1+4(FP), BX | ||||
| 	MOVL	a2+8(FP), CX | ||||
| 	MOVL	a3+12(FP), DX | ||||
| 	MOVL	$0, SI | ||||
| 	MOVL	$0, DI | ||||
| 	INVOKE_SYSCALL | ||||
| 	MOVL	AX, r1+16(FP) | ||||
| 	MOVL	DX, r2+20(FP) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·socketcall(SB),NOSPLIT,$0-36 | ||||
| 	JMP	syscall·socketcall(SB) | ||||
|  | ||||
| TEXT ·rawsocketcall(SB),NOSPLIT,$0-36 | ||||
| 	JMP	syscall·rawsocketcall(SB) | ||||
|  | ||||
| TEXT ·seek(SB),NOSPLIT,$0-28 | ||||
| 	JMP	syscall·seek(SB) | ||||
							
								
								
									
										57
									
								
								vendor/golang.org/x/sys/unix/asm_linux_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								vendor/golang.org/x/sys/unix/asm_linux_amd64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System calls for AMD64, Linux | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT ·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT ·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT ·SyscallNoError(SB),NOSPLIT,$0-48 | ||||
| 	CALL	runtime·entersyscall(SB) | ||||
| 	MOVQ	a1+8(FP), DI | ||||
| 	MOVQ	a2+16(FP), SI | ||||
| 	MOVQ	a3+24(FP), DX | ||||
| 	MOVQ	$0, R10 | ||||
| 	MOVQ	$0, R8 | ||||
| 	MOVQ	$0, R9 | ||||
| 	MOVQ	trap+0(FP), AX	// syscall entry | ||||
| 	SYSCALL | ||||
| 	MOVQ	AX, r1+32(FP) | ||||
| 	MOVQ	DX, r2+40(FP) | ||||
| 	CALL	runtime·exitsyscall(SB) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT ·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
|  | ||||
| TEXT ·RawSyscallNoError(SB),NOSPLIT,$0-48 | ||||
| 	MOVQ	a1+8(FP), DI | ||||
| 	MOVQ	a2+16(FP), SI | ||||
| 	MOVQ	a3+24(FP), DX | ||||
| 	MOVQ	$0, R10 | ||||
| 	MOVQ	$0, R8 | ||||
| 	MOVQ	$0, R9 | ||||
| 	MOVQ	trap+0(FP), AX	// syscall entry | ||||
| 	SYSCALL | ||||
| 	MOVQ	AX, r1+32(FP) | ||||
| 	MOVQ	DX, r2+40(FP) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·gettimeofday(SB),NOSPLIT,$0-16 | ||||
| 	JMP	syscall·gettimeofday(SB) | ||||
							
								
								
									
										56
									
								
								vendor/golang.org/x/sys/unix/asm_linux_arm.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								vendor/golang.org/x/sys/unix/asm_linux_arm.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| // Copyright 2009 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System calls for arm, Linux | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT ·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT ·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	B	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT ·SyscallNoError(SB),NOSPLIT,$0-24 | ||||
| 	BL	runtime·entersyscall(SB) | ||||
| 	MOVW	trap+0(FP), R7 | ||||
| 	MOVW	a1+4(FP), R0 | ||||
| 	MOVW	a2+8(FP), R1 | ||||
| 	MOVW	a3+12(FP), R2 | ||||
| 	MOVW	$0, R3 | ||||
| 	MOVW	$0, R4 | ||||
| 	MOVW	$0, R5 | ||||
| 	SWI	$0 | ||||
| 	MOVW	R0, r1+16(FP) | ||||
| 	MOVW	$0, R0 | ||||
| 	MOVW	R0, r2+20(FP) | ||||
| 	BL	runtime·exitsyscall(SB) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT ·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	B	syscall·RawSyscall6(SB) | ||||
|  | ||||
| TEXT ·RawSyscallNoError(SB),NOSPLIT,$0-24 | ||||
| 	MOVW	trap+0(FP), R7	// syscall entry | ||||
| 	MOVW	a1+4(FP), R0 | ||||
| 	MOVW	a2+8(FP), R1 | ||||
| 	MOVW	a3+12(FP), R2 | ||||
| 	SWI	$0 | ||||
| 	MOVW	R0, r1+16(FP) | ||||
| 	MOVW	$0, R0 | ||||
| 	MOVW	R0, r2+20(FP) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·seek(SB),NOSPLIT,$0-28 | ||||
| 	B	syscall·seek(SB) | ||||
							
								
								
									
										52
									
								
								vendor/golang.org/x/sys/unix/asm_linux_arm64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								vendor/golang.org/x/sys/unix/asm_linux_arm64.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| // Copyright 2015 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build linux | ||||
| // +build arm64 | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT ·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	B	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT ·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	B	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT ·SyscallNoError(SB),NOSPLIT,$0-48 | ||||
| 	BL	runtime·entersyscall(SB) | ||||
| 	MOVD	a1+8(FP), R0 | ||||
| 	MOVD	a2+16(FP), R1 | ||||
| 	MOVD	a3+24(FP), R2 | ||||
| 	MOVD	$0, R3 | ||||
| 	MOVD	$0, R4 | ||||
| 	MOVD	$0, R5 | ||||
| 	MOVD	trap+0(FP), R8	// syscall entry | ||||
| 	SVC | ||||
| 	MOVD	R0, r1+32(FP)	// r1 | ||||
| 	MOVD	R1, r2+40(FP)	// r2 | ||||
| 	BL	runtime·exitsyscall(SB) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	B	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT ·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	B	syscall·RawSyscall6(SB) | ||||
|  | ||||
| TEXT ·RawSyscallNoError(SB),NOSPLIT,$0-48 | ||||
| 	MOVD	a1+8(FP), R0 | ||||
| 	MOVD	a2+16(FP), R1 | ||||
| 	MOVD	a3+24(FP), R2 | ||||
| 	MOVD	$0, R3 | ||||
| 	MOVD	$0, R4 | ||||
| 	MOVD	$0, R5 | ||||
| 	MOVD	trap+0(FP), R8	// syscall entry | ||||
| 	SVC | ||||
| 	MOVD	R0, r1+32(FP) | ||||
| 	MOVD	R1, r2+40(FP) | ||||
| 	RET | ||||
							
								
								
									
										56
									
								
								vendor/golang.org/x/sys/unix/asm_linux_mips64x.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								vendor/golang.org/x/sys/unix/asm_linux_mips64x.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| // Copyright 2015 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build linux | ||||
| // +build mips64 mips64le | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System calls for mips64, Linux | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT ·Syscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·Syscall(SB) | ||||
|  | ||||
| TEXT ·Syscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT ·SyscallNoError(SB),NOSPLIT,$0-48 | ||||
| 	JAL	runtime·entersyscall(SB) | ||||
| 	MOVV	a1+8(FP), R4 | ||||
| 	MOVV	a2+16(FP), R5 | ||||
| 	MOVV	a3+24(FP), R6 | ||||
| 	MOVV	R0, R7 | ||||
| 	MOVV	R0, R8 | ||||
| 	MOVV	R0, R9 | ||||
| 	MOVV	trap+0(FP), R2	// syscall entry | ||||
| 	SYSCALL | ||||
| 	MOVV	R2, r1+32(FP) | ||||
| 	MOVV	R3, r2+40(FP) | ||||
| 	JAL	runtime·exitsyscall(SB) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-56 | ||||
| 	JMP	syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT ·RawSyscall6(SB),NOSPLIT,$0-80 | ||||
| 	JMP	syscall·RawSyscall6(SB) | ||||
|  | ||||
| TEXT ·RawSyscallNoError(SB),NOSPLIT,$0-48 | ||||
| 	MOVV	a1+8(FP), R4 | ||||
| 	MOVV	a2+16(FP), R5 | ||||
| 	MOVV	a3+24(FP), R6 | ||||
| 	MOVV	R0, R7 | ||||
| 	MOVV	R0, R8 | ||||
| 	MOVV	R0, R9 | ||||
| 	MOVV	trap+0(FP), R2	// syscall entry | ||||
| 	SYSCALL | ||||
| 	MOVV	R2, r1+32(FP) | ||||
| 	MOVV	R3, r2+40(FP) | ||||
| 	RET | ||||
							
								
								
									
										54
									
								
								vendor/golang.org/x/sys/unix/asm_linux_mipsx.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								vendor/golang.org/x/sys/unix/asm_linux_mipsx.s
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,54 @@ | ||||
| // Copyright 2016 The Go Authors. All rights reserved. | ||||
| // Use of this source code is governed by a BSD-style | ||||
| // license that can be found in the LICENSE file. | ||||
|  | ||||
| // +build linux | ||||
| // +build mips mipsle | ||||
| // +build !gccgo | ||||
|  | ||||
| #include "textflag.h" | ||||
|  | ||||
| // | ||||
| // System calls for mips, Linux | ||||
| // | ||||
|  | ||||
| // Just jump to package syscall's implementation for all these functions. | ||||
| // The runtime may know about them. | ||||
|  | ||||
| TEXT ·Syscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP syscall·Syscall(SB) | ||||
|  | ||||
| TEXT ·Syscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP syscall·Syscall6(SB) | ||||
|  | ||||
| TEXT ·Syscall9(SB),NOSPLIT,$0-52 | ||||
| 	JMP syscall·Syscall9(SB) | ||||
|  | ||||
| TEXT ·SyscallNoError(SB),NOSPLIT,$0-24 | ||||
| 	JAL	runtime·entersyscall(SB) | ||||
| 	MOVW	a1+4(FP), R4 | ||||
| 	MOVW	a2+8(FP), R5 | ||||
| 	MOVW	a3+12(FP), R6 | ||||
| 	MOVW	R0, R7 | ||||
| 	MOVW	trap+0(FP), R2	// syscall entry | ||||
| 	SYSCALL | ||||
| 	MOVW	R2, r1+16(FP)	// r1 | ||||
| 	MOVW	R3, r2+20(FP)	// r2 | ||||
| 	JAL	runtime·exitsyscall(SB) | ||||
| 	RET | ||||
|  | ||||
| TEXT ·RawSyscall(SB),NOSPLIT,$0-28 | ||||
| 	JMP syscall·RawSyscall(SB) | ||||
|  | ||||
| TEXT ·RawSyscall6(SB),NOSPLIT,$0-40 | ||||
| 	JMP syscall·RawSyscall6(SB) | ||||
|  | ||||
| TEXT ·RawSyscallNoError(SB),NOSPLIT,$0-24 | ||||
| 	MOVW	a1+4(FP), R4 | ||||
| 	MOVW	a2+8(FP), R5 | ||||
| 	MOVW	a3+12(FP), R6 | ||||
| 	MOVW	trap+0(FP), R2	// syscall entry | ||||
| 	SYSCALL | ||||
| 	MOVW	R2, r1+16(FP) | ||||
| 	MOVW	R3, r2+20(FP) | ||||
| 	RET | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user