bind.go 2.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. package main
  2. import (
  3. "bufio"
  4. "fmt"
  5. "os"
  6. "strings"
  7. )
  8. func parseFunc(val string) value {
  9. if !isList(val) {
  10. return value{body: errNop}
  11. }
  12. list := strings.Split(val, " ")
  13. if list[1] != "[" {
  14. return value{body: errNop}
  15. }
  16. i := 2
  17. for ; list[i] != "]"; i++ {
  18. if !isName(`"` + list[i]) {
  19. return value{body: errNop}
  20. }
  21. }
  22. if i == len(list)-2 {
  23. return value{body: errNop}
  24. }
  25. j, bracketCnt := i+1, 0
  26. for ; j < len(list)-1; j++ {
  27. if list[j] == "[" {
  28. bracketCnt++
  29. }
  30. if list[j] == "]" {
  31. bracketCnt--
  32. }
  33. if bracketCnt == 0 && j < len(list)-2 {
  34. return value{body: errNop}
  35. }
  36. }
  37. local := make(environ)
  38. for name, val := range env[len(env)-1] {
  39. local[name] = val
  40. }
  41. return value{
  42. val: val,
  43. param: list[2:i],
  44. body: strings.Join(list[i+1:len(list)-1], " "),
  45. env: local}
  46. }
  47. func getFunc(name string) value {
  48. if !isName(`"` + name) {
  49. return value{body: errNop}
  50. }
  51. envs := []int{len(env) - 1, len(env) - 2, 1}
  52. for _, id := range envs {
  53. val, ok := env[id][name]
  54. if ok {
  55. return val
  56. }
  57. }
  58. return value{body: errNop}
  59. }
  60. func opMake(val1, val2 value) value {
  61. name := toName(val1.val)
  62. _, ok := reserved[name]
  63. if ok {
  64. panic(fmt.Errorf("make: %s (%s)", errNameReserved, name))
  65. }
  66. if isList(val2.val) && val2.body == "" {
  67. val2 = parseFunc(val2.val)
  68. }
  69. env[len(env)-1][name] = val2
  70. return val2
  71. }
  72. func opThing(val1 value) value {
  73. name := toName(val1.val)
  74. envs := []int{len(env) - 1, len(env) - 2, 1}
  75. for _, id := range envs {
  76. val, ok := env[id][name]
  77. if ok {
  78. return val
  79. }
  80. }
  81. panic(fmt.Errorf("thing: %s (%s)", errNameNotFound, name))
  82. }
  83. func opErase(val1 value) value {
  84. name := toName(val1.val)
  85. val, ok := env[len(env)-1][name]
  86. if ok {
  87. delete(env[len(env)-1], name)
  88. return val
  89. }
  90. panic(fmt.Errorf("erase: %s (%s)", errNameNotFound, name))
  91. }
  92. func opExport(val1 value) value {
  93. name := toName(val1.val)
  94. val, ok := env[len(env)-1][name]
  95. if ok {
  96. env[1][name] = val
  97. return val
  98. }
  99. panic(fmt.Errorf("export: %s (%s)", errNameNotFound, name))
  100. }
  101. func opSave(val1 value) value {
  102. name := toName(val1.val)
  103. s := ""
  104. for name, val := range env[len(env)-1] {
  105. s += `make "` + name + " " + val.val + "\n"
  106. }
  107. err := os.WriteFile(name, []byte(s), 0666)
  108. if err != nil {
  109. panic(fmt.Errorf("save: %s (%s)", errFileError, name))
  110. }
  111. return val1
  112. }
  113. func opLoad(val1 value) value {
  114. name := toName(val1.val)
  115. s, err := os.ReadFile(name)
  116. if err != nil {
  117. panic(fmt.Errorf("load: %s (%s)", errFileError, name))
  118. }
  119. interpret(bufio.NewScanner(strings.NewReader(string(s))))
  120. return value{val: "true"}
  121. }
  122. func opErAll() value {
  123. env[len(env)-1] = make(environ)
  124. return value{val: "true"}
  125. }
  126. func opPoAll() value {
  127. list := []string{}
  128. for name := range env[len(env)-1] {
  129. list = append(list, name)
  130. }
  131. return value{val: "[ " + strings.Join(list, " ") + " ]"}
  132. }