mirror of
https://github.com/imgproxy/imgproxy.git
synced 2024-12-09 09:56:01 +02:00
543 lines
16 KiB
Go
543 lines
16 KiB
Go
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/hex"
|
|
"flag"
|
|
"fmt"
|
|
"math"
|
|
"os"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
)
|
|
|
|
func intEnvConfig(i *int, name string) {
|
|
if env, err := strconv.Atoi(os.Getenv(name)); err == nil {
|
|
*i = env
|
|
}
|
|
}
|
|
|
|
func floatEnvConfig(i *float64, name string) {
|
|
if env, err := strconv.ParseFloat(os.Getenv(name), 64); err == nil {
|
|
*i = env
|
|
}
|
|
}
|
|
|
|
func megaIntEnvConfig(f *int, name string) {
|
|
if env, err := strconv.ParseFloat(os.Getenv(name), 64); err == nil {
|
|
*f = int(env * 1000000)
|
|
}
|
|
}
|
|
|
|
func strEnvConfig(s *string, name string) {
|
|
if env := os.Getenv(name); len(env) > 0 {
|
|
*s = env
|
|
}
|
|
}
|
|
|
|
func strSliceEnvConfig(s *[]string, name string) {
|
|
if env := os.Getenv(name); len(env) > 0 {
|
|
parts := strings.Split(env, ",")
|
|
|
|
for i, p := range parts {
|
|
parts[i] = strings.TrimSpace(p)
|
|
}
|
|
|
|
*s = parts
|
|
|
|
return
|
|
}
|
|
|
|
*s = []string{}
|
|
}
|
|
|
|
func boolEnvConfig(b *bool, name string) {
|
|
if env, err := strconv.ParseBool(os.Getenv(name)); err == nil {
|
|
*b = env
|
|
}
|
|
}
|
|
|
|
func hexEnvConfig(b *[]securityKey, name string) error {
|
|
var err error
|
|
|
|
if env := os.Getenv(name); len(env) > 0 {
|
|
parts := strings.Split(env, ",")
|
|
|
|
keys := make([]securityKey, len(parts))
|
|
|
|
for i, part := range parts {
|
|
if keys[i], err = hex.DecodeString(part); err != nil {
|
|
return fmt.Errorf("%s expected to be hex-encoded strings. Invalid: %s\n", name, part)
|
|
}
|
|
}
|
|
|
|
*b = keys
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func hexFileConfig(b *[]securityKey, filepath string) error {
|
|
if len(filepath) == 0 {
|
|
return nil
|
|
}
|
|
|
|
f, err := os.Open(filepath)
|
|
if err != nil {
|
|
return fmt.Errorf("Can't open file %s\n", filepath)
|
|
}
|
|
|
|
keys := []securityKey{}
|
|
|
|
scanner := bufio.NewScanner(f)
|
|
for scanner.Scan() {
|
|
part := scanner.Text()
|
|
|
|
if len(part) == 0 {
|
|
continue
|
|
}
|
|
|
|
if key, err := hex.DecodeString(part); err == nil {
|
|
keys = append(keys, key)
|
|
} else {
|
|
return fmt.Errorf("%s expected to contain hex-encoded strings. Invalid: %s\n", filepath, part)
|
|
}
|
|
}
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
return fmt.Errorf("Failed to read file %s: %s", filepath, err)
|
|
}
|
|
|
|
*b = keys
|
|
|
|
return nil
|
|
}
|
|
|
|
func presetEnvConfig(p presets, name string) error {
|
|
if env := os.Getenv(name); len(env) > 0 {
|
|
presetStrings := strings.Split(env, ",")
|
|
|
|
for _, presetStr := range presetStrings {
|
|
if err := parsePreset(p, presetStr); err != nil {
|
|
return fmt.Errorf(err.Error())
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func presetFileConfig(p presets, filepath string) error {
|
|
if len(filepath) == 0 {
|
|
return nil
|
|
}
|
|
|
|
f, err := os.Open(filepath)
|
|
if err != nil {
|
|
return fmt.Errorf("Can't open file %s\n", filepath)
|
|
}
|
|
|
|
scanner := bufio.NewScanner(f)
|
|
for scanner.Scan() {
|
|
if err := parsePreset(p, scanner.Text()); err != nil {
|
|
return fmt.Errorf(err.Error())
|
|
}
|
|
}
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
return fmt.Errorf("Failed to read presets file: %s", err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
type config struct {
|
|
Network string
|
|
Bind string
|
|
ReadTimeout int
|
|
WriteTimeout int
|
|
KeepAliveTimeout int
|
|
DownloadTimeout int
|
|
Concurrency int
|
|
MaxClients int
|
|
|
|
TTL int
|
|
CacheControlPassthrough bool
|
|
|
|
SoReuseport bool
|
|
|
|
PathPrefix string
|
|
|
|
MaxSrcDimension int
|
|
MaxSrcResolution int
|
|
MaxSrcFileSize int
|
|
MaxAnimationFrames int
|
|
MaxSvgCheckBytes int
|
|
|
|
JpegProgressive bool
|
|
PngInterlaced bool
|
|
PngQuantize bool
|
|
PngQuantizationColors int
|
|
Quality int
|
|
GZipCompression int
|
|
StripMetadata bool
|
|
|
|
EnableWebpDetection bool
|
|
EnforceWebp bool
|
|
EnableClientHints bool
|
|
|
|
UseLinearColorspace bool
|
|
DisableShrinkOnLoad bool
|
|
|
|
Keys []securityKey
|
|
Salts []securityKey
|
|
AllowInsecure bool
|
|
SignatureSize int
|
|
|
|
Secret string
|
|
|
|
AllowOrigin string
|
|
|
|
UserAgent string
|
|
|
|
IgnoreSslVerification bool
|
|
DevelopmentErrorsMode bool
|
|
|
|
AllowedSources []string
|
|
LocalFileSystemRoot string
|
|
S3Enabled bool
|
|
S3Region string
|
|
S3Endpoint string
|
|
GCSEnabled bool
|
|
GCSKey string
|
|
|
|
ETagEnabled bool
|
|
|
|
BaseURL string
|
|
|
|
Presets presets
|
|
OnlyPresets bool
|
|
|
|
WatermarkData string
|
|
WatermarkPath string
|
|
WatermarkURL string
|
|
WatermarkOpacity float64
|
|
|
|
NewRelicAppName string
|
|
NewRelicKey string
|
|
|
|
PrometheusBind string
|
|
|
|
BugsnagKey string
|
|
BugsnagStage string
|
|
HoneybadgerKey string
|
|
HoneybadgerEnv string
|
|
SentryDSN string
|
|
SentryEnvironment string
|
|
SentryRelease string
|
|
|
|
ReportDownloadingErrors bool
|
|
|
|
FreeMemoryInterval int
|
|
DownloadBufferSize int
|
|
GZipBufferSize int
|
|
BufferPoolCalibrationThreshold int
|
|
}
|
|
|
|
var conf = config{
|
|
Network: "tcp",
|
|
Bind: ":8080",
|
|
ReadTimeout: 10,
|
|
WriteTimeout: 10,
|
|
KeepAliveTimeout: 10,
|
|
DownloadTimeout: 5,
|
|
Concurrency: runtime.NumCPU() * 2,
|
|
TTL: 3600,
|
|
MaxSrcResolution: 16800000,
|
|
MaxAnimationFrames: 1,
|
|
MaxSvgCheckBytes: 32 * 1024,
|
|
SignatureSize: 32,
|
|
PngQuantizationColors: 256,
|
|
Quality: 80,
|
|
StripMetadata: true,
|
|
UserAgent: fmt.Sprintf("imgproxy/%s", version),
|
|
Presets: make(presets),
|
|
WatermarkOpacity: 1,
|
|
BugsnagStage: "production",
|
|
HoneybadgerEnv: "production",
|
|
SentryEnvironment: "production",
|
|
SentryRelease: fmt.Sprintf("imgproxy/%s", version),
|
|
ReportDownloadingErrors: true,
|
|
FreeMemoryInterval: 10,
|
|
BufferPoolCalibrationThreshold: 1024,
|
|
}
|
|
|
|
func configure() error {
|
|
keyPath := flag.String("keypath", "", "path of the file with hex-encoded key")
|
|
saltPath := flag.String("saltpath", "", "path of the file with hex-encoded salt")
|
|
presetsPath := flag.String("presets", "", "path of the file with presets")
|
|
flag.Parse()
|
|
|
|
if port := os.Getenv("PORT"); len(port) > 0 {
|
|
conf.Bind = fmt.Sprintf(":%s", port)
|
|
}
|
|
|
|
strEnvConfig(&conf.Network, "IMGPROXY_NETWORK")
|
|
strEnvConfig(&conf.Bind, "IMGPROXY_BIND")
|
|
intEnvConfig(&conf.ReadTimeout, "IMGPROXY_READ_TIMEOUT")
|
|
intEnvConfig(&conf.WriteTimeout, "IMGPROXY_WRITE_TIMEOUT")
|
|
intEnvConfig(&conf.KeepAliveTimeout, "IMGPROXY_KEEP_ALIVE_TIMEOUT")
|
|
intEnvConfig(&conf.DownloadTimeout, "IMGPROXY_DOWNLOAD_TIMEOUT")
|
|
intEnvConfig(&conf.Concurrency, "IMGPROXY_CONCURRENCY")
|
|
intEnvConfig(&conf.MaxClients, "IMGPROXY_MAX_CLIENTS")
|
|
|
|
intEnvConfig(&conf.TTL, "IMGPROXY_TTL")
|
|
boolEnvConfig(&conf.CacheControlPassthrough, "IMGPROXY_CACHE_CONTROL_PASSTHROUGH")
|
|
|
|
boolEnvConfig(&conf.SoReuseport, "IMGPROXY_SO_REUSEPORT")
|
|
|
|
strEnvConfig(&conf.PathPrefix, "IMGPROXY_PATH_PREFIX")
|
|
|
|
intEnvConfig(&conf.MaxSrcDimension, "IMGPROXY_MAX_SRC_DIMENSION")
|
|
megaIntEnvConfig(&conf.MaxSrcResolution, "IMGPROXY_MAX_SRC_RESOLUTION")
|
|
intEnvConfig(&conf.MaxSrcFileSize, "IMGPROXY_MAX_SRC_FILE_SIZE")
|
|
intEnvConfig(&conf.MaxSvgCheckBytes, "IMGPROXY_MAX_SVG_CHECK_BYTES")
|
|
|
|
if _, ok := os.LookupEnv("IMGPROXY_MAX_GIF_FRAMES"); ok {
|
|
logWarning("`IMGPROXY_MAX_GIF_FRAMES` is deprecated and will be removed in future versions. Use `IMGPROXY_MAX_ANIMATION_FRAMES` instead")
|
|
intEnvConfig(&conf.MaxAnimationFrames, "IMGPROXY_MAX_GIF_FRAMES")
|
|
}
|
|
intEnvConfig(&conf.MaxAnimationFrames, "IMGPROXY_MAX_ANIMATION_FRAMES")
|
|
|
|
strSliceEnvConfig(&conf.AllowedSources, "IMGPROXY_ALLOWED_SOURCES")
|
|
|
|
boolEnvConfig(&conf.JpegProgressive, "IMGPROXY_JPEG_PROGRESSIVE")
|
|
boolEnvConfig(&conf.PngInterlaced, "IMGPROXY_PNG_INTERLACED")
|
|
boolEnvConfig(&conf.PngQuantize, "IMGPROXY_PNG_QUANTIZE")
|
|
intEnvConfig(&conf.PngQuantizationColors, "IMGPROXY_PNG_QUANTIZATION_COLORS")
|
|
intEnvConfig(&conf.Quality, "IMGPROXY_QUALITY")
|
|
intEnvConfig(&conf.GZipCompression, "IMGPROXY_GZIP_COMPRESSION")
|
|
boolEnvConfig(&conf.StripMetadata, "IMGPROXY_STRIP_METADATA")
|
|
|
|
boolEnvConfig(&conf.EnableWebpDetection, "IMGPROXY_ENABLE_WEBP_DETECTION")
|
|
boolEnvConfig(&conf.EnforceWebp, "IMGPROXY_ENFORCE_WEBP")
|
|
boolEnvConfig(&conf.EnableClientHints, "IMGPROXY_ENABLE_CLIENT_HINTS")
|
|
|
|
boolEnvConfig(&conf.UseLinearColorspace, "IMGPROXY_USE_LINEAR_COLORSPACE")
|
|
boolEnvConfig(&conf.DisableShrinkOnLoad, "IMGPROXY_DISABLE_SHRINK_ON_LOAD")
|
|
|
|
if err := hexEnvConfig(&conf.Keys, "IMGPROXY_KEY"); err != nil {
|
|
return err
|
|
}
|
|
if err := hexEnvConfig(&conf.Salts, "IMGPROXY_SALT"); err != nil {
|
|
return err
|
|
}
|
|
intEnvConfig(&conf.SignatureSize, "IMGPROXY_SIGNATURE_SIZE")
|
|
|
|
if err := hexFileConfig(&conf.Keys, *keyPath); err != nil {
|
|
return err
|
|
}
|
|
if err := hexFileConfig(&conf.Salts, *saltPath); err != nil {
|
|
return err
|
|
}
|
|
|
|
strEnvConfig(&conf.Secret, "IMGPROXY_SECRET")
|
|
|
|
strEnvConfig(&conf.AllowOrigin, "IMGPROXY_ALLOW_ORIGIN")
|
|
|
|
strEnvConfig(&conf.UserAgent, "IMGPROXY_USER_AGENT")
|
|
|
|
boolEnvConfig(&conf.IgnoreSslVerification, "IMGPROXY_IGNORE_SSL_VERIFICATION")
|
|
boolEnvConfig(&conf.DevelopmentErrorsMode, "IMGPROXY_DEVELOPMENT_ERRORS_MODE")
|
|
|
|
strEnvConfig(&conf.LocalFileSystemRoot, "IMGPROXY_LOCAL_FILESYSTEM_ROOT")
|
|
|
|
boolEnvConfig(&conf.S3Enabled, "IMGPROXY_USE_S3")
|
|
strEnvConfig(&conf.S3Region, "IMGPROXY_S3_REGION")
|
|
strEnvConfig(&conf.S3Endpoint, "IMGPROXY_S3_ENDPOINT")
|
|
|
|
boolEnvConfig(&conf.GCSEnabled, "IMGPROXY_USE_GCS")
|
|
strEnvConfig(&conf.GCSKey, "IMGPROXY_GCS_KEY")
|
|
|
|
boolEnvConfig(&conf.ETagEnabled, "IMGPROXY_USE_ETAG")
|
|
|
|
strEnvConfig(&conf.BaseURL, "IMGPROXY_BASE_URL")
|
|
|
|
if err := presetEnvConfig(conf.Presets, "IMGPROXY_PRESETS"); err != nil {
|
|
return err
|
|
}
|
|
if err := presetFileConfig(conf.Presets, *presetsPath); err != nil {
|
|
return err
|
|
}
|
|
boolEnvConfig(&conf.OnlyPresets, "IMGPROXY_ONLY_PRESETS")
|
|
|
|
strEnvConfig(&conf.WatermarkData, "IMGPROXY_WATERMARK_DATA")
|
|
strEnvConfig(&conf.WatermarkPath, "IMGPROXY_WATERMARK_PATH")
|
|
strEnvConfig(&conf.WatermarkURL, "IMGPROXY_WATERMARK_URL")
|
|
floatEnvConfig(&conf.WatermarkOpacity, "IMGPROXY_WATERMARK_OPACITY")
|
|
|
|
strEnvConfig(&conf.NewRelicAppName, "IMGPROXY_NEW_RELIC_APP_NAME")
|
|
strEnvConfig(&conf.NewRelicKey, "IMGPROXY_NEW_RELIC_KEY")
|
|
|
|
strEnvConfig(&conf.PrometheusBind, "IMGPROXY_PROMETHEUS_BIND")
|
|
|
|
strEnvConfig(&conf.BugsnagKey, "IMGPROXY_BUGSNAG_KEY")
|
|
strEnvConfig(&conf.BugsnagStage, "IMGPROXY_BUGSNAG_STAGE")
|
|
strEnvConfig(&conf.HoneybadgerKey, "IMGPROXY_HONEYBADGER_KEY")
|
|
strEnvConfig(&conf.HoneybadgerEnv, "IMGPROXY_HONEYBADGER_ENV")
|
|
strEnvConfig(&conf.SentryDSN, "IMGPROXY_SENTRY_DSN")
|
|
strEnvConfig(&conf.SentryEnvironment, "IMGPROXY_SENTRY_ENVIRONMENT")
|
|
strEnvConfig(&conf.SentryRelease, "IMGPROXY_SENTRY_RELEASE")
|
|
boolEnvConfig(&conf.ReportDownloadingErrors, "IMGPROXY_REPORT_DOWNLOADING_ERRORS")
|
|
|
|
intEnvConfig(&conf.FreeMemoryInterval, "IMGPROXY_FREE_MEMORY_INTERVAL")
|
|
intEnvConfig(&conf.DownloadBufferSize, "IMGPROXY_DOWNLOAD_BUFFER_SIZE")
|
|
intEnvConfig(&conf.GZipBufferSize, "IMGPROXY_GZIP_BUFFER_SIZE")
|
|
intEnvConfig(&conf.BufferPoolCalibrationThreshold, "IMGPROXY_BUFFER_POOL_CALIBRATION_THRESHOLD")
|
|
|
|
if len(conf.Keys) != len(conf.Salts) {
|
|
return fmt.Errorf("Number of keys and number of salts should be equal. Keys: %d, salts: %d", len(conf.Keys), len(conf.Salts))
|
|
}
|
|
if len(conf.Keys) == 0 {
|
|
logWarning("No keys defined, so signature checking is disabled")
|
|
conf.AllowInsecure = true
|
|
}
|
|
if len(conf.Salts) == 0 {
|
|
logWarning("No salts defined, so signature checking is disabled")
|
|
conf.AllowInsecure = true
|
|
}
|
|
|
|
if conf.SignatureSize < 1 || conf.SignatureSize > 32 {
|
|
return fmt.Errorf("Signature size should be within 1 and 32, now - %d\n", conf.SignatureSize)
|
|
}
|
|
|
|
if len(conf.Bind) == 0 {
|
|
return fmt.Errorf("Bind address is not defined")
|
|
}
|
|
|
|
if conf.ReadTimeout <= 0 {
|
|
return fmt.Errorf("Read timeout should be greater than 0, now - %d\n", conf.ReadTimeout)
|
|
}
|
|
|
|
if conf.WriteTimeout <= 0 {
|
|
return fmt.Errorf("Write timeout should be greater than 0, now - %d\n", conf.WriteTimeout)
|
|
}
|
|
if conf.KeepAliveTimeout < 0 {
|
|
return fmt.Errorf("KeepAlive timeout should be greater than or equal to 0, now - %d\n", conf.KeepAliveTimeout)
|
|
}
|
|
|
|
if conf.DownloadTimeout <= 0 {
|
|
return fmt.Errorf("Download timeout should be greater than 0, now - %d\n", conf.DownloadTimeout)
|
|
}
|
|
|
|
if conf.Concurrency <= 0 {
|
|
return fmt.Errorf("Concurrency should be greater than 0, now - %d\n", conf.Concurrency)
|
|
}
|
|
|
|
if conf.MaxClients <= 0 {
|
|
conf.MaxClients = conf.Concurrency * 10
|
|
}
|
|
|
|
if conf.TTL <= 0 {
|
|
return fmt.Errorf("TTL should be greater than 0, now - %d\n", conf.TTL)
|
|
}
|
|
|
|
if conf.MaxSrcDimension < 0 {
|
|
return fmt.Errorf("Max src dimension should be greater than or equal to 0, now - %d\n", conf.MaxSrcDimension)
|
|
} else if conf.MaxSrcDimension > 0 {
|
|
logWarning("IMGPROXY_MAX_SRC_DIMENSION is deprecated and can be removed in future versions. Use IMGPROXY_MAX_SRC_RESOLUTION")
|
|
}
|
|
|
|
if conf.MaxSrcResolution <= 0 {
|
|
return fmt.Errorf("Max src resolution should be greater than 0, now - %d\n", conf.MaxSrcResolution)
|
|
}
|
|
|
|
if conf.MaxSrcFileSize < 0 {
|
|
return fmt.Errorf("Max src file size should be greater than or equal to 0, now - %d\n", conf.MaxSrcFileSize)
|
|
}
|
|
|
|
if conf.MaxAnimationFrames <= 0 {
|
|
return fmt.Errorf("Max animation frames should be greater than 0, now - %d\n", conf.MaxAnimationFrames)
|
|
}
|
|
|
|
if conf.PngQuantizationColors < 2 {
|
|
return fmt.Errorf("Png quantization colors should be greater than 1, now - %d\n", conf.PngQuantizationColors)
|
|
} else if conf.PngQuantizationColors > 256 {
|
|
return fmt.Errorf("Png quantization colors can't be greater than 256, now - %d\n", conf.PngQuantizationColors)
|
|
}
|
|
|
|
if conf.Quality <= 0 {
|
|
return fmt.Errorf("Quality should be greater than 0, now - %d\n", conf.Quality)
|
|
} else if conf.Quality > 100 {
|
|
return fmt.Errorf("Quality can't be greater than 100, now - %d\n", conf.Quality)
|
|
}
|
|
|
|
if conf.GZipCompression < 0 {
|
|
return fmt.Errorf("GZip compression should be greater than or equal to 0, now - %d\n", conf.GZipCompression)
|
|
} else if conf.GZipCompression > 9 {
|
|
return fmt.Errorf("GZip compression can't be greater than 9, now - %d\n", conf.GZipCompression)
|
|
}
|
|
|
|
if conf.GZipCompression > 0 {
|
|
logWarning("GZip compression is deprecated and can be removed in future versions")
|
|
}
|
|
|
|
if conf.IgnoreSslVerification {
|
|
logWarning("Ignoring SSL verification is very unsafe")
|
|
}
|
|
|
|
if conf.LocalFileSystemRoot != "" {
|
|
stat, err := os.Stat(conf.LocalFileSystemRoot)
|
|
|
|
if err != nil {
|
|
return fmt.Errorf("Cannot use local directory: %s", err)
|
|
}
|
|
|
|
if !stat.IsDir() {
|
|
return fmt.Errorf("Cannot use local directory: not a directory")
|
|
}
|
|
|
|
if conf.LocalFileSystemRoot == "/" {
|
|
logWarning("Exposing root via IMGPROXY_LOCAL_FILESYSTEM_ROOT is unsafe")
|
|
}
|
|
}
|
|
|
|
if _, ok := os.LookupEnv("IMGPROXY_USE_GCS"); !ok && len(conf.GCSKey) > 0 {
|
|
logWarning("Set IMGPROXY_USE_GCS to true since it may be required by future versions to enable GCS support")
|
|
conf.GCSEnabled = true
|
|
}
|
|
|
|
if conf.WatermarkOpacity <= 0 {
|
|
return fmt.Errorf("Watermark opacity should be greater than 0")
|
|
} else if conf.WatermarkOpacity > 1 {
|
|
return fmt.Errorf("Watermark opacity should be less than or equal to 1")
|
|
}
|
|
|
|
if len(conf.PrometheusBind) > 0 && conf.PrometheusBind == conf.Bind {
|
|
return fmt.Errorf("Can't use the same binding for the main server and Prometheus")
|
|
}
|
|
|
|
if conf.FreeMemoryInterval <= 0 {
|
|
return fmt.Errorf("Free memory interval should be greater than zero")
|
|
}
|
|
|
|
if conf.DownloadBufferSize < 0 {
|
|
return fmt.Errorf("Download buffer size should be greater than or equal to 0")
|
|
} else if conf.DownloadBufferSize > math.MaxInt32 {
|
|
return fmt.Errorf("Download buffer size can't be greater than %d", math.MaxInt32)
|
|
}
|
|
|
|
if conf.GZipBufferSize < 0 {
|
|
return fmt.Errorf("GZip buffer size should be greater than or equal to 0")
|
|
} else if conf.GZipBufferSize > math.MaxInt32 {
|
|
return fmt.Errorf("GZip buffer size can't be greater than %d", math.MaxInt32)
|
|
}
|
|
|
|
if conf.BufferPoolCalibrationThreshold < 64 {
|
|
return fmt.Errorf("Buffer pool calibration threshold should be greater than or equal to 64")
|
|
}
|
|
|
|
return nil
|
|
}
|