parser.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "os"
  6. "strings"
  7. )
  8. // A Parser for handling Endless Sky save format
  9. type Parser struct {
  10. Debug bool // Do we print debug info (to logs)
  11. }
  12. // Parses a given file returning a tree of Nodes
  13. func (p *Parser) Load(filename string) (Root *Node, err error) {
  14. Root = &Node{
  15. Tag: "root",
  16. Line: filename,
  17. Depth: -1,
  18. }
  19. file, err := os.Open(filename)
  20. if err != nil {
  21. return
  22. }
  23. defer file.Close()
  24. radar := bufio.NewScanner(file)
  25. content := []string{}
  26. for radar.Scan() {
  27. line := radar.Text()
  28. line = strings.ReplaceAll(line, "\n", "")
  29. content = append(content, line)
  30. }
  31. p.parse(&content, 0, Root)
  32. return
  33. }
  34. // Assistant func
  35. //
  36. // Processes the current line and next line for making the node tree/structure
  37. func (p *Parser) parse(file *[]string, linenum int, node *Node) {
  38. if linenum+1 > len(*file) {
  39. if p.Debug {
  40. fmt.Println("EOF")
  41. }
  42. return
  43. }
  44. line := (*file)[linenum]
  45. var next_line string
  46. if linenum+2 <= len(*file) {
  47. next_line = (*file)[linenum+1]
  48. } else {
  49. next_line = ""
  50. }
  51. next_depth := FindDepth(next_line)
  52. if p.Debug {
  53. fmt.Printf("%4d | '%s'\nNXTL | '%s'\n", linenum, line, next_line)
  54. }
  55. kid := node.NewChild()
  56. kid.Depth = FindDepth(line)
  57. kid.Line = CleanLine(line)
  58. if next_depth == kid.Depth {
  59. if p.Debug {
  60. fmt.Printf("Same Depth (%d)\n", node.Depth)
  61. }
  62. p.parse(file, linenum+1, node)
  63. } else if next_depth > kid.Depth {
  64. if p.Debug {
  65. fmt.Printf("Deeper (%d->%d)\n", kid.Depth, next_depth)
  66. }
  67. p.parse(file, linenum+1, kid)
  68. } else if next_depth < kid.Depth {
  69. diff := kid.Depth - next_depth
  70. at := node
  71. if p.Debug {
  72. fmt.Printf("Surface (%d<-%d) diff=%d\n", kid.Depth, next_depth, diff)
  73. }
  74. for up := 0; up < diff; up++ {
  75. if at.Parent != nil {
  76. at = at.Parent
  77. }
  78. }
  79. p.parse(file, linenum+1, at)
  80. }
  81. }
  82. // Assistant func
  83. //
  84. // # Supports writing the current node's line, and the node's children too
  85. //
  86. // This uses recursive calls to write out the children
  87. func (p *Parser) saveassist(file *os.File, node *Node) error {
  88. _, err := file.WriteString(node.String() + "\n")
  89. if err != nil {
  90. return err
  91. }
  92. if node.Len() != 0 {
  93. for _, kid := range node.Children {
  94. err := p.saveassist(file, kid)
  95. if err != nil {
  96. return err
  97. }
  98. }
  99. }
  100. return nil
  101. }
  102. // Saves the node and it's children to the given filename
  103. func (p *Parser) Save(filename string, Root *Node) error {
  104. file, err := os.OpenFile(filename, os.O_WRONLY|os.O_TRUNC|os.O_CREATE, 0660)
  105. if err != nil {
  106. return err
  107. }
  108. defer file.Close()
  109. for _, kid := range Root.Children {
  110. err = p.saveassist(file, kid)
  111. if err != nil {
  112. return err
  113. }
  114. }
  115. return nil
  116. }
  117. // TODO: Move these Util funcs to utils.go (update tests)
  118. // Util func
  119. //
  120. // Finds how many tabs deep the line is
  121. func FindDepth(line string) int {
  122. if strings.HasPrefix(line, "\t") {
  123. depth := 0
  124. for _, c := range line {
  125. if c != '\t' {
  126. break
  127. }
  128. depth += 1
  129. }
  130. return depth
  131. }
  132. return 0
  133. }
  134. // Util func
  135. //
  136. // # Cleans the starting tabs from the line
  137. //
  138. // Utilizes FindDepth to chop off that many characters in the beginning of the line
  139. func CleanLine(line string) string {
  140. if strings.HasPrefix(line, "\t") {
  141. depth := FindDepth(line)
  142. return line[depth:]
  143. }
  144. return line
  145. }