| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111 |
- package rendering
- import (
- "context"
- "fmt"
- "os"
- "os/exec"
- "path/filepath"
- "runtime"
- "strings"
- "time"
- "github.com/grafana/grafana/pkg/log"
- "github.com/grafana/grafana/pkg/middleware"
- )
- func (rs *RenderingService) renderViaPhantomJS(ctx context.Context, opts Opts) (*RenderResult, error) {
- rs.log.Info("Rendering", "path", opts.Path)
- var executable = "phantomjs"
- if runtime.GOOS == "windows" {
- executable = executable + ".exe"
- }
- url := rs.getURL(opts.Path)
- binPath, _ := filepath.Abs(filepath.Join(rs.Cfg.PhantomDir, executable))
- if _, err := os.Stat(binPath); os.IsNotExist(err) {
- rs.log.Error("executable not found", "executable", binPath)
- return nil, ErrPhantomJSNotInstalled
- }
- scriptPath, _ := filepath.Abs(filepath.Join(rs.Cfg.PhantomDir, "render.js"))
- pngPath := rs.getFilePathForNewImage()
- renderKey := middleware.AddRenderAuthKey(opts.OrgId, opts.UserId, opts.OrgRole)
- defer middleware.RemoveRenderAuthKey(renderKey)
- phantomDebugArg := "--debug=false"
- if log.GetLogLevelFor("renderer") >= log.LvlDebug {
- phantomDebugArg = "--debug=true"
- }
- cmdArgs := []string{
- "--ignore-ssl-errors=true",
- "--web-security=false",
- phantomDebugArg,
- scriptPath,
- fmt.Sprintf("url=%v", url),
- fmt.Sprintf("width=%v", opts.Width),
- fmt.Sprintf("height=%v", opts.Height),
- fmt.Sprintf("png=%v", pngPath),
- fmt.Sprintf("domain=%v", rs.domain),
- fmt.Sprintf("timeout=%v", opts.Timeout.Seconds()),
- fmt.Sprintf("renderKey=%v", renderKey),
- }
- if opts.Encoding != "" {
- cmdArgs = append([]string{fmt.Sprintf("--output-encoding=%s", opts.Encoding)}, cmdArgs...)
- }
- commandCtx, cancel := context.WithTimeout(ctx, opts.Timeout+time.Second*2)
- defer cancel()
- cmd := exec.CommandContext(commandCtx, binPath, cmdArgs...)
- cmd.Stderr = cmd.Stdout
- if opts.Timezone != "" {
- baseEnviron := os.Environ()
- cmd.Env = appendEnviron(baseEnviron, "TZ", isoTimeOffsetToPosixTz(opts.Timezone))
- }
- out, err := cmd.Output()
- // check for timeout first
- if commandCtx.Err() == context.DeadlineExceeded {
- rs.log.Info("Rendering timed out")
- return nil, ErrTimeout
- }
- if err != nil {
- rs.log.Error("Phantomjs exited with non zero exit code", "error", err)
- return nil, err
- }
- rs.log.Debug("Phantomjs output", "out", string(out))
- rs.log.Debug("Image rendered", "path", pngPath)
- return &RenderResult{FilePath: pngPath}, nil
- }
- func isoTimeOffsetToPosixTz(isoOffset string) string {
- // invert offset
- if strings.HasPrefix(isoOffset, "UTC+") {
- return strings.Replace(isoOffset, "UTC+", "UTC-", 1)
- }
- if strings.HasPrefix(isoOffset, "UTC-") {
- return strings.Replace(isoOffset, "UTC-", "UTC+", 1)
- }
- return isoOffset
- }
- func appendEnviron(baseEnviron []string, name string, value string) []string {
- results := make([]string, 0)
- prefix := fmt.Sprintf("%s=", name)
- for _, v := range baseEnviron {
- if !strings.HasPrefix(v, prefix) {
- results = append(results, v)
- }
- }
- return append(results, fmt.Sprintf("%s=%s", name, value))
- }
|