214 lines
6.0 KiB
Go
214 lines
6.0 KiB
Go
|
package pongo2
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"io/ioutil"
|
||
|
"log"
|
||
|
"net/http"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
)
|
||
|
|
||
|
// LocalFilesystemLoader represents a local filesystem loader with basic
|
||
|
// BaseDirectory capabilities. The access to the local filesystem is unrestricted.
|
||
|
type LocalFilesystemLoader struct {
|
||
|
baseDir string
|
||
|
}
|
||
|
|
||
|
// MustNewLocalFileSystemLoader creates a new LocalFilesystemLoader instance
|
||
|
// and panics if there's any error during instantiation. The parameters
|
||
|
// are the same like NewLocalFileSystemLoader.
|
||
|
func MustNewLocalFileSystemLoader(baseDir string) *LocalFilesystemLoader {
|
||
|
fs, err := NewLocalFileSystemLoader(baseDir)
|
||
|
if err != nil {
|
||
|
log.Panic(err)
|
||
|
}
|
||
|
return fs
|
||
|
}
|
||
|
|
||
|
// NewLocalFileSystemLoader creates a new LocalFilesystemLoader and allows
|
||
|
// templatesto be loaded from disk (unrestricted). If any base directory
|
||
|
// is given (or being set using SetBaseDir), this base directory is being used
|
||
|
// for path calculation in template inclusions/imports. Otherwise the path
|
||
|
// is calculated based relatively to the including template's path.
|
||
|
func NewLocalFileSystemLoader(baseDir string) (*LocalFilesystemLoader, error) {
|
||
|
fs := &LocalFilesystemLoader{}
|
||
|
if baseDir != "" {
|
||
|
if err := fs.SetBaseDir(baseDir); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
return fs, nil
|
||
|
}
|
||
|
|
||
|
// SetBaseDir sets the template's base directory. This directory will
|
||
|
// be used for any relative path in filters, tags and From*-functions to determine
|
||
|
// your template. See the comment for NewLocalFileSystemLoader as well.
|
||
|
func (fs *LocalFilesystemLoader) SetBaseDir(path string) error {
|
||
|
// Make the path absolute
|
||
|
if !filepath.IsAbs(path) {
|
||
|
abs, err := filepath.Abs(path)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
path = abs
|
||
|
}
|
||
|
|
||
|
// Check for existence
|
||
|
fi, err := os.Stat(path)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if !fi.IsDir() {
|
||
|
return fmt.Errorf("The given path '%s' is not a directory.", path)
|
||
|
}
|
||
|
|
||
|
fs.baseDir = path
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// Get reads the path's content from your local filesystem.
|
||
|
func (fs *LocalFilesystemLoader) Get(path string) (io.Reader, error) {
|
||
|
buf, err := ioutil.ReadFile(path)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return bytes.NewReader(buf), nil
|
||
|
}
|
||
|
|
||
|
// Abs resolves a filename relative to the base directory. Absolute paths are allowed.
|
||
|
// When there's no base dir set, the absolute path to the filename
|
||
|
// will be calculated based on either the provided base directory (which
|
||
|
// might be a path of a template which includes another template) or
|
||
|
// the current working directory.
|
||
|
func (fs *LocalFilesystemLoader) Abs(base, name string) string {
|
||
|
if filepath.IsAbs(name) {
|
||
|
return name
|
||
|
}
|
||
|
|
||
|
// Our own base dir has always priority; if there's none
|
||
|
// we use the path provided in base.
|
||
|
var err error
|
||
|
if fs.baseDir == "" {
|
||
|
if base == "" {
|
||
|
base, err = os.Getwd()
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return filepath.Join(base, name)
|
||
|
}
|
||
|
|
||
|
return filepath.Join(filepath.Dir(base), name)
|
||
|
}
|
||
|
|
||
|
return filepath.Join(fs.baseDir, name)
|
||
|
}
|
||
|
|
||
|
// SandboxedFilesystemLoader is still WIP.
|
||
|
type SandboxedFilesystemLoader struct {
|
||
|
*LocalFilesystemLoader
|
||
|
}
|
||
|
|
||
|
// NewSandboxedFilesystemLoader creates a new sandboxed local file system instance.
|
||
|
func NewSandboxedFilesystemLoader(baseDir string) (*SandboxedFilesystemLoader, error) {
|
||
|
fs, err := NewLocalFileSystemLoader(baseDir)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return &SandboxedFilesystemLoader{
|
||
|
LocalFilesystemLoader: fs,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
// Move sandbox to a virtual fs
|
||
|
|
||
|
/*
|
||
|
if len(set.SandboxDirectories) > 0 {
|
||
|
defer func() {
|
||
|
// Remove any ".." or other crap
|
||
|
resolvedPath = filepath.Clean(resolvedPath)
|
||
|
|
||
|
// Make the path absolute
|
||
|
absPath, err := filepath.Abs(resolvedPath)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
resolvedPath = absPath
|
||
|
|
||
|
// Check against the sandbox directories (once one pattern matches, we're done and can allow it)
|
||
|
for _, pattern := range set.SandboxDirectories {
|
||
|
matched, err := filepath.Match(pattern, resolvedPath)
|
||
|
if err != nil {
|
||
|
panic("Wrong sandbox directory match pattern (see http://golang.org/pkg/path/filepath/#Match).")
|
||
|
}
|
||
|
if matched {
|
||
|
// OK!
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// No pattern matched, we have to log+deny the request
|
||
|
set.logf("Access attempt outside of the sandbox directories (blocked): '%s'", resolvedPath)
|
||
|
resolvedPath = ""
|
||
|
}()
|
||
|
}
|
||
|
*/
|
||
|
|
||
|
// HttpFilesystemLoader supports loading templates
|
||
|
// from an http.FileSystem - useful for using one of several
|
||
|
// file-to-code generators that packs static files into
|
||
|
// a go binary (ex: https://github.com/jteeuwen/go-bindata)
|
||
|
type HttpFilesystemLoader struct {
|
||
|
fs http.FileSystem
|
||
|
baseDir string
|
||
|
}
|
||
|
|
||
|
// MustNewHttpFileSystemLoader creates a new HttpFilesystemLoader instance
|
||
|
// and panics if there's any error during instantiation. The parameters
|
||
|
// are the same like NewHttpFilesystemLoader.
|
||
|
func MustNewHttpFileSystemLoader(httpfs http.FileSystem, baseDir string) *HttpFilesystemLoader {
|
||
|
fs, err := NewHttpFileSystemLoader(httpfs, baseDir)
|
||
|
if err != nil {
|
||
|
log.Panic(err)
|
||
|
}
|
||
|
return fs
|
||
|
}
|
||
|
|
||
|
// NewHttpFileSystemLoader creates a new HttpFileSystemLoader and allows
|
||
|
// templates to be loaded from the virtual filesystem. The path
|
||
|
// is calculated based relatively from the root of the http.Filesystem.
|
||
|
func NewHttpFileSystemLoader(httpfs http.FileSystem, baseDir string) (*HttpFilesystemLoader, error) {
|
||
|
hfs := &HttpFilesystemLoader{
|
||
|
fs: httpfs,
|
||
|
baseDir: baseDir,
|
||
|
}
|
||
|
if httpfs == nil {
|
||
|
err := errors.New("httpfs cannot be nil")
|
||
|
return nil, err
|
||
|
}
|
||
|
return hfs, nil
|
||
|
}
|
||
|
|
||
|
// Abs in this instance simply returns the filename, since
|
||
|
// there's no potential for an unexpanded path in an http.FileSystem
|
||
|
func (h *HttpFilesystemLoader) Abs(base, name string) string {
|
||
|
return name
|
||
|
}
|
||
|
|
||
|
// Get returns an io.Reader where the template's content can be read from.
|
||
|
func (h *HttpFilesystemLoader) Get(path string) (io.Reader, error) {
|
||
|
fullPath := path
|
||
|
if h.baseDir != "" {
|
||
|
fullPath = fmt.Sprintf(
|
||
|
"%s/%s",
|
||
|
h.baseDir,
|
||
|
fullPath,
|
||
|
)
|
||
|
}
|
||
|
|
||
|
return h.fs.Open(fullPath)
|
||
|
}
|