1
0

svg.go 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. package svg
  2. import (
  3. "bytes"
  4. "errors"
  5. "io"
  6. "strings"
  7. "sync"
  8. "github.com/tdewolff/parse/v2"
  9. "github.com/tdewolff/parse/v2/xml"
  10. "github.com/imgproxy/imgproxy/v3/imagedata"
  11. "github.com/imgproxy/imgproxy/v3/imagetype"
  12. )
  13. var pool = sync.Pool{
  14. New: func() any {
  15. return bytes.NewBuffer(nil)
  16. },
  17. }
  18. func Sanitize(data imagedata.ImageData) (imagedata.ImageData, error) {
  19. r := data.Reader()
  20. l := xml.NewLexer(parse.NewInput(r))
  21. buf, ok := pool.Get().(*bytes.Buffer)
  22. if !ok {
  23. return nil, newSanitizeError(errors.New("svg.Sanitize: failed to get buffer from pool"))
  24. }
  25. buf.Reset()
  26. cancel := func() {
  27. pool.Put(buf)
  28. }
  29. ignoreTag := 0
  30. var curTagName string
  31. for {
  32. tt, tdata := l.Next()
  33. if tt == xml.ErrorToken {
  34. if l.Err() != io.EOF {
  35. cancel()
  36. return nil, newSanitizeError(l.Err())
  37. }
  38. break
  39. }
  40. if ignoreTag > 0 {
  41. switch tt {
  42. case xml.EndTagToken, xml.StartTagCloseVoidToken:
  43. ignoreTag--
  44. case xml.StartTagToken:
  45. ignoreTag++
  46. }
  47. continue
  48. }
  49. switch tt {
  50. case xml.StartTagToken:
  51. curTagName = strings.ToLower(string(l.Text()))
  52. if curTagName == "script" {
  53. ignoreTag++
  54. continue
  55. }
  56. buf.Write(tdata)
  57. case xml.AttributeToken:
  58. attrName := strings.ToLower(string(l.Text()))
  59. if _, unsafe := unsafeAttrs[attrName]; unsafe {
  60. continue
  61. }
  62. if curTagName == "use" && (attrName == "href" || attrName == "xlink:href") {
  63. val := strings.TrimSpace(strings.Trim(string(l.AttrVal()), `"'`))
  64. if len(val) > 0 && val[0] != '#' {
  65. continue
  66. }
  67. }
  68. buf.Write(tdata)
  69. default:
  70. buf.Write(tdata)
  71. }
  72. }
  73. newData := imagedata.NewFromBytesWithFormat(
  74. imagetype.SVG,
  75. buf.Bytes(),
  76. )
  77. newData.AddCancel(cancel)
  78. return newData, nil
  79. }