node.go 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275
  1. package main
  2. import (
  3. "fmt"
  4. "log"
  5. "strconv"
  6. "strings"
  7. )
  8. // A interconnected structure which connects to multiple entities
  9. //
  10. // Handles depth (in tabs) for the Endless sky save format (which is ugly)
  11. type Node struct {
  12. Tag string // A tag, useful for identifying this node from other nodes in Children
  13. Line string // The line of text for the node (the Parser removes the depth number of tabs)
  14. Depth int // Number of tabs deep this node is (root and such are depth 0)
  15. Parent *Node `json:"-"` // The parent of this node (used for traversing up the node tree), must be json ignored or recursive looping
  16. Children []*Node // Any other nodes that can be tied to this node
  17. }
  18. // Returns the Endless sky representation (node depth is number of tabs)
  19. //
  20. // Useful for representing the node's line at the correct depth
  21. func (n *Node) String() string {
  22. return fmt.Sprintf("%s%s", strings.Repeat("\t", n.Depth), n.Line)
  23. }
  24. // Strings everything (the current line plus any children)
  25. func (n *Node) StringAll() string {
  26. out := strings.Builder{}
  27. out.WriteString(n.String() + "\n")
  28. if n.Len() != 0 {
  29. // Recurse over children calling this function for the line and other children
  30. for _, kid := range n.Children {
  31. out.WriteString(kid.StringAll())
  32. }
  33. }
  34. return out.String()
  35. }
  36. // Adds the given node as a child
  37. //
  38. // Used to build complex structures
  39. func (n *Node) AddChild(child *Node) {
  40. child.Parent = n
  41. n.Children = append(n.Children, child)
  42. }
  43. // Creates a new child node and returns it
  44. //
  45. // Used to build complex structures
  46. func (n *Node) NewChild() *Node {
  47. kid := Node{
  48. Parent: n,
  49. }
  50. n.AddChild(&kid)
  51. return &kid
  52. }
  53. // Removes the child given index
  54. func (n *Node) RmChild(index int) error {
  55. if index > n.Len() || index < 0 {
  56. return fmt.Errorf("invalid index %d (min=0, max=%d)", index, n.Len())
  57. }
  58. var kids []*Node
  59. for idx, k := range n.Children {
  60. if idx != index {
  61. kids = append(kids, k)
  62. } else {
  63. k.Parent = nil
  64. }
  65. }
  66. n.Children = kids
  67. return nil
  68. }
  69. // Removes all children
  70. //
  71. // Useful for when you will be reassigning a collection of node children underneath
  72. func (n *Node) RmAllChildren() {
  73. for _, k := range n.Children {
  74. k.Parent = nil
  75. }
  76. n.Children = []*Node{}
  77. }
  78. // Removes the current node from the parent
  79. //
  80. // Similar to RmChild or RmAllChildren except called from the child's perspective
  81. func (n *Node) DetachFromParent() (*Node, error) {
  82. if n.Parent != nil {
  83. p := n.Parent
  84. index := -1
  85. for idx, k := range p.Children {
  86. if k.Line == n.Line && k.Depth == n.Depth && k.Tag == n.Tag && k.Len() == n.Len() {
  87. index = idx
  88. }
  89. }
  90. if index != -1 {
  91. return p.SplitChild(index)
  92. } else {
  93. return nil, fmt.Errorf("failed detaching, didn't find ourselves in parent")
  94. }
  95. }
  96. return nil, fmt.Errorf("failed detaching, invalid parent")
  97. }
  98. // Obtain a pointer to the child
  99. func (n *Node) Child(index int) (*Node, error) {
  100. if index > n.Len() || index < 0 {
  101. return nil, fmt.Errorf("invalid index %d (min=0, max=%d)", index, n.Len())
  102. }
  103. return n.Children[index], nil
  104. }
  105. // Replaces the child by index
  106. func (n *Node) ReplaceChild(index int, node *Node) error {
  107. if index > n.Len() || index < 0 {
  108. return fmt.Errorf("invalid index %d (min=0, max=%d)", index, n.Len())
  109. }
  110. n.Children[index].Parent = nil
  111. node.Parent = n
  112. n.Children[index] = node
  113. return nil
  114. }
  115. // Removes the child but returns it (useful for if you plan on replacing it)
  116. func (n *Node) SplitChild(index int) (*Node, error) {
  117. if index > n.Len() || index < 0 {
  118. return nil, fmt.Errorf("invalid index %d (min=0, max=%d)", index, n.Len())
  119. }
  120. k := n.Children[index]
  121. k.Parent = nil
  122. err := n.RmChild(index)
  123. if err != nil {
  124. k.Parent = n // Some error recovery
  125. return nil, err
  126. }
  127. return k, nil
  128. }
  129. // Returns the number of children underneath
  130. func (n *Node) Len() int {
  131. return len(n.Children)
  132. }
  133. // Endless sky stores most data as 'key value'
  134. //
  135. // With special cases:
  136. //
  137. // As String '"some key" value' (Or '"some key" "some value"')
  138. //
  139. // As Int: '"some key" 13'
  140. //
  141. // As Float: '"some key" 9.81'
  142. //
  143. // Return's key part
  144. func (n *Node) Key() string {
  145. parts := strings.Split(n.Line, " ")
  146. if strings.Contains(parts[0], "\"") {
  147. pref := ""
  148. for _, part := range parts {
  149. if strings.HasSuffix(part, "\"") {
  150. if pref != "" {
  151. pref += " "
  152. }
  153. pref += part
  154. break
  155. }
  156. if pref != "" {
  157. pref += " "
  158. }
  159. pref += part
  160. }
  161. pref = strings.ReplaceAll(pref, "\"", "")
  162. return pref
  163. }
  164. return parts[0]
  165. }
  166. // Endless sky stores mose data as 'key value'
  167. //
  168. // With special cases:
  169. //
  170. // As String '"some key" value' (Or '"some key" "some value"')
  171. //
  172. // As Int: '"some key" 13'
  173. //
  174. // As Float: '"some key" 9.81'
  175. //
  176. // Return's value part (as string)
  177. func (n *Node) Value() string {
  178. key := n.Key()
  179. val := strings.Replace(n.Line, n.Format(key)+" ", "", 1)
  180. val = strings.ReplaceAll(val, "\"", "")
  181. return val
  182. }
  183. // Returns the value as integer (errors return 0)
  184. func (n *Node) ValueInt() int {
  185. v, err := strconv.Atoi(n.Value())
  186. if err != nil {
  187. log.Printf("ValueInt('%s') => %v", n.Value(), err)
  188. return 0
  189. }
  190. return v
  191. }
  192. // Returns the value as float64 (errors return 0.0, but are logged)
  193. func (n *Node) ValueFloat64() float64 {
  194. v, err := strconv.ParseFloat(n.Value(), 64)
  195. if err != nil {
  196. log.Printf("ValueFloat64('%s') => %v", n.Value(), err)
  197. return 0.0
  198. }
  199. return v
  200. }
  201. // Returns the value as float32 (errors return 0.0, but are logged)
  202. func (n *Node) ValueFloat32() float32 {
  203. v, err := strconv.ParseFloat(n.Value(), 32)
  204. if err != nil {
  205. log.Printf("ValueFloat32('%s') => %v", n.Value(), err)
  206. return 0.0
  207. }
  208. return float32(v)
  209. }
  210. // Format function
  211. //
  212. // Endless sky demands any multi word key or value must be wrapped in "s
  213. //
  214. // Both Node.Key() and Node.Value() remove the "s so this must be called to form the new Node.Line (for setters)
  215. func (n *Node) Format(value string) string {
  216. if strings.Contains(value, " ") {
  217. return "\"" + value + "\""
  218. }
  219. return value
  220. }
  221. // Assigns a new value for the Node
  222. //
  223. // This updates/changes the Node.Line (as that's the only way to do it)
  224. func (n *Node) SetValue(val interface{}) {
  225. if val == nil { // Make it a flag/boolean
  226. n.Line = n.Format(n.Key())
  227. return
  228. }
  229. str_val := fmt.Sprintf("%v", val)
  230. n.Line = fmt.Sprintf("%s %s", n.Format(n.Key()), n.Format(str_val))
  231. }
  232. // Assigns a new key for the Node
  233. //
  234. // This must be a string (key's can't be anything else)
  235. //
  236. // This updates/changes the Node.Line (as that's the only way to do it)
  237. func (n *Node) SetKey(key string) {
  238. val := n.Value()
  239. if val != "" {
  240. n.Line = fmt.Sprintf("%s %s", n.Format(key), n.Format(val))
  241. } else {
  242. n.Line = n.Format(key)
  243. }
  244. }
  245. // Assigns the whole line from the key to the value
  246. func (n *Node) Set(key string, val interface{}) {
  247. if val == nil {
  248. n.Line = n.Format(key)
  249. return
  250. }
  251. str_val := fmt.Sprintf("%v", val)
  252. n.Line = fmt.Sprintf("%s %s", n.Format(key), n.Format(str_val))
  253. }