config.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
  1. package config
  2. import (
  3. "flag"
  4. "fmt"
  5. "math"
  6. "os"
  7. "regexp"
  8. "runtime"
  9. log "github.com/sirupsen/logrus"
  10. "github.com/imgproxy/imgproxy/v3/config/configurators"
  11. "github.com/imgproxy/imgproxy/v3/imagetype"
  12. "github.com/imgproxy/imgproxy/v3/version"
  13. )
  14. var (
  15. Network string
  16. Bind string
  17. ReadTimeout int
  18. WriteTimeout int
  19. KeepAliveTimeout int
  20. DownloadTimeout int
  21. Concurrency int
  22. MaxClients int
  23. TTL int
  24. CacheControlPassthrough bool
  25. SetCanonicalHeader bool
  26. SoReuseport bool
  27. PathPrefix string
  28. MaxSrcResolution int
  29. MaxSrcFileSize int
  30. MaxAnimationFrames int
  31. MaxSvgCheckBytes int
  32. MaxRedirects int
  33. JpegProgressive bool
  34. PngInterlaced bool
  35. PngQuantize bool
  36. PngQuantizationColors int
  37. AvifSpeed int
  38. Quality int
  39. FormatQuality map[imagetype.Type]int
  40. StripMetadata bool
  41. StripColorProfile bool
  42. AutoRotate bool
  43. EnableWebpDetection bool
  44. EnforceWebp bool
  45. EnableAvifDetection bool
  46. EnforceAvif bool
  47. EnableClientHints bool
  48. SkipProcessingFormats []imagetype.Type
  49. UseLinearColorspace bool
  50. DisableShrinkOnLoad bool
  51. Keys [][]byte
  52. Salts [][]byte
  53. SignatureSize int
  54. Secret string
  55. AllowOrigin string
  56. UserAgent string
  57. IgnoreSslVerification bool
  58. DevelopmentErrorsMode bool
  59. AllowedSources []*regexp.Regexp
  60. CookiePassthrough bool
  61. CookieBaseURL string
  62. LocalFileSystemRoot string
  63. S3Enabled bool
  64. S3Region string
  65. S3Endpoint string
  66. GCSEnabled bool
  67. GCSKey string
  68. ABSEnabled bool
  69. ABSName string
  70. ABSKey string
  71. ABSEndpoint string
  72. ETagEnabled bool
  73. ETagBuster string
  74. BaseURL string
  75. Presets []string
  76. OnlyPresets bool
  77. WatermarkData string
  78. WatermarkPath string
  79. WatermarkURL string
  80. WatermarkOpacity float64
  81. FallbackImageData string
  82. FallbackImagePath string
  83. FallbackImageURL string
  84. FallbackImageHTTPCode int
  85. DataDogEnable bool
  86. NewRelicAppName string
  87. NewRelicKey string
  88. PrometheusBind string
  89. PrometheusNamespace string
  90. BugsnagKey string
  91. BugsnagStage string
  92. HoneybadgerKey string
  93. HoneybadgerEnv string
  94. SentryDSN string
  95. SentryEnvironment string
  96. SentryRelease string
  97. AirbrakeProjecID int
  98. AirbrakeProjecKey string
  99. AirbrakeEnv string
  100. ReportDownloadingErrors bool
  101. EnableDebugHeaders bool
  102. FreeMemoryInterval int
  103. DownloadBufferSize int
  104. BufferPoolCalibrationThreshold int
  105. HealthCheckPath string
  106. )
  107. var (
  108. keyPath string
  109. saltPath string
  110. presetsPath string
  111. )
  112. func init() {
  113. Reset()
  114. flag.StringVar(&keyPath, "keypath", "", "path of the file with hex-encoded key")
  115. flag.StringVar(&saltPath, "saltpath", "", "path of the file with hex-encoded salt")
  116. flag.StringVar(&presetsPath, "presets", "", "path of the file with presets")
  117. }
  118. func Reset() {
  119. Network = "tcp"
  120. Bind = ":8080"
  121. ReadTimeout = 10
  122. WriteTimeout = 10
  123. KeepAliveTimeout = 10
  124. DownloadTimeout = 5
  125. Concurrency = runtime.NumCPU() * 2
  126. MaxClients = 0
  127. TTL = 3600
  128. CacheControlPassthrough = false
  129. SetCanonicalHeader = false
  130. SoReuseport = false
  131. PathPrefix = ""
  132. MaxSrcResolution = 16800000
  133. MaxSrcFileSize = 0
  134. MaxAnimationFrames = 1
  135. MaxSvgCheckBytes = 32 * 1024
  136. MaxRedirects = 10
  137. JpegProgressive = false
  138. PngInterlaced = false
  139. PngQuantize = false
  140. PngQuantizationColors = 256
  141. AvifSpeed = 5
  142. Quality = 80
  143. FormatQuality = map[imagetype.Type]int{imagetype.AVIF: 50}
  144. StripMetadata = true
  145. StripColorProfile = true
  146. AutoRotate = true
  147. EnableWebpDetection = false
  148. EnforceWebp = false
  149. EnableAvifDetection = false
  150. EnforceAvif = false
  151. EnableClientHints = false
  152. SkipProcessingFormats = make([]imagetype.Type, 0)
  153. UseLinearColorspace = false
  154. DisableShrinkOnLoad = false
  155. Keys = make([][]byte, 0)
  156. Salts = make([][]byte, 0)
  157. SignatureSize = 32
  158. Secret = ""
  159. AllowOrigin = ""
  160. UserAgent = fmt.Sprintf("imgproxy/%s", version.Version())
  161. IgnoreSslVerification = false
  162. DevelopmentErrorsMode = false
  163. AllowedSources = make([]*regexp.Regexp, 0)
  164. CookiePassthrough = false
  165. CookieBaseURL = ""
  166. LocalFileSystemRoot = ""
  167. S3Enabled = false
  168. S3Region = ""
  169. S3Endpoint = ""
  170. GCSEnabled = false
  171. GCSKey = ""
  172. ABSEnabled = false
  173. ABSName = ""
  174. ABSKey = ""
  175. ABSEndpoint = ""
  176. ETagEnabled = false
  177. ETagBuster = ""
  178. BaseURL = ""
  179. Presets = make([]string, 0)
  180. OnlyPresets = false
  181. WatermarkData = ""
  182. WatermarkPath = ""
  183. WatermarkURL = ""
  184. WatermarkOpacity = 1
  185. FallbackImageData = ""
  186. FallbackImagePath = ""
  187. FallbackImageURL = ""
  188. FallbackImageHTTPCode = 200
  189. DataDogEnable = false
  190. NewRelicAppName = ""
  191. NewRelicKey = ""
  192. PrometheusBind = ""
  193. PrometheusNamespace = ""
  194. BugsnagKey = ""
  195. BugsnagStage = "production"
  196. HoneybadgerKey = ""
  197. HoneybadgerEnv = "production"
  198. SentryDSN = ""
  199. SentryEnvironment = "production"
  200. SentryRelease = fmt.Sprintf("imgproxy@%s", version.Version())
  201. AirbrakeProjecID = 0
  202. AirbrakeProjecKey = ""
  203. AirbrakeEnv = "production"
  204. ReportDownloadingErrors = true
  205. EnableDebugHeaders = false
  206. FreeMemoryInterval = 10
  207. DownloadBufferSize = 0
  208. BufferPoolCalibrationThreshold = 1024
  209. HealthCheckPath = ""
  210. }
  211. func Configure() error {
  212. if port := os.Getenv("PORT"); len(port) > 0 {
  213. Bind = fmt.Sprintf(":%s", port)
  214. }
  215. configurators.String(&Network, "IMGPROXY_NETWORK")
  216. configurators.String(&Bind, "IMGPROXY_BIND")
  217. configurators.Int(&ReadTimeout, "IMGPROXY_READ_TIMEOUT")
  218. configurators.Int(&WriteTimeout, "IMGPROXY_WRITE_TIMEOUT")
  219. configurators.Int(&KeepAliveTimeout, "IMGPROXY_KEEP_ALIVE_TIMEOUT")
  220. configurators.Int(&DownloadTimeout, "IMGPROXY_DOWNLOAD_TIMEOUT")
  221. configurators.Int(&Concurrency, "IMGPROXY_CONCURRENCY")
  222. configurators.Int(&MaxClients, "IMGPROXY_MAX_CLIENTS")
  223. configurators.Int(&TTL, "IMGPROXY_TTL")
  224. configurators.Bool(&CacheControlPassthrough, "IMGPROXY_CACHE_CONTROL_PASSTHROUGH")
  225. configurators.Bool(&SetCanonicalHeader, "IMGPROXY_SET_CANONICAL_HEADER")
  226. configurators.Bool(&SoReuseport, "IMGPROXY_SO_REUSEPORT")
  227. configurators.String(&PathPrefix, "IMGPROXY_PATH_PREFIX")
  228. configurators.MegaInt(&MaxSrcResolution, "IMGPROXY_MAX_SRC_RESOLUTION")
  229. configurators.Int(&MaxSrcFileSize, "IMGPROXY_MAX_SRC_FILE_SIZE")
  230. configurators.Int(&MaxSvgCheckBytes, "IMGPROXY_MAX_SVG_CHECK_BYTES")
  231. configurators.Int(&MaxAnimationFrames, "IMGPROXY_MAX_ANIMATION_FRAMES")
  232. configurators.Int(&MaxRedirects, "IMGPROXY_MAX_REDIRECTS")
  233. configurators.Patterns(&AllowedSources, "IMGPROXY_ALLOWED_SOURCES")
  234. configurators.Bool(&JpegProgressive, "IMGPROXY_JPEG_PROGRESSIVE")
  235. configurators.Bool(&PngInterlaced, "IMGPROXY_PNG_INTERLACED")
  236. configurators.Bool(&PngQuantize, "IMGPROXY_PNG_QUANTIZE")
  237. configurators.Int(&PngQuantizationColors, "IMGPROXY_PNG_QUANTIZATION_COLORS")
  238. configurators.Int(&AvifSpeed, "IMGPROXY_AVIF_SPEED")
  239. configurators.Int(&Quality, "IMGPROXY_QUALITY")
  240. if err := configurators.ImageTypesQuality(FormatQuality, "IMGPROXY_FORMAT_QUALITY"); err != nil {
  241. return err
  242. }
  243. configurators.Bool(&StripMetadata, "IMGPROXY_STRIP_METADATA")
  244. configurators.Bool(&StripColorProfile, "IMGPROXY_STRIP_COLOR_PROFILE")
  245. configurators.Bool(&AutoRotate, "IMGPROXY_AUTO_ROTATE")
  246. configurators.Bool(&EnableWebpDetection, "IMGPROXY_ENABLE_WEBP_DETECTION")
  247. configurators.Bool(&EnforceWebp, "IMGPROXY_ENFORCE_WEBP")
  248. configurators.Bool(&EnableAvifDetection, "IMGPROXY_ENABLE_AVIF_DETECTION")
  249. configurators.Bool(&EnforceAvif, "IMGPROXY_ENFORCE_AVIF")
  250. configurators.Bool(&EnableClientHints, "IMGPROXY_ENABLE_CLIENT_HINTS")
  251. configurators.String(&HealthCheckPath, "IMGPROXY_HEALTH_CHECK_PATH")
  252. if err := configurators.ImageTypes(&SkipProcessingFormats, "IMGPROXY_SKIP_PROCESSING_FORMATS"); err != nil {
  253. return err
  254. }
  255. configurators.Bool(&UseLinearColorspace, "IMGPROXY_USE_LINEAR_COLORSPACE")
  256. configurators.Bool(&DisableShrinkOnLoad, "IMGPROXY_DISABLE_SHRINK_ON_LOAD")
  257. if err := configurators.Hex(&Keys, "IMGPROXY_KEY"); err != nil {
  258. return err
  259. }
  260. if err := configurators.Hex(&Salts, "IMGPROXY_SALT"); err != nil {
  261. return err
  262. }
  263. configurators.Int(&SignatureSize, "IMGPROXY_SIGNATURE_SIZE")
  264. if err := configurators.HexFile(&Keys, keyPath); err != nil {
  265. return err
  266. }
  267. if err := configurators.HexFile(&Salts, saltPath); err != nil {
  268. return err
  269. }
  270. configurators.String(&Secret, "IMGPROXY_SECRET")
  271. configurators.String(&AllowOrigin, "IMGPROXY_ALLOW_ORIGIN")
  272. configurators.String(&UserAgent, "IMGPROXY_USER_AGENT")
  273. configurators.Bool(&IgnoreSslVerification, "IMGPROXY_IGNORE_SSL_VERIFICATION")
  274. configurators.Bool(&DevelopmentErrorsMode, "IMGPROXY_DEVELOPMENT_ERRORS_MODE")
  275. configurators.Bool(&CookiePassthrough, "IMGPROXY_COOKIE_PASSTHROUGH")
  276. configurators.String(&CookieBaseURL, "IMGPROXY_COOKIE_BASE_URL")
  277. configurators.String(&LocalFileSystemRoot, "IMGPROXY_LOCAL_FILESYSTEM_ROOT")
  278. configurators.Bool(&S3Enabled, "IMGPROXY_USE_S3")
  279. configurators.String(&S3Region, "IMGPROXY_S3_REGION")
  280. configurators.String(&S3Endpoint, "IMGPROXY_S3_ENDPOINT")
  281. configurators.Bool(&GCSEnabled, "IMGPROXY_USE_GCS")
  282. configurators.String(&GCSKey, "IMGPROXY_GCS_KEY")
  283. configurators.Bool(&ABSEnabled, "IMGPROXY_USE_ABS")
  284. configurators.String(&ABSName, "IMGPROXY_ABS_NAME")
  285. configurators.String(&ABSKey, "IMGPROXY_ABS_KEY")
  286. configurators.String(&ABSEndpoint, "IMGPROXY_ABS_ENDPOINT")
  287. configurators.Bool(&ETagEnabled, "IMGPROXY_USE_ETAG")
  288. configurators.String(&ETagBuster, "IMGPROXY_ETAG_BUSTER")
  289. configurators.String(&BaseURL, "IMGPROXY_BASE_URL")
  290. configurators.StringSlice(&Presets, "IMGPROXY_PRESETS")
  291. if err := configurators.StringSliceFile(&Presets, presetsPath); err != nil {
  292. return err
  293. }
  294. configurators.Bool(&OnlyPresets, "IMGPROXY_ONLY_PRESETS")
  295. configurators.String(&WatermarkData, "IMGPROXY_WATERMARK_DATA")
  296. configurators.String(&WatermarkPath, "IMGPROXY_WATERMARK_PATH")
  297. configurators.String(&WatermarkURL, "IMGPROXY_WATERMARK_URL")
  298. configurators.Float(&WatermarkOpacity, "IMGPROXY_WATERMARK_OPACITY")
  299. configurators.String(&FallbackImageData, "IMGPROXY_FALLBACK_IMAGE_DATA")
  300. configurators.String(&FallbackImagePath, "IMGPROXY_FALLBACK_IMAGE_PATH")
  301. configurators.String(&FallbackImageURL, "IMGPROXY_FALLBACK_IMAGE_URL")
  302. configurators.Int(&FallbackImageHTTPCode, "IMGPROXY_FALLBACK_IMAGE_HTTP_CODE")
  303. configurators.Bool(&DataDogEnable, "IMGPROXY_DATADOG_ENABLE")
  304. configurators.String(&NewRelicAppName, "IMGPROXY_NEW_RELIC_APP_NAME")
  305. configurators.String(&NewRelicKey, "IMGPROXY_NEW_RELIC_KEY")
  306. configurators.String(&PrometheusBind, "IMGPROXY_PROMETHEUS_BIND")
  307. configurators.String(&PrometheusNamespace, "IMGPROXY_PROMETHEUS_NAMESPACE")
  308. configurators.String(&BugsnagKey, "IMGPROXY_BUGSNAG_KEY")
  309. configurators.String(&BugsnagStage, "IMGPROXY_BUGSNAG_STAGE")
  310. configurators.String(&HoneybadgerKey, "IMGPROXY_HONEYBADGER_KEY")
  311. configurators.String(&HoneybadgerEnv, "IMGPROXY_HONEYBADGER_ENV")
  312. configurators.String(&SentryDSN, "IMGPROXY_SENTRY_DSN")
  313. configurators.String(&SentryEnvironment, "IMGPROXY_SENTRY_ENVIRONMENT")
  314. configurators.String(&SentryRelease, "IMGPROXY_SENTRY_RELEASE")
  315. configurators.Int(&AirbrakeProjecID, "IMGPROXY_AIRBRAKE_PROJECT_ID")
  316. configurators.String(&AirbrakeProjecKey, "IMGPROXY_AIRBRAKE_PROJECT_KEY")
  317. configurators.String(&AirbrakeEnv, "IMGPROXY_AIRBRAKE_ENVIRONMENT")
  318. configurators.Bool(&ReportDownloadingErrors, "IMGPROXY_REPORT_DOWNLOADING_ERRORS")
  319. configurators.Bool(&EnableDebugHeaders, "IMGPROXY_ENABLE_DEBUG_HEADERS")
  320. configurators.Int(&FreeMemoryInterval, "IMGPROXY_FREE_MEMORY_INTERVAL")
  321. configurators.Int(&DownloadBufferSize, "IMGPROXY_DOWNLOAD_BUFFER_SIZE")
  322. configurators.Int(&BufferPoolCalibrationThreshold, "IMGPROXY_BUFFER_POOL_CALIBRATION_THRESHOLD")
  323. if len(Keys) != len(Salts) {
  324. return fmt.Errorf("Number of keys and number of salts should be equal. Keys: %d, salts: %d", len(Keys), len(Salts))
  325. }
  326. if len(Keys) == 0 {
  327. log.Warning("No keys defined, so signature checking is disabled")
  328. }
  329. if len(Salts) == 0 {
  330. log.Warning("No salts defined, so signature checking is disabled")
  331. }
  332. if SignatureSize < 1 || SignatureSize > 32 {
  333. return fmt.Errorf("Signature size should be within 1 and 32, now - %d\n", SignatureSize)
  334. }
  335. if len(Bind) == 0 {
  336. return fmt.Errorf("Bind address is not defined")
  337. }
  338. if ReadTimeout <= 0 {
  339. return fmt.Errorf("Read timeout should be greater than 0, now - %d\n", ReadTimeout)
  340. }
  341. if WriteTimeout <= 0 {
  342. return fmt.Errorf("Write timeout should be greater than 0, now - %d\n", WriteTimeout)
  343. }
  344. if KeepAliveTimeout < 0 {
  345. return fmt.Errorf("KeepAlive timeout should be greater than or equal to 0, now - %d\n", KeepAliveTimeout)
  346. }
  347. if DownloadTimeout <= 0 {
  348. return fmt.Errorf("Download timeout should be greater than 0, now - %d\n", DownloadTimeout)
  349. }
  350. if Concurrency <= 0 {
  351. return fmt.Errorf("Concurrency should be greater than 0, now - %d\n", Concurrency)
  352. }
  353. if MaxClients <= 0 {
  354. MaxClients = Concurrency * 10
  355. }
  356. if TTL <= 0 {
  357. return fmt.Errorf("TTL should be greater than 0, now - %d\n", TTL)
  358. }
  359. if MaxSrcResolution <= 0 {
  360. return fmt.Errorf("Max src resolution should be greater than 0, now - %d\n", MaxSrcResolution)
  361. }
  362. if MaxSrcFileSize < 0 {
  363. return fmt.Errorf("Max src file size should be greater than or equal to 0, now - %d\n", MaxSrcFileSize)
  364. }
  365. if MaxAnimationFrames <= 0 {
  366. return fmt.Errorf("Max animation frames should be greater than 0, now - %d\n", MaxAnimationFrames)
  367. }
  368. if PngQuantizationColors < 2 {
  369. return fmt.Errorf("Png quantization colors should be greater than 1, now - %d\n", PngQuantizationColors)
  370. } else if PngQuantizationColors > 256 {
  371. return fmt.Errorf("Png quantization colors can't be greater than 256, now - %d\n", PngQuantizationColors)
  372. }
  373. if AvifSpeed < 0 {
  374. return fmt.Errorf("Avif speed should be greater than 0, now - %d\n", AvifSpeed)
  375. } else if AvifSpeed > 8 {
  376. return fmt.Errorf("Avif speed can't be greater than 8, now - %d\n", AvifSpeed)
  377. }
  378. if Quality <= 0 {
  379. return fmt.Errorf("Quality should be greater than 0, now - %d\n", Quality)
  380. } else if Quality > 100 {
  381. return fmt.Errorf("Quality can't be greater than 100, now - %d\n", Quality)
  382. }
  383. if IgnoreSslVerification {
  384. log.Warning("Ignoring SSL verification is very unsafe")
  385. }
  386. if LocalFileSystemRoot != "" {
  387. stat, err := os.Stat(LocalFileSystemRoot)
  388. if err != nil {
  389. return fmt.Errorf("Cannot use local directory: %s", err)
  390. }
  391. if !stat.IsDir() {
  392. return fmt.Errorf("Cannot use local directory: not a directory")
  393. }
  394. if LocalFileSystemRoot == "/" {
  395. log.Warning("Exposing root via IMGPROXY_LOCAL_FILESYSTEM_ROOT is unsafe")
  396. }
  397. }
  398. if _, ok := os.LookupEnv("IMGPROXY_USE_GCS"); !ok && len(GCSKey) > 0 {
  399. log.Warning("Set IMGPROXY_USE_GCS to true since it may be required by future versions to enable GCS support")
  400. GCSEnabled = true
  401. }
  402. if WatermarkOpacity <= 0 {
  403. return fmt.Errorf("Watermark opacity should be greater than 0")
  404. } else if WatermarkOpacity > 1 {
  405. return fmt.Errorf("Watermark opacity should be less than or equal to 1")
  406. }
  407. if FallbackImageHTTPCode < 100 || FallbackImageHTTPCode > 599 {
  408. return fmt.Errorf("Fallback image HTTP code should be between 100 and 599")
  409. }
  410. if len(PrometheusBind) > 0 && PrometheusBind == Bind {
  411. return fmt.Errorf("Can't use the same binding for the main server and Prometheus")
  412. }
  413. if FreeMemoryInterval <= 0 {
  414. return fmt.Errorf("Free memory interval should be greater than zero")
  415. }
  416. if DownloadBufferSize < 0 {
  417. return fmt.Errorf("Download buffer size should be greater than or equal to 0")
  418. } else if DownloadBufferSize > math.MaxInt32 {
  419. return fmt.Errorf("Download buffer size can't be greater than %d", math.MaxInt32)
  420. }
  421. if BufferPoolCalibrationThreshold < 64 {
  422. return fmt.Errorf("Buffer pool calibration threshold should be greater than or equal to 64")
  423. }
  424. return nil
  425. }