文档章节

nginx+graphicsmagick+lua动态处理图片

基督山伯爵
 基督山伯爵
发布于 2014/06/04 13:50
字数 830
阅读 233
收藏 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
杭州
数据库管理员
私信 提问
dll文件反编译,c#、vb动态库反编译

最近开发遇到一个项目,对方提供一个c#编写的动态库,图片处理需要调用该动态库方法,发现一张图片处理起来需要5s时间,对方无法提供有效解决手段,抱着试一试的想法反编译的对方的动态库,发...

程序猿||攻城狮
11/06
0
0
Nginx与Tomcat实现请求动态数据与请求静态资源的分离

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

左羽
07/12
0
0
腾讯高性能的图片框架 LKImageKit 正式开源

LKImageKit 是一个高性能的图片框架,包括了图片控件,图片下载、内存缓存、磁盘缓存、图片解码、图片处理等一系列能力。合理的架构和线程模型,并特别针对不同场景进行优化,能充分发挥硬件...

达尔文
02/28
6.4K
8
【深入浅出MyBatis系列四】强大的动态SQL

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

陶邦仁
2015/12/22
1K
0
apiCloud中图片裁剪模块FNImageClip的使用

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

桃子红了呐
01/07
0
0

没有更多内容

加载失败,请刷新页面

加载更多

Windows / Linux / MacOS 设置代理上网的方法汇总

本文汇总了 Windows / Linux / MacOS 设置代理上网的各种方法,总结如下: 1、设置系统代理(Windows、Linux、MacOS) 2、设置代理插件(Chrome、Chromium、Firefox、Opera、QQ等浏览器) 3、...

sunboy2050
昨天
1
0
自定义 Maven 的 repositories

有时,应用中需要一些比较新的依赖,而这些依赖并没有正式发布,还是处于milestone或者是snapshot阶段,并不能从中央仓库或者镜像站上下载到。此时,就需要 自定义Maven的<repositories>。 ...

waylau
昨天
1
0
徒手写一个es6代码库

mkdir democd demonpm initnpm install -g babelnpm install -g babel-clinpm install --save-dev babel-preset-es2015-node5 在项目目录创建两个文件夹 functional-playground ......

lilugirl
昨天
3
0
linux定位应用问题的一些常用命令,特别针对内存和线程分析的dump命令

1.jps找出进程号,找到对应的进程号后面才好继续操作 2.linux查看进程详细信息 ps -ef | grep 进程ID 3. dump内存信息 Jmap -dump:format=b,file=YYMMddhhmm.dump pid 4.top查看cpu占用信息 ...

noob_chr
昨天
2
0
Android TV开发-按键焦点

写在前面 按键焦点过程了解 2.1 dispatchKeyEvent 过程了解 2.2 焦点查找请求过程了解 1.2.1 第一次获取焦点 1.2.3 按键焦点 焦点控制 焦点记忆 应用场景 参考资料 [TOC] 1. 写在前面 工...

冰雪情缘l
昨天
1
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部