使用 OpenResty 构建高效的动态图片水印代理服务20250127

使用 OpenResty 构建高效的动态图片水印代理服务

在当今数字化的时代,图片在各种业务场景中广泛应用。为了保护版权、统一品牌形象,动态图片水印功能显得尤为重要。然而,直接在后端服务中集成水印功能,往往会带来代码复杂度增加、兼容性问题等诸多挑战。为了解决这些问题,我们可以利用 OpenResty 和 Lua 构建一套独立于后端应用的动态水印代理服务,既能大幅降低后端负担,又能增强系统的灵活性。
在这里插入图片描述

一、需求与解决方案

1.1 需求分析

在实际业务中,我们面临着多方面的需求,主要可以分为功能性需求、性能需求和扩展性需求:

功能性需求
  • 按用户和图片 ID 获取图片并添加水印:根据用户提供的用户和图片 ID,获取对应的图片,并为其动态添加水印。
  • 从指定 URL 加载图片并添加水印:支持用户输入指定的图片 URL,系统从该 URL 动态加载图片,并为其添加水印。
性能需求
  • 高效处理图片:在高并发的情况下,能够快速、高效地处理图片,确保系统的响应速度。
  • 避免重复处理:对于相同的图片,避免进行重复的处理,提高系统的性能和资源利用率。
扩展性需求
  • 兼容多种图片格式:支持常见的图片格式,如 JPG、PNG、WEBP 等,以满足不同用户的需求。
  • 方便添加其他处理功能:系统应具有良好的扩展性,能够方便地添加其他图片处理功能,如裁剪、缩放等。

1.2 解决方案

为了实现上述需求,我们采用以下方案:

  • 使用 OpenResty 提供图片代理服务:OpenResty 是一个基于 Nginx 与 Lua 的高性能 Web 平台,通过 Lua 模块可以方便地实现图片水印功能。
  • 解耦图片处理逻辑:将图片处理逻辑从后端服务中分离出来,通过代理实现独立的图片处理流程,避免对后端代码的修改。
  • 利用 lua - resty - magick 与 Nginx 缓存机制:lua - resty - magick 是一个用于处理图片的 Lua 模块,能够高效地添加水印。同时,使用 Nginx 的缓存机制可以优化系统性能,减少重复处理。

二、实现步骤

2.1 安装与配置 OpenResty

以下是在 CentOS 系统上安装 OpenResty 和相关依赖的详细步骤:

Step 1: 更新系统并安装基础工具
sudo yum update -y
sudo yum groupinstall -y "Development Tools"
sudo yum install -y readline-devel pcre-devel openssl-devel tar gcc make tree perl curl

这些命令的作用是更新系统软件包,并安装编译和运行 OpenResty 所需的基础工具。

Step 2: 添加 OpenResty 官方 YUM 仓库并安装 OpenResty
wget https://openresty.org/package/centos/openresty.repo
sudo mv openresty.repo /etc/yum.repos.d/
sudo yum check-update
sudo yum install -y openresty

通过这些命令,我们添加了 OpenResty 的官方 YUM 仓库,并安装了 OpenResty。

Step 3: 安装 ImageMagick 和 LuaRocks
sudo yum install -y epel-release
sudo yum install -y ImageMagick ImageMagick-devel
sudo yum install -y luarocks
luarocks install magick

这里安装了 ImageMagick 及其开发库,用于图片处理。同时,安装了 LuaRocks 并使用它安装了 magick 模块。

验证安装
  • 确认 OpenResty 正常运行
sudo /usr/local/openresty/nginx/sbin/nginx
curl -I http://localhost

启动 OpenResty 的 Nginx 服务,并使用 curl 命令检查服务是否正常响应。

  • 验证 magick 模块
luajit -e "local magick = require('magick'); print('Magick module loaded successfully')"

运行此命令,如果输出 Magick module loaded successfully,则说明 magick 模块安装成功。

2.2 图片代理服务实现

Lua 图片处理模块

以下是核心的图片处理逻辑,封装在 image_handler.lua 文件中:

local _M = {}
local magick = require("magick")
local tmp_dir = "/tmp/"function _M.process_image_with_watermark(image_data, ext)local input_path = tmp_dir .. "input_image." .. extlocal output_path = tmp_dir .. "output_image." .. extlocal watermark_path = "/var/www/images/watermark.png"-- 保存图片到临时路径local input_file = io.open(input_path, "wb")if not input_file thenngx.log(ngx.ERR, "Failed to open file for writing: ", input_path)ngx.status = ngx.HTTP_INTERNAL_SERVER_ERRORngx.say("Failed to process image")returnendinput_file:write(image_data)input_file:close()-- 添加水印逻辑local success, err = pcall(function()local img = magick.load_image(input_path)local watermark = magick.load_image(watermark_path)watermark:resize(img:get_width(), img:get_height())img:composite(watermark, 0, 0, "OverCompositeOp")img:write(output_path)img:destroy()watermark:destroy()end)if not success thenngx.log(ngx.ERR, "Image processing failed: ", err)ngx.status = ngx.HTTP_INTERNAL_SERVER_ERRORngx.say("Failed to process image")returnend-- 返回处理后的图片local output_file = io.open(output_path, "rb")if not output_file thenngx.log(ngx.ERR, "Failed to open processed image")ngx.status = ngx.HTTP_INTERNAL_SERVER_ERRORngx.say("Failed to process image")returnendlocal output_data = output_file:read("*all")output_file:close()ngx.header.content_type = "image/" .. extngx.print(output_data)-- 清理临时文件os.remove(input_path)os.remove(output_path)
endreturn _M

该模块定义了一个函数 process_image_with_watermark,用于处理图片并添加水印。它将图片数据保存到临时文件,添加水印后再读取处理后的图片并返回给客户端,最后清理临时文件。

Nginx 配置文件

以下是支持代理服务和 URL 动态图片处理的 Nginx 配置:

worker_processes auto;
error_log logs/error.log debug;events {worker_connections 10240;
}http {include mime.types;default_type application/octet-stream;lua_package_path "/usr/local/openresty/lualib/?.lua;/usr/share/lua/5.1/?.lua;;";lua_package_cpath "/usr/lib64/lua/5.1/?.so;/usr/lib/lua/5.1/?.so;/usr/local/openresty/lualib/?.so;;";log_format main '$remote_addr - $remote_user [$time_local] "$request" ''$status $body_bytes_sent "$http_referer" ''"$http_user_agent" "$http_x_forwarded_for"';access_log /usr/local/openresty/nginx/logs/access.log main;resolver 8.8.8.8 8.8.4.4;  # 使用 Google Public DNSserver {listen 80;server_name example.com;location /users/ {content_by_lua_block {local http = require("resty.http")local image_handler = require("image_handler")local backend_url = "http://backend_service" .. ngx.var.request_urilocal httpc = http.new()local res, err = httpc:request_uri(backend_url, { method = "GET" })if not res or res.status ~= 200 thenngx.status = ngx.HTTP_BAD_GATEWAYngx.say("Failed to fetch image")returnendimage_handler.process_image_with_watermark(res.body, "jpg")}}location /process_url_image {content_by_lua_block {local http = require("resty.http")local image_handler = require("image_handler")local res, err = http.new():request_uri("http://example.com/sample_image.jpg", { method = "GET" })if not res or res.status ~= 200 thenngx.status = ngx.HTTP_BAD_GATEWAYngx.say("Failed to fetch image")returnendimage_handler.process_image_with_watermark(res.body, "jpg")}}}
}

在这个配置文件中,我们定义了两个 location 块。/users/ 块用于处理根据用户和图片 ID 获取图片的请求,从后端服务获取图片后调用 image_handler 模块添加水印。/process_url_image 块用于处理从指定 URL 加载图片的请求,同样调用 image_handler 模块添加水印。

三、总结与优化方向

3.1 实现亮点

  • 解耦架构:通过代理实现图片处理与应用的分离,无需修改后端代码,降低了系统的耦合度,提高了可维护性。
  • 动态水印处理:支持多种图片格式,能够实时为图片添加水印,满足了业务的功能性需求。
  • 高性能设计:结合 Nginx 的缓存优化,能够高效处理高并发请求,提升了系统的性能。

3.2 优化方向

  • 扩展功能:进一步支持更多的图片格式和水印样式,如不同的水印位置、透明度等,以满足更复杂的业务需求。
  • 自动化部署:提供一键安装脚本,简化系统的配置和部署流程,降低运维成本。
  • 可视化管理:开发前端页面,允许用户动态配置水印参数,如水印文本、字体、颜色等,提高用户体验。

通过 OpenResty 的灵活性和高性能,我们可以快速实现动态图片处理功能,并显著提升系统的可扩展性和维护性。期待您在实际应用中进行更多的探索与实践!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/69593.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【MySQL — 数据库增删改查操作】深入解析MySQL的 Update 和 Delete 操作

1. 测试数据 mysql> select* from exam1; ----------------------------------------- | id | name | Chinese | Math | English | ----------------------------------------- | 1 | 唐三藏 | 67.0 | 98.0 | 56.0 | | 2 | 孙悟空 | 87.0 | 78.…

webAPI -DOM 相关知识点总结(非常细)

title: WebAPI语法 date: 2025-01-28 12:00:00 tags:- 前端 categories:- 前端WEB API 了解DOM的结构并掌握其基本的操作,体验 DOM 在开发中的作用 API简介 就是使用js来操作html和浏览器 什么是DOM? 就是一个文档对象模型,是用来呈现预计于任意htm…

图论——最小生成树的扩展应用

最小生成树相关原理 acwing1146.新的开始 假设存在一个“超级发电站” 在每一个矿井修发电站相当于从这个“超级发电站”到各个矿井连一条长度为 v [ i ] v[i] v[i]的边。 这样一来这就是一个最短路的模板题。 #include <iostream> #include <cstring> using na…

K8S中高级存储之PV和PVC

高级存储 PV和PVC 由于kubernetes支持的存储系统有很多&#xff0c;要求客户全都掌握&#xff0c;显然不现实。为了能够屏蔽底层存储实现的细节&#xff0c;方便用户使用&#xff0c; kubernetes引入PV和PVC两种资源对象。 PV&#xff08;Persistent Volume&#xff09; PV是…

宝塔面板SSL加密访问设置教程

参考:https://www.bt.cn/bbs/thread-117246-1-1.html 如何快速使用证书加密访问面板 因早期默认未开启https访问所以没有相关的风险提醒&#xff0c;现面板默认已开启https加密访问、提升安全性 由于采用的是服务器内部本身签发证书&#xff0c;不被公网浏览器信任请参考以下步…

深入探讨数据库索引类型:B-tree、Hash、GIN与GiST的对比与应用

title: 深入探讨数据库索引类型:B-tree、Hash、GIN与GiST的对比与应用 date: 2025/1/26 updated: 2025/1/26 author: cmdragon excerpt: 在现代数据库管理系统中,索引技术是提高查询性能的重要手段。当数据量不断增长时,如何快速、有效地访问这些数据成为了数据库设计的核…

21.2-工程中添加FreeRTOS(掌握) 用STM32CubeMX添加FreeRTOS

这个是全网最详细的STM32项目教学视频。 第一篇在这里: 视频在这里 STM32智能小车V3-STM32入门教程-openmv与STM32循迹小车-stm32f103c8t6-电赛 嵌入式学习 PID控制算法 编码器电机 跟随 **V3:HAL库开发、手把手教学下面功能&#xff1a;PID速度控制、PID循迹、PID跟随、遥控、…

【越学学糊涂的Linux系统】Linux指令篇(二)

一、pwd指令&#xff1a; 00x0:打印该用户当前目录下所属的文件路径 看指令框可以看出我用的是一个叫sw的用户&#xff0c;我们的路径就是在一个home目录下的sw目录下的class113文件路径。 也可以说是指出当前所处的工作目录 补充&#xff1a;&#x1f386;​​​​​​​Wi…

LangGraph系列-1:用LangGraph构建简单聊天机器人

在快速发展的人工智能和大型语言模型&#xff08;llm&#xff09;世界中&#xff0c;开发人员不断寻求创建更灵活、更强大、更直观的人工智能代理的方法。 虽然LangChain已经改变了这个领域的游戏规则&#xff0c;允许创建复杂的链和代理&#xff0c;但对代理运行时的更复杂控制…

进程池的制作(linux进程间通信,匿名管道... ...)

目录 一、进程间通信的理解 1.为什么进程间要通信 2.如何进行通信 二、匿名管道 1.管道的理解 2.匿名管道的使用 3.管道的五种特性 4.管道的四种通信情况 5.管道缓冲区容量 三、进程池 1.进程池的理解 2.进程池的制作 四、源码 1.ProcessPool.hpp 2.Task.hpp 3…

Linux学习笔记——用户管理

一、用户管理命令 useradd #用户增加命令 usermod #用户修改命令 passwd #密码修改命令 userdel #用户删除命令 su #用户提权命令 1、useradd命令&#xff08;加用户&#xff09;&#xff1a; 创建并设置用户信息&#xff0c;使用us…

怎样在PPT中启用演讲者视图功能?

怎样在PPT中启用演讲者视图功能&#xff1f; 如果你曾经参加过重要的会议或者演讲&#xff0c;你就会知道&#xff0c;演讲者视图&#xff08;Presenter View&#xff09;对PPT展示至关重要。它不仅能帮助演讲者更好地掌控演讲节奏&#xff0c;还能提供额外的提示和支持&#…

【Unity3D】实现2D小地图效果

目录 一、玩家脚本Player 二、Canvas组件设置 三、小地图相关 四、GameLogicMap脚本修改 基于&#xff1a;【Unity3D】Tilemap俯视角像素游戏案例-CSDN博客 2D玩家添加Dotween移动DOPath效果&#xff0c;移动完成后进行刷新小地图&#xff08;小地图会顺便刷新大地图&…

四.3 Redis 五大数据类型/结构的详细说明/详细使用( hash 哈希表数据类型详解和使用)

四.3 Redis 五大数据类型/结构的详细说明/详细使用&#xff08; hash 哈希表数据类型详解和使用&#xff09; 文章目录 四.3 Redis 五大数据类型/结构的详细说明/详细使用&#xff08; hash 哈希表数据类型详解和使用&#xff09;2.hash 哈希表常用指令(详细讲解说明)2.1 hset …

C#通过3E帧SLMP/MC协议读写三菱FX5U/Q系列PLC数据案例

C#通过3E帧SLMP/MC协议读写三菱FX5U/Q系列PLC数据案例&#xff0c;仅做数据读写报文测试。附带自己整理的SLMP/MC通讯协议表。 SLMP以太网读写PLC数据20191206/.vs/WindowsFormsApp7/v15/.suo , 73216 SLMP以太网读写PLC数据20191206/SLMP与MC协议3E帧通讯协议表.xlsx , 10382…

【算法】经典博弈论问题——威佐夫博弈 python

目录 威佐夫博弈(Wythoff Game)【模板】 威佐夫博弈(Wythoff Game) 有两堆石子&#xff0c;数量任意&#xff0c;可以不同&#xff0c;游戏开始由两个人轮流取石子 游戏规定&#xff0c;每次有两种不同的取法 1)在任意的一堆中取走任意多的石子 2)可以在两堆中同时取走相同数量…

具身智能研究报告

参考&#xff1a; &#xff08;1&#xff09;GTC大会&Figure&#xff1a;“具身智能”奇点已至 &#xff08;2&#xff09;2024中国具身智能创投报告 &#xff08;3&#xff09;2024年具身智能产业发展研究报告 &#xff08;4&#xff09;具身智能行业深度&#xff1a;发展…

把本地搭建的hexo博客部署到自己的服务器上

配置远程服务器的git 安装git 安装依赖工具包 yum install -y curl-devel expat-devel gettext-devel openssl-devel zlib-devel安装编译工具 yum install -y gcc perl-ExtUtils-MakeMaker package下载git&#xff0c;也可以去官网下载了传到服务器上 wget https://www.ke…

STM32 旋转编码器

旋转编码器简介 旋转编码器&#xff1a;用来测量位置、速度或旋转方向的装置&#xff0c;当其旋转轴旋转时&#xff0c;其输出端可以输出与旋转速度和方向对应的方波信号&#xff0c;读取方波信号的频率和相位信息即可得知旋转轴的速度和方向 类型&#xff1a;机械触点式/霍尔传…

后盾人JS--闭包明明白白

延伸函数环境生命周期 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>Document</title> <…