forked from devnw-forks/structs
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathpkg.go
More file actions
177 lines (177 loc) · 4.59 KB
/
pkg.go
File metadata and controls
177 lines (177 loc) · 4.59 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
// Forked from github.com/devnw/structs
// Package structs contains various utilities to work with Go (Golang) structs. It was
// initially used by me to convert a struct into a `map[string]any`. With
// time I've added other utilities for structs. It's basically a high level
// package based on primitives from the reflect package. Feel free to add new
// functions or improve the existing code.
//
// ## Install
//
// `go get github.com/midgarco/structs`
//
// ## Usage and Examples
//
// Just like the standard lib `strings`, `bytes` and co packages, `structs` has
// many global functions to manipulate or organize your struct data. Lets define
// and declare a struct:
//
// ```go
//
// type Server struct {
// Name string `json:"name,omitempty"`
// ID int
// Enabled bool
// users []string // not exported
// http.Server // embedded
// }
//
// server := &Server{
// Name: "gopher",
// ID: 123456,
// Enabled: true,
// }
//
// ```
//
// ```go
//
// // Convert a struct to a map[string]any
//
// // => {"Name":"gopher", "ID":123456, "Enabled":true}
// m := structs.Map(server)
//
// // Convert the values of a struct to a []any
// // => ["gopher", 123456, true]
// v := structs.Values(server)
//
// // Convert the names of a struct to a []string
// // (see "Names methods" for more info about fields)
// n := structs.Names(server)
//
// // Convert the values of a struct to a []*Field
// // (see "Field methods" for more info about fields)
// f := structs.Fields(server)
//
// // Return the struct name => "Server"
// n := structs.Name(server)
//
// // Check if any field of a struct is initialized or not.
// h := structs.HasZero(server)
//
// // Check if all fields of a struct is initialized or not.
// z := structs.IsZero(server)
//
// // Check if server is a struct or a pointer to struct
// i := structs.IsStruct(server)
//
// ```
//
// ### Struct methods
//
// The structs functions can be also used as independent methods by creating a new
// `*structs.Struct`. This is handy if you want to have more control over the
// structs (such as retrieving a single Field).
//
// ```go
//
// // Create a new struct type:
// s := structs.New(server)
//
// m := s.Map() // Get a map[string]any
// v := s.Values() // Get a []any
// f := s.Fields() // Get a []*Field
// n := s.Names() // Get a []string
// f := s.Field(name) // Get a *Field based on the given field name
// f, ok := s.FieldOk(name) // Get a *Field based on the given field name
// n := s.Name() // Get the struct name
// h := s.HasZero() // Check if any field is uninitialized
// z := s.IsZero() // Check if all fields are uninitialized
//
// ```
//
// ### Field methods
//
// We can easily examine a single Field for more detail. Below you can see how we
// get and interact with various field methods:
//
// ```go
//
// s := structs.New(server)
//
// // Get the Field struct for the "Name" field
// name := s.Field("Name")
//
// // Get the underlying value, value => "gopher"
// value := name.Value().(string)
//
// // Set the field's value
// name.Set("another gopher")
//
// // Get the field's kind, kind => "string"
// name.Kind()
//
// // Check if the field is exported or not
//
// if name.IsExported() {
// fmt.Println("Name field is exported")
// }
//
// // Check if the value is a zero value, such as "" for string, 0 for int
//
// if !name.IsZero() {
// fmt.Println("Name is initialized")
// }
//
// // Check if the field is an anonymous (embedded) field
//
// if !name.IsEmbedded() {
// fmt.Println("Name is not an embedded field")
// }
//
// // Get the Field's tag value for tag name "json", tag value => "name,omitempty"
// tagValue := name.Tag("json")
//
// ```
//
// Nested structs are supported too:
//
// ```go
//
// addrField := s.Field("Server").Field("Addr")
//
// // Get the value for addr
// a := addrField.Value().(string)
//
// // Or get all fields
// httpServer := s.Field("Server").Fields()
//
// ```
//
// We can also get a slice of Fields from the Struct type to iterate over all
// fields. This is handy if you wish to examine all fields:
//
// ```go
//
// s := structs.New(server)
//
// for _, f := range s.Fields() {
// fmt.Printf("field name: %+v\n", f.Name())
//
// if f.IsExported() {
// fmt.Printf("value : %+v\n", f.Value())
// fmt.Printf("is zero : %+v\n", f.IsZero())
// }
// }
//
// ```
//
// ## Credits
//
// - [] (https://github.com/devnw)
// - [Fatih Arslan](https://github.com/fatih)
// - [Cihangir Savas](https://github.com/cihangir)
//
// ## License
//
// The MIT License (MIT) - see LICENSE.md for more details
package structs