文档章节

nginx+graphicsmagick+lua动态处理图片

基督山伯爵
 基督山伯爵
发布于 2014/06/04 13:50
字数 830
阅读 230
收藏 2

第一步,LUA版本目前不支持5.2,用系统自带的5.1就可以了。不过还是要安装下LUA的开发库

yum install lua-devel

第二步,安装 luajit,http://luajit.org/download.html到这里下载最新版,解压

make
make install

第三步,NGINX可以用淘宝出的tengine,已加载lua插件,编译参数如下

./configure --user=www --group=www --prefix=/usr/local/nginx --with-http_stub_status_module --with-http_ssl_module --with-http_gzip_static_module --with-http_concat_module --with-http_lua_module --with-luajit-inc=/usr/local/include/luajit-2.0 --with-luajit-lib=/usr/local/lib

nginx.conf配置如下:

        location ~ '/(.*)/(.*).(jpg|JPG|jpeg|png|gif|GIF)_([0-9]+)x([0-9]+).(jpg|jpeg|png|gif)$' {
                root /home/images;
                set $image_root /home/images;
                set $thumbnail_root /home/images/thumbnail_root;
#               set $uri $request_filename;
                if (!-e $request_filename){
                        rewrite_by_lua_file conf/image.lua;
                }
        }

image.lua文件内容如下:

-- Nginx thumbnail module by lua
-- last update: 2014/1/11
-- ver: 0.3

-- 是否记录日志
local is_log = true;
-- 允许默认图片
local enable_default_img = false;
-- graphicsmagick安装路径
local gm_path="/usr/local/GraphicsMagick/bin/gm";
-- 链接地址,如/goods/0007/541/001_328x328.jpg
local uri = ngx.var.uri;
-- 图片目录
local ngx_img_root = ngx.var.image_root
-- 缩略图目录
local ngx_thumbnail_root = ngx.var.thumbnail_root
-- img_width:缩略图宽度 img_width:缩略图高度  img_size:缩略图宽x高 img_crop_type:缩略图裁剪类型
local img_width,img_height,img_size,img_crop_type = 0;

-- 配置项,对目录、缩略图尺寸、裁剪类型进行配置,匹配后才进行缩略图处理
local cfg = {
    {dir="",sizes={"76x38!","100x100!","138x77!","232x135!","640x240^","180x105!","300x242!","320x88!","180x120!"}}
--    {dir="manage",sizes={"138x77!","232x135!","500x500$","800x800!"}}
}



-- 日志函数
-- log_level: ngx.STDERR , ngx.EMERG , ngx.ALERT , ngx.CRIT , ngx.ERR , ngx.WARN , ngx.NOTICE , ngx.INFO , ngx.DEBUG
-- 默认为ngx.NOTICE
function lua_log(msg,log_level)
    if (log_level == nil) then log_level =  ngx.NOTICE end;
    if(is_log) then ngx.log(log_level,msg) end;
end

-- 判断链接是否符合规则
function table.contains(table,element)
    -- 遍历table
    for _, value in pairs(cfg) do
        local dir = value["dir"];
        local sizes = value["sizes"];
        -- 根据正则匹配缩略图宽、高
        _,_,img_width,img_height = string.find(uri,""..dir.."_([0-9]+)x([0-9]+)");
        if(img_width ~= nil and img_height ~= nil) then
            -- 缩略图尺寸
            img_size = img_width.."x"..img_height;
            for _, value in pairs(sizes) do
                -- 1.保持原图比例,实际尺寸可能小于请求尺寸
                if (img_size == value) then
                    img_crop_type=1;
                return true;
                -- 2.拉伸,图片有可能变形
                elseif (img_size.."!" == value) then
                    img_crop_type=2;
                    return true;
                -- 3.保证大小与比例,但图有可能裁剪不完整
                elseif (img_size.."^" == value) then
                    img_crop_type=3;
                    return true;
                -- 4.只限制宽度
                elseif (img_size.."$" == value) then
                    img_crop_type=4;
                    img_size = img_width.."x";
                    return true;
                end
            end
        end
    end
    return false
end

-- 原图链接
local img_original_uri = string.gsub(uri, "_[0-9]+x[0-9]+.[jpg|png|gif]+","");
-- 判断原图是否存在
local img_exist=io.open(ngx_img_root .. img_original_uri);
if not img_exist then
    if not enable_default_img then
        lua_log(img_original_uri.." isn't exist!",ngx.ERR);
        ngx.exit(404);
    else
        local default_img_original_uri = "/empty/empty.jpg";
        img_exist=io.open(ngx_img_root ..  default_img_original_uri);
        if img_exist then
            lua_log(img_original_uri .. "isn't exist! crop image with default image");
            img_original_uri = default_img_original_uri;
        else
            lua_log(img_original_uri.." isn't exist!",ngx.ERR);
            ngx.exit(404);
        end
    end;
end;

if not table.contains(cfg, uri) then
    lua_log(uri.." don't match!",ngx.ERR);
    ngx.exit(404);
else
    -- 开始生成缩略图
    local gm_command;
    -- 缩略图文件路径
    local img_thumbnail_path = ngx_thumbnail_root .. uri;
    -- 原图文件路径
    local img_original_path = ngx_img_root .. img_original_uri;
    -- 执行gm命令
    if (img_crop_type == 1) then
        gm_command = gm_path .. " convert " .. img_original_path  .. " -thumbnail "  .. img_size .. " -background white -gravity center -strip +profile '*' -quality 90 -extent " .. img_size .. " " .. img_thumbnail_path
    elseif (img_crop_type == 2) then
        gm_command = gm_path .. " convert " .. img_original_path  .. " -thumbnail "  .. img_size .. "! -strip +profile '*' -quality 90 -extent ".. img_size .." " .. img_thumbnail_path;
    elseif (img_crop_type == 3) then
        gm_command = gm_path .. " convert " .. img_original_path  .. " -thumbnail "  .. img_size .. "^ -strip +profile '*' -quality 90 -extent ".. img_size .." " ..img_thumbnail_path;
    elseif (img_crop_type == 4) then
        gm_command = gm_path .. " convert " .. img_original_path  .. " -resize '"  .. img_size .. ">' -strip +profile '*' -quality 90 " ..img_thumbnail_path;
    else
        lua_log("img_crop_type error:"..img_crop_type,ngx.ERR);
        ngx.exit(404);
    end

-- 判断图是否存在
    local new_img_exist=io.open(img_thumbnail_path);
    --lua_log(img_thumbnail_path .. uri);
    -- 执行gm命令 and not new_img_exist
    if (gm_command ~= nil and not new_img_exist) then
        -- 获取缩略图路径及文件名
        _,_,img_thumbnail_dir,img__thumbnail_filename=string.find(img_thumbnail_path,'(.-)([^/]*)$')
        -- 先创建缩略图所在目录,避免报错
        os.execute("mkdir -p "..img_thumbnail_dir);
        -- 执行gm命令
        os.execute(gm_command);
        lua_log("gm_command======"..img_crop_type..gm_command);
    end
    -- 转发请求至缩略图
    ngx.req.set_uri("/thumbnail_root"..uri,true);
end


© 著作权归作者所有

共有 人打赏支持
基督山伯爵
粉丝 7
博文 17
码字总数 6779
作品 0
杭州
数据库管理员
Nginx与Tomcat实现请求动态数据与请求静态资源的分离

  上篇博客说明了Nginx在应用架构中的作用,以及负载均衡的思路。这篇实践一下其中的访问静态资源与访问动态资源的操作。 一、认识访问静态资源与访问动态资源的区别   静态资源:指存储...

左羽
07/12
0
0
【深入浅出MyBatis系列四】强大的动态SQL

深入浅出MyBatis系列 【深入浅出MyBatis系列一】MyBatis入门 【深入浅出MyBatis系列二】配置简介(MyBatis源码篇) 【深入浅出MyBatis系列三】Mapper映射文件配置 【深入浅出MyBatis系列四】...

陶邦仁
2015/12/22
1K
0
Nginx + Apache + Tomcat架构方式,为什么需要Apache?Apache的作用?

Nginx:一款能承受高并发的HTTP服务器,异步的,多个连接(万级别)可以对应一个进程,进行响应。基于事件驱动模型。 Nginx优点:负载均衡、反向代理、处理静态文件优势。 Apache:相对于Nginx...

Nathans
2014/11/06
8.6K
11
AlexGao/WSHttpHelper

轻量级Http请求框架。清晰明了的请求接口描述,灵活的扩展机制。 快捷接口: 支持同步调用; 支持异步调用,异步回调; 支持JSON自动转换为指定类型; 支持自定义输入参数验证; 支持自定义结果解...

AlexGao
2015/12/20
0
0
apiCloud中图片裁剪模块FNImageClip的使用

思路 1.获取需裁剪图片的地址 2.跳转到裁剪页面 3.裁剪成功返回新图片地址 4.替换原有图片地址 增加修饰和事件 上面是动态生成的图片html布局数据,增加一个id标识,增加一个点击事件 打开裁剪...

桃子红了呐
01/07
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Kali Linux Docker 練習

docker pull kalilinux/kali-linux-docker docker run -t -i kalilinux/kali-linux-docker /bin/bash apt-get update apt-get install htop apt-get install nmap apt-get install wpscan ap......

BaiyuanLab
今天
1
0
通俗大白话来理解TCP协议的三次握手和四次分手

最近在恶补计算机网络方面的知识,之前对于TCP的三次握手和四次分手也是模模糊糊,对于其中的细节更是浑然不知,最近看了很多这方面的知识,也在系统的学习计算机网络,加深自己的CS功底,就...

onedotdot
今天
2
0
TiDB 在爱奇艺的应用及实践

爱奇艺,中国高品质视频娱乐服务提供者,2010 年 4 月 22 日正式上线,推崇品质、青春、时尚的品牌内涵如今已深入人心,网罗了全球广大的年轻用户群体,积极推动产品、技术、内容、营销等全方...

TiDB
今天
1
0
Web系统大规模并发:电商秒杀与抢购

一、大规模并发带来的挑战 在过去的工作中,我曾经面对过5w每秒的高并发秒杀功能,在这个过程中,整个Web系统遇到了很多的问题和挑战。如果Web系统不做针对性的优化,会轻而易举地陷入到异常...

xtof
今天
3
0
代码质量管理平台-sonarqube

在工作中,往往开发的时候会不怎么注重代码质量的人很多,存在着很多的漏洞和隐患等问题,sonarqube可以进行代码质量的审核,而且十分的残酷。。。。。接下来我们说下怎么安装 进入官网下载:...

落叶清风
今天
9
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部