phantom_renderer.go 1.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667
  1. package components
  2. import (
  3. "crypto/md5"
  4. "encoding/hex"
  5. "io"
  6. "os"
  7. "os/exec"
  8. "path/filepath"
  9. "time"
  10. log "github.com/alecthomas/log4go"
  11. )
  12. type PhantomRenderer struct {
  13. ImagesDir string
  14. PhantomDir string
  15. }
  16. func (self *PhantomRenderer) RenderToPng(url string) (string, error) {
  17. log.Info("PhantomRenderer::renderToPng url %v", url)
  18. binPath, _ := filepath.Abs(filepath.Join(self.PhantomDir, "phantomjs"))
  19. scriptPath, _ := filepath.Abs(filepath.Join(self.PhantomDir, "render.js"))
  20. pngPath, _ := filepath.Abs(filepath.Join(self.ImagesDir, getHash(url)))
  21. pngPath = pngPath + ".png"
  22. cmd := exec.Command(binPath, scriptPath, "url="+url, "width=100", "height=100", "png="+pngPath)
  23. stdout, err := cmd.StdoutPipe()
  24. if err != nil {
  25. return "", err
  26. }
  27. stderr, err := cmd.StderrPipe()
  28. if err != nil {
  29. return "", err
  30. }
  31. err = cmd.Start()
  32. if err != nil {
  33. return "", err
  34. }
  35. go io.Copy(os.Stdout, stdout)
  36. go io.Copy(os.Stdout, stderr)
  37. done := make(chan error)
  38. go func() {
  39. cmd.Wait()
  40. close(done)
  41. }()
  42. select {
  43. case <-time.After(10 * time.Second):
  44. if err := cmd.Process.Kill(); err != nil {
  45. log.Error("failed to kill: %v", err)
  46. }
  47. case <-done:
  48. }
  49. return pngPath, nil
  50. }
  51. func getHash(text string) string {
  52. hasher := md5.New()
  53. hasher.Write([]byte(text))
  54. return hex.EncodeToString(hasher.Sum(nil))
  55. }