123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125 |
- package frankenphp
- import (
- "github.com/dunglas/frankenphp/internal/fastabs"
- "net/http"
- "path/filepath"
- "sync"
- "sync/atomic"
- "go.uber.org/zap"
- )
- // RequestOption instances allow to configure a FrankenPHP Request.
- type RequestOption func(h *FrankenPHPContext) error
- var (
- documentRootCache sync.Map
- documentRootCacheLen atomic.Uint32
- )
- // WithRequestDocumentRoot sets the root directory of the PHP application.
- // if resolveSymlink is true, oath declared as root directory will be resolved
- // to its absolute value after the evaluation of any symbolic links.
- // Due to the nature of PHP opcache, root directory path is cached: when
- // using a symlinked directory as root this could generate errors when
- // symlink is changed without PHP being restarted; enabling this
- // directive will set $_SERVER['DOCUMENT_ROOT'] to the real directory path.
- func WithRequestDocumentRoot(documentRoot string, resolveSymlink bool) RequestOption {
- return func(o *FrankenPHPContext) (err error) {
- v, ok := documentRootCache.Load(documentRoot)
- if !ok {
- // make sure file root is absolute
- v, err = fastabs.FastAbs(documentRoot)
- if err != nil {
- return err
- }
- // prevent the cache to grow forever, this is a totally arbitrary value
- if documentRootCacheLen.Load() < 1024 {
- documentRootCache.LoadOrStore(documentRoot, v)
- documentRootCacheLen.Add(1)
- }
- }
- if resolveSymlink {
- if v, err = filepath.EvalSymlinks(v.(string)); err != nil {
- return err
- }
- }
- o.documentRoot = v.(string)
- return nil
- }
- }
- // WithRequestResolvedDocumentRoot is similar to WithRequestDocumentRoot
- // but doesn't do any checks or resolving on the path to improve performance.
- func WithRequestResolvedDocumentRoot(documentRoot string) RequestOption {
- return func(o *FrankenPHPContext) error {
- o.documentRoot = documentRoot
- return nil
- }
- }
- // WithRequestSplitPath contains a list of split path strings.
- //
- // The path in the URL will be split into two, with the first piece ending
- // with the value of splitPath. The first piece will be assumed as the
- // actual resource (CGI script) name, and the second piece will be set to
- // PATH_INFO for the CGI script to use.
- //
- // Future enhancements should be careful to avoid CVE-2019-11043,
- // which can be mitigated with use of a try_files-like behavior
- // that 404s if the FastCGI path info is not found.
- func WithRequestSplitPath(splitPath []string) RequestOption {
- return func(o *FrankenPHPContext) error {
- o.splitPath = splitPath
- return nil
- }
- }
- type PreparedEnv = map[string]string
- func PrepareEnv(env map[string]string) PreparedEnv {
- preparedEnv := make(PreparedEnv, len(env))
- for k, v := range env {
- preparedEnv[k+"\x00"] = v
- }
- return preparedEnv
- }
- // WithRequestEnv set CGI-like environment variables that will be available in $_SERVER.
- // Values set with WithEnv always have priority over automatically populated values.
- func WithRequestEnv(env map[string]string) RequestOption {
- return WithRequestPreparedEnv(PrepareEnv(env))
- }
- func WithRequestPreparedEnv(env PreparedEnv) RequestOption {
- return func(o *FrankenPHPContext) error {
- o.env = env
- return nil
- }
- }
- func WithOriginalRequest(r *http.Request) RequestOption {
- return func(o *FrankenPHPContext) error {
- o.originalRequest = r
- return nil
- }
- }
- // WithRequestLogger sets the logger associated with the current request
- func WithRequestLogger(logger *zap.Logger) RequestOption {
- return func(o *FrankenPHPContext) error {
- o.logger = logger
- return nil
- }
- }
|