write.go 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. package door
  2. import (
  3. "bytes"
  4. "fmt"
  5. "log"
  6. "regexp"
  7. "strconv"
  8. "strings"
  9. )
  10. var FindANSIColor *regexp.Regexp
  11. func init() {
  12. FindANSIColor = regexp.MustCompile("\x1b\\[([0-9;]*)m")
  13. }
  14. /*
  15. find \x1b[0;1;37;40m codes.
  16. return array of start/end positions in the given string.
  17. */
  18. func find_ansicolor(text string) [][]int {
  19. var color_codes [][]int
  20. // word, _ := regexp.Compile("\x1b\\[([0-9;]+)m")
  21. var colors [][]int = FindANSIColor.FindAllStringIndex(text, -1)
  22. // regexp seems to be ignoring the capture groups.
  23. // colors := word.FindAllSubmatchIndex([]byte(text), len(text))
  24. for _, pos := range colors {
  25. var txt string = text[pos[0]+2 : pos[1]-1]
  26. if txt == "" {
  27. txt = "0"
  28. }
  29. // log.Printf("Text: [%s]\n", txt)
  30. var codes []string = strings.Split(txt, ";")
  31. // log.Printf("Codes: [%#v]\n", codes)
  32. var code []int = make([]int, len(codes))
  33. for idx, c := range codes {
  34. var err error
  35. code[idx], err = strconv.Atoi(c)
  36. if err != nil {
  37. log.Printf("Atoi: %#v [%s]\n", err, c)
  38. }
  39. color_codes = append(color_codes, code)
  40. }
  41. }
  42. return color_codes
  43. }
  44. type ANSIColorParts struct {
  45. Bold bool
  46. Blink bool
  47. FG int
  48. BG int
  49. }
  50. // Parse an array of codes into their ANSIColorParts.
  51. func ParseColorArray(colorarray []int) ANSIColorParts {
  52. var acp ANSIColorParts
  53. acp.FG = -1
  54. acp.BG = -1
  55. if len(colorarray) == 0 {
  56. colorarray = []int{0}
  57. }
  58. for _, c := range colorarray {
  59. switch c {
  60. case 0:
  61. acp.FG = 7
  62. acp.BG = 0
  63. acp.Bold = false
  64. acp.Blink = false
  65. case 1:
  66. acp.Bold = true
  67. case 5:
  68. acp.Blink = true
  69. case 30, 31, 32, 33, 34, 35, 36, 37:
  70. acp.FG = c - 30
  71. case 40, 41, 42, 43, 44, 45, 46, 47:
  72. acp.BG = c - 40
  73. }
  74. }
  75. return acp
  76. }
  77. // Update the LastColor with the latest codes.
  78. func (d *Door) UpdateLastColor(output string, LastColor *[]int) {
  79. // use the last color information + whatever is in the string to
  80. // track the last color set
  81. var updated ANSIColorParts = ParseColorArray(*LastColor)
  82. var colors [][]int = find_ansicolor(output)
  83. for _, codes := range colors {
  84. if codes[0] == 0 {
  85. updated = ParseColorArray(codes)
  86. } else {
  87. newCode := ParseColorArray(codes)
  88. if newCode.Bold {
  89. updated.Bold = true
  90. }
  91. if newCode.Blink {
  92. updated.Blink = true
  93. }
  94. if (newCode.FG != -1) && (newCode.FG != updated.FG) {
  95. updated.FG = newCode.FG
  96. }
  97. if (newCode.BG != -1) && (newCode.BG != updated.BG) {
  98. updated.BG = newCode.BG
  99. }
  100. }
  101. }
  102. *LastColor = make([]int, 1)
  103. (*LastColor)[0] = 0
  104. if updated.Blink {
  105. *LastColor = append(*LastColor, 5)
  106. }
  107. if updated.Bold {
  108. *LastColor = append(*LastColor, 1)
  109. }
  110. if updated.FG != -1 {
  111. *LastColor = append(*LastColor, updated.FG+30)
  112. }
  113. if updated.BG != -1 {
  114. *LastColor = append(*LastColor, updated.BG+40)
  115. }
  116. }
  117. /*
  118. reading from a closed channel is easy to detect.
  119. res, ok := <-channel
  120. ok == false
  121. writing to a closed channel is a panic.
  122. Have the Write manage itself.
  123. */
  124. // For now, update is SavePos + output + RestorePos.
  125. // FUTURE: Access the screen struct to "save" Color+Attr+Pos.
  126. func (d *Door) Update(output []byte) {
  127. // sync.Pool ?
  128. var buffer bytes.Buffer
  129. buffer.WriteString(SAVE_POS)
  130. buffer.Write(output)
  131. buffer.WriteString(RESTORE_POS)
  132. d.Writer.Write(buffer.Bytes())
  133. // d.Write(SAVE_POS + output + RESTORE_POS)
  134. }
  135. func (d *Door) WriteS(output string) {
  136. d.Write([]byte(output))
  137. }
  138. func (d *Door) WriteA(a ...interface{}) {
  139. d.writeMutex.Lock()
  140. defer d.writeMutex.Unlock()
  141. for _, item := range a {
  142. switch item.(type) {
  143. case string:
  144. d.LockedWrite([]byte(item.(string)))
  145. case []byte:
  146. d.LockedWrite(item.([]byte))
  147. case *bytes.Buffer:
  148. d.LockedWrite(item.(*bytes.Buffer).Bytes())
  149. default:
  150. log.Printf("Unknown/unsupported type: %T\n", item)
  151. }
  152. }
  153. }
  154. func EndCSI(c byte) bool {
  155. return (c >= 0x40) && (c <= 0x7f)
  156. }
  157. func (d *Door) ANSIProcess() {
  158. var csi byte = d.ansiCode[len(d.ansiCode)-1]
  159. switch csi {
  160. case 's':
  161. // Save Color
  162. var slen int = len(d.LastColor)
  163. if cap(d.Writer.LastSavedColor) < slen {
  164. d.Writer.LastSavedColor = make([]int, slen)
  165. } else {
  166. d.Writer.LastSavedColor = d.Writer.LastSavedColor[0:slen]
  167. }
  168. copy(d.Writer.LastSavedColor, d.LastColor)
  169. log.Printf("ColorSave: %d\n", d.Writer.LastSavedColor)
  170. case 'u':
  171. // Restore Color
  172. var slen int = len(d.Writer.LastSavedColor)
  173. if cap(d.LastColor) < slen {
  174. d.LastColor = make([]int, slen)
  175. } else {
  176. d.LastColor = d.LastColor[0:slen]
  177. }
  178. copy(d.LastColor, d.Writer.LastSavedColor)
  179. log.Printf("ColorRestore: %d\n", d.LastColor)
  180. case 'm':
  181. // Process color code
  182. var color [5]int
  183. var cpos int
  184. var code int
  185. var pos int
  186. for pos = 0; pos < len(d.ansiCode); pos++ {
  187. var b byte = d.ansiCode[pos]
  188. switch b {
  189. case ';':
  190. color[cpos] = code
  191. code = 0
  192. cpos++
  193. case '1', '2', '3', '4', '5', '6', '7', '8', '9', '0':
  194. code *= 10
  195. code += int(b - '0')
  196. }
  197. }
  198. if code != 0 {
  199. color[cpos] = code
  200. code = 0
  201. cpos++
  202. }
  203. // Make sure there is always at least one code here.
  204. if cpos == 0 {
  205. color[cpos] = 0
  206. cpos++
  207. }
  208. log.Printf("color: [%d]", color[0:cpos])
  209. var newColor = ParseColorArray(d.LastColor)
  210. for _, c := range color[0:cpos] {
  211. switch c {
  212. case 0:
  213. newColor.FG = 7
  214. newColor.BG = 0
  215. newColor.Bold = false
  216. newColor.Blink = false
  217. case 1:
  218. newColor.Bold = true
  219. case 5:
  220. newColor.Blink = true
  221. case 30, 31, 32, 33, 34, 35, 36, 37:
  222. newColor.FG = c - 30
  223. case 40, 41, 42, 43, 44, 45, 46, 47:
  224. newColor.BG = c - 40
  225. }
  226. }
  227. // Reset LastColor
  228. d.LastColor = d.LastColor[0:0]
  229. d.LastColor = append(d.LastColor, 0)
  230. if newColor.Blink {
  231. d.LastColor = append(d.LastColor, 5)
  232. }
  233. if newColor.Bold {
  234. d.LastColor = append(d.LastColor, 1)
  235. }
  236. if newColor.FG != -1 {
  237. d.LastColor = append(d.LastColor, newColor.FG+30)
  238. }
  239. if newColor.BG != -1 {
  240. d.LastColor = append(d.LastColor, newColor.BG+40)
  241. }
  242. log.Printf("LastColor: [%d]", d.LastColor)
  243. }
  244. var outputByte [20]byte
  245. var output []byte = outputByte[0:0]
  246. output = fmt.Appendf(output, "ANSI: [%q]\n", d.ansiCode)
  247. log.Printf("%s", output)
  248. d.ansiCode = d.ansiCode[0:0]
  249. d.ansiCSI = false
  250. }
  251. func (d *Door) ANSIScan(output []byte) {
  252. var pos, nextpos int
  253. var olen int = len(output)
  254. var c byte
  255. if d.ansiCSI {
  256. for pos < olen {
  257. c = output[pos]
  258. d.ansiCode = append(d.ansiCode, c)
  259. pos++
  260. if EndCSI(c) {
  261. d.ANSIProcess()
  262. break
  263. }
  264. }
  265. if pos == olen {
  266. return
  267. // pos == -1 // end
  268. }
  269. }
  270. for pos != -1 {
  271. nextpos = bytes.Index(output[pos:], []byte{'\x1b'})
  272. if nextpos != -1 {
  273. // Found ESC
  274. nextpos += pos + 1
  275. if output[nextpos] == '[' {
  276. d.ansiCSI = true
  277. nextpos++
  278. for nextpos < olen {
  279. c = output[nextpos]
  280. d.ansiCode = append(d.ansiCode, c)
  281. nextpos++
  282. if EndCSI(c) {
  283. d.ANSIProcess()
  284. break
  285. }
  286. }
  287. }
  288. pos = nextpos
  289. } else {
  290. return
  291. }
  292. }
  293. }
  294. func (d *Door) LockedWrite(output []byte) {
  295. if d.writeMutex.TryLock() {
  296. log.Panic("LockedWrite: mutex was NOT locked.")
  297. }
  298. // var lastColorBytes [32]byte
  299. var lastColor string // []byte = lastColorBytes[0:0]
  300. /*
  301. if (bytes.HasPrefix(output, []byte(SavePos))) &&
  302. (bytes.HasSuffix(output, []byte(RestorePos))) {
  303. */
  304. /*
  305. if bytes.HasSuffix(output, []byte(RestorePos)) {
  306. // Write the current color
  307. lastColor = Color(d.LastColor)
  308. //fmt.Append(lastColor, []byte(Color(d.LastColor)))
  309. log.Printf("Restore LastColor: %d => %q", d.LastColor, lastColor)
  310. }
  311. */
  312. log.Printf(">> %q\n", output)
  313. d.Writer.Write(output)
  314. d.ANSIScan(output)
  315. if bytes.HasSuffix(output, []byte(RestorePos)) {
  316. lastColor = Color(d.LastColor)
  317. //fmt.Append(lastColor, []byte(Color(d.LastColor)))
  318. log.Printf("Restore LastColor: %d => %q", d.LastColor, lastColor)
  319. d.Writer.Write([]byte(lastColor))
  320. d.ANSIScan([]byte(lastColor))
  321. }
  322. /*
  323. if len(lastColor) != 0 {
  324. log.Println("Restored HERE...")
  325. d.Writer.Write([]byte(lastColor))
  326. d.ANSIScan([]byte(lastColor))
  327. }
  328. */
  329. // else {
  330. // Only update if we're NOT restoring...
  331. // Update
  332. // d.ANSIScan(output)
  333. // }
  334. }
  335. func (d *Door) Write(output []byte) {
  336. if len(output) == 0 {
  337. return
  338. }
  339. d.writeMutex.Lock()
  340. defer d.writeMutex.Unlock()
  341. d.LockedWrite(output)
  342. }