使用Lua GD库动态生成验证码图片(2)

在前一篇文章中初步的实现了验证码的随机生成,觉得很容易被破解,因此加了一些干扰;

鹤冲天建议使用表达式作为验证码,即使破解程序识别了验证码,也要计算结果才能通过验证,在一定程度上增加了破解难度。


因此我对代码进行了重新整理,通过控制运行时配置选项来达到不同效果。

现在主要做到了:

(1)字符内容随机(普通字符串或表达式)

(2)每个字符的字体随机

(3)每个字符大小随机

(4)每个字符倾斜角度随机

(5)干扰线条随机

除了(4)其他都可以通过配置来控制。


下面是代码,有详细的注释,就不多说了:

  1  require ( " gd " )
  2  require ( " lfs " )
  3 
  4  -- ---------------------------------------------------------------------------------------------------------------------------------
  5  -- 运行配置项
  6  -- ---------------------------------------------------------------------------------------------------------------------------------
  7  -- 字体:-1-使用gd.FONT_GIANT字体;1-使用随机字体;其他-使用“fonts”中第一个字体
  8  -- 字体预先在变量“fonts”中定义;如果fonts没有值,将搜索系统中的所有字体,字体路径在“FONT_PATH”中预定义
  9  FONT = 1
 10 
 11  -- 每个字符字体随机:1-是,其他-否
 12  -- 仅当“FONT”值为“1”时,本变量起作用
 13  FONT_RANDOM_CHAR = 0
 14 
 15  -- 每个字符字体大小是否随机:1-是,其他-否
 16  -- 仅当FONT^=-1时起作用
 17  FONT_SIZE_RANDOM = 1
 18 
 19  -- 是否增加线条干扰:1-是;其他-否
 20  XLINE_FALG = 1
 21 
 22  -- 干扰线条的最多条数
 23  -- 仅当“XLINE_FALG”的值为“是”是,本变量起作用
 24  XLINE_LIMIT = 6
 25 
 26  -- 验证码类型:TEXT-字符串;EXPRESSION-表达式
 27  MARK_TYPE = " TEXT "
 28  -- MARK_TYPE="EXPRESSION"
 29 
 30  -- 字符个数:仅当“MARK_TYPE”=“TEXT”时,本变量起作用
 31  TEXT_NUM = 6
 32 
 33  -- 表达式项数限制(最多不超过EXPRESSION_ITEMS项):仅当“MARK_TYPE”=“EXPRESSION”时,本变量起作用
 34  EXPRESSION_ITEMS = 3
 35 
 36  -- 生成验证码个数
 37  MARK_NUM = 100000
 38 
 39 
 40  -- ---------------------------------------------------------------------------------------------------------------------------------
 41  -- 预定义变量
 42  -- ---------------------------------------------------------------------------------------------------------------------------------
 43  -- 词典
 44  dict = { ' a ' , ' b ' , ' c ' , ' d ' , ' e ' , ' f ' , ' g ' , ' h ' , ' i ' , ' j ' , ' k ' , ' l ' , ' m ' , ' n ' , ' o ' , ' p ' , ' q ' , ' r ' , ' s ' , ' t ' , ' u ' , ' v ' , ' w ' , ' x ' , ' y ' , ' z ' , ' A ' , ' B ' , ' C ' , ' D ' , ' E ' , ' F ' , ' G ' , ' H ' , ' I ' , ' J ' , ' K ' , ' L ' , ' M ' , ' N ' , ' O ' , ' P ' , ' Q ' , ' R ' , ' S ' , ' T ' , ' U ' , ' V ' , ' W ' , ' X ' , ' Y ' , ' Z ' , ' 1 ' , ' 2 ' , ' 3 ' , ' 4 ' , ' 5 ' , ' 6 ' , ' 7 ' , ' 8 ' , ' 9 ' , ' 0 ' }
 45  numbers = { " 1 " , " 2 " , " 3 " , " 4 " , " 5 " , " 6 " , " 7 " , " 8 " , " 9 " } -- 表达式可使用数字,排除0
 46  operators = { " + " , " - " , " * " } -- 表达式可使用运算符,不支持“/”
 47 
 48  -- 随机种子,防止通过获取系统时间得到随机数种子,直接计算出验证码
 49  math.randomseed ( os.time ())
 50 
 51  -- 大小
 52  IMG_WIDTH = 100
 53  IMG_HEIGHT = 40
 54 
 55  -- 颜色
 56  im2  =  gd.createTrueColor(IMG_WIDTH,IMG_HEIGHT)
 57  fg  =  im2:colorAllocate( 129 , 32 , 28 ) -- 前景色
 58  bg  =  im2:colorAllocate( 216 , 235 , 238 ) -- 背景色
 59 
 60  -- 字体
 61  FONT_PATH = " C:/WINDOWS/Fonts/ " -- 系统字体路径
 62  fonts = {}
 63  -- fonts={"courbd.ttf","courbi.ttf","DejaVuMonoSans.ttf","DejaVuMonoSansBold.ttf","DejaVuMonoSansBoldOblique.ttf","DejaVuMonoSansOblique.ttf","lucon.ttf","monosbi.ttf","nina.ttf","simhei.ttf","simkai.ttf","swissci.ttf","tahomabd.ttf","timesbd.ttf","timesbi.ttf","timesi.ttf","trebuc.ttf","trebucit.ttf"}
 64  font_size = { 14 , 15 , 16 , 17 , 18 , 19 , 20 } -- 随机字体大小
 65 
 66  -- 生成的随机码
 67  stringmark = ""
 68 
 69  -- ---------------------------------------------------------------------------------------------------------------------------------
 70  -- 功能函数
 71  -- ---------------------------------------------------------------------------------------------------------------------------------
 72  -- 初始化:创建图片、设置背景
 73  function  init()
 74      im2  =  gd.createTrueColor(IMG_WIDTH, IMG_HEIGHT)
 75      im2:filledRectangle( 0 , 0 ,IMG_WIDTH,IMG_HEIGHT,bg)
 76      stringmark = ""
 77  end
 78 
 79  -- 查找字体
 80  function  searchFont()
 81       if  table.getn(fonts) == 0   then   -- 没有指定字体,就搜索系统字体
 82           local  i = 1
 83           for  file  in  lfs.dir(FONT_PATH)  do
 84               if   string.find (file, " .ttf " ) and   not   string.find (file, " esri " )   then   -- 排除特定字体
 85                  fonts[i] = file
 86                  i = i + 1
 87               end
 88           end
 89       end
 90  end
 91 
 92  -- 生成text字符串
 93  function  makeText()
 94       local  num = table.getn(dict)
 95       for  i = 1 ,TEXT_NUM  do
 96          stringmark = stringmark..dict[ math.random (num)]
 97       end
 98  end
 99 
100  -- 生成表达式字符串
101  function  makeExpression()
102       local  n = math.random ( 2 , 3 ) -- 表达式项数
103       local  strings = {}
104       for  i = 1 ,n * 2 - 1   do -- 表达式项数+运算符项数
105           local  str = ""
106           if  i % 2 == 1   then -- 数字
107               local  n2 = math.random ( 1 , 2 )
108               for  j = 1 ,n2  do -- 每个数字最多2位
109                  str = str..numbers[ math.random ( 9 )]
110               end
111           else -- 运算符
112              str = operators[ math.random ( 3 )]
113           end
114          strings[i] = str
115       end  
116       return  strings
117  end
118 
119  -- 主函数
120  function  doIt()
121      searchFont()
122       local  numfonts = table.getn(fonts)
123       if  numfonts < 1   then
124           print ( " 没有找到字体! " )
125           return
126       end
127 
128       for  i = 1 ,MARK_NUM  do
129          init()
130           local  font = fonts[ 0 ];
131           local  fontsize = 20 ;
132          
133           if  MARK_TYPE == " TEXT "   then -- 普通字符串验证码
134              makeText()
135  --             print(stringmark)    
136               if  FONT ==- 1   then
137                  im2:string(gd.FONT_GIANT, 18 , 10 ,stringmark, fg)
138               else
139                   for  nIndex = 1 , string.len (stringmark)  do
140                       if  FONT == 1   then  font = fonts[ math.random (numfonts)]  end
141                       if  FONT_SIZE_RANDOM == 1   then  fontsize = font_size[ math.random ( 7 )]  end
142                      im2:stringFT(fg,FONT_PATH..font,fontsize, math.random () / math.pi , 5 + (nIndex - 1 ) * 15 25 string.sub (stringmark,nIndex,nIndex))
143                   end         
144               end
145           elseif  MARK_TYPE == " EXPRESSION "   then -- 表达式验证码
146               local  strings = makeExpression()
147               local  raise = 0
148               local  ncharacter = 0  
149               for  j = 1 ,table.getn(strings)  do
150                   if  j % 2 == 0   then  raise = 3   end
151                  stringmark = stringmark..strings[j]
152                   if  FONT == 1   then  font = fonts[ math.random (numfonts)]  end
153                   if  FONT_SIZE_RANDOM == 1   then  fontsize = font_size[ math.random ( 5 )]  end           
154                  im2:stringFT(fg,FONT_PATH..font,fontsize, math.random () / math.pi , 5 + ncharacter * 12 + raise,  25 , strings[j])
155                  ncharacter = ncharacter + string.len (strings[j])
156               end
157  --             print(stringmark) 
158  --             value=tonumber(stringmark)
159  --             print(value)
160           end
161          
162           --   随机线条干扰
163           if  XLINE_FALG == 1   then
164               local  xlineNum = math.random (XLINE_LIMIT)
165               for  i = 1 ,xlineNum  do
166                  im2:line( math.random (IMG_WIDTH), math.random (IMG_HEIGHT), math.random (IMG_WIDTH), math.random (IMG_HEIGHT),fg)
167               end
168           end
169 
170          im2:png( " ./output/ " ..font.. " .png " , 100 )                
171       end
172  end
173 
174 -- start = os.clock ()
175  doIt()
176 -- print ( os.clock () - start)


效果大致如下:

你可能感兴趣的:(lua)