2017-09-19 04:15:14 +01:00
|
|
|
// Package captcha provides a simple API for captcha generation
|
2017-09-16 11:04:28 +01:00
|
|
|
package captcha
|
|
|
|
|
|
|
|
import (
|
2017-09-19 03:10:02 +01:00
|
|
|
"github.com/golang/freetype"
|
|
|
|
"github.com/golang/freetype/truetype"
|
|
|
|
"golang.org/x/image/font"
|
2017-09-16 11:04:28 +01:00
|
|
|
"image"
|
2017-09-17 07:50:28 +01:00
|
|
|
"image/color"
|
2017-09-19 03:10:02 +01:00
|
|
|
"image/draw"
|
2017-09-16 11:04:28 +01:00
|
|
|
"image/png"
|
2017-09-17 07:50:28 +01:00
|
|
|
"io"
|
2017-09-19 03:10:02 +01:00
|
|
|
"math"
|
2017-09-16 11:04:28 +01:00
|
|
|
"math/rand"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
const charPreset = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
|
|
|
|
|
2017-09-19 03:10:02 +01:00
|
|
|
var rng = rand.New(rand.NewSource(time.Now().UnixNano()))
|
|
|
|
var ttfFont *truetype.Font
|
|
|
|
|
2017-09-19 04:15:14 +01:00
|
|
|
// Options manage captcha generation details.
|
2017-09-16 11:04:28 +01:00
|
|
|
type Options struct {
|
2017-09-19 04:15:14 +01:00
|
|
|
// BackgroundColor is captcha image's background color.
|
|
|
|
// It defaults to color.Transparent.
|
2017-09-19 03:10:02 +01:00
|
|
|
BackgroundColor color.Color
|
2017-09-19 04:15:14 +01:00
|
|
|
// CharPreset decides what text will be on captcha image.
|
|
|
|
// It defaults to digit 0-9 and all English alphabet.
|
|
|
|
// ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789
|
|
|
|
CharPreset string
|
|
|
|
// TextLength is the length of captcha text.
|
|
|
|
// It defaults to 4.
|
|
|
|
TextLength int
|
|
|
|
// CurveNumber is the number of curves to draw on captcha image.
|
2017-09-19 04:52:25 +01:00
|
|
|
// It defaults to 2.
|
2017-09-19 04:15:14 +01:00
|
|
|
CurveNumber int
|
|
|
|
|
2017-09-19 06:12:03 +01:00
|
|
|
width int
|
|
|
|
height int
|
2017-09-16 11:04:28 +01:00
|
|
|
}
|
|
|
|
|
2017-09-17 07:50:28 +01:00
|
|
|
func newDefaultOption(width, height int) *Options {
|
2017-09-16 11:04:28 +01:00
|
|
|
return &Options{
|
2017-09-19 03:10:02 +01:00
|
|
|
BackgroundColor: color.Transparent,
|
|
|
|
CharPreset: charPreset,
|
2017-09-19 04:15:14 +01:00
|
|
|
TextLength: 4,
|
2017-09-19 04:52:25 +01:00
|
|
|
CurveNumber: 2,
|
2017-09-19 03:10:02 +01:00
|
|
|
width: width,
|
|
|
|
height: height,
|
2017-09-16 11:04:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-19 04:15:14 +01:00
|
|
|
// SetOption is a function that can be used to modify default options.
|
2017-09-19 03:10:02 +01:00
|
|
|
type SetOption func(*Options)
|
2017-09-16 11:04:28 +01:00
|
|
|
|
2017-09-19 04:15:14 +01:00
|
|
|
// Data is the result of captcha generation.
|
|
|
|
// It has a `Text` field and a private `img` field that will
|
|
|
|
// be used in `WriteTo` receiver
|
2017-09-16 11:04:28 +01:00
|
|
|
type Data struct {
|
2017-09-19 04:15:14 +01:00
|
|
|
// Text is captcha solution
|
2017-09-16 11:04:28 +01:00
|
|
|
Text string
|
|
|
|
|
|
|
|
img *image.NRGBA
|
|
|
|
}
|
|
|
|
|
2017-09-19 04:15:14 +01:00
|
|
|
// WriteTo encodes image data and writes to an io.Writer.
|
|
|
|
// It returns possible error from PNG encoding
|
2017-09-16 11:04:28 +01:00
|
|
|
func (data *Data) WriteTo(w io.Writer) error {
|
|
|
|
return png.Encode(w, data.img)
|
|
|
|
}
|
|
|
|
|
2017-09-19 03:10:02 +01:00
|
|
|
func init() {
|
|
|
|
var err error
|
2017-09-19 07:10:49 +01:00
|
|
|
ttfFont, err = freetype.ParseFont(ttf)
|
2017-09-19 03:10:02 +01:00
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-19 04:15:14 +01:00
|
|
|
// New creates a new captcha.
|
|
|
|
// It returns captcha data and any freetype drawing error encountered
|
2017-09-19 03:10:02 +01:00
|
|
|
func New(width int, height int, option ...SetOption) (*Data, error) {
|
2017-09-17 07:50:28 +01:00
|
|
|
options := newDefaultOption(width, height)
|
2017-09-16 11:04:28 +01:00
|
|
|
for _, setOption := range option {
|
|
|
|
setOption(options)
|
|
|
|
}
|
|
|
|
|
|
|
|
text := randomText(options)
|
|
|
|
img := image.NewNRGBA(image.Rect(0, 0, width, height))
|
2017-09-19 03:10:02 +01:00
|
|
|
draw.Draw(img, img.Bounds(), &image.Uniform{options.BackgroundColor}, image.ZP, draw.Src)
|
2017-09-17 07:50:28 +01:00
|
|
|
drawNoise(img, options)
|
2017-09-19 04:15:14 +01:00
|
|
|
drawCurves(img, options)
|
2017-09-19 03:10:02 +01:00
|
|
|
err := drawText(text, img, options)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-09-16 11:04:28 +01:00
|
|
|
|
2017-09-19 03:10:02 +01:00
|
|
|
return &Data{Text: text, img: img}, nil
|
2017-09-16 11:04:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func randomText(opts *Options) (text string) {
|
|
|
|
n := len(opts.CharPreset)
|
2017-09-19 04:15:14 +01:00
|
|
|
for i := 0; i < opts.TextLength; i++ {
|
2017-09-17 07:50:28 +01:00
|
|
|
text += string(opts.CharPreset[rng.Intn(n)])
|
2017-09-16 11:04:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return text
|
|
|
|
}
|
2017-09-17 07:50:28 +01:00
|
|
|
|
|
|
|
func drawNoise(img *image.NRGBA, opts *Options) {
|
2017-09-19 04:52:25 +01:00
|
|
|
noiseCount := (opts.width * opts.height) / 28
|
2017-09-17 07:50:28 +01:00
|
|
|
for i := 0; i < noiseCount; i++ {
|
|
|
|
x := rng.Intn(opts.width)
|
|
|
|
y := rng.Intn(opts.height)
|
|
|
|
img.Set(x, y, randomColor())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func randomColor() color.RGBA {
|
|
|
|
red := rng.Intn(255)
|
|
|
|
green := rng.Intn(255)
|
|
|
|
blue := rng.Intn(255)
|
|
|
|
|
|
|
|
return color.RGBA{R: uint8(red), G: uint8(green), B: uint8(blue), A: uint8(255)}
|
|
|
|
}
|
2017-09-19 03:10:02 +01:00
|
|
|
|
2017-09-19 04:15:14 +01:00
|
|
|
func drawCurves(img *image.NRGBA, opts *Options) {
|
|
|
|
for i := 0; i < opts.CurveNumber; i++ {
|
2017-09-19 03:10:02 +01:00
|
|
|
drawSineCurve(img, opts)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ideally we want to draw bezier curves
|
|
|
|
// For now sine curves will do the job
|
|
|
|
func drawSineCurve(img *image.NRGBA, opts *Options) {
|
|
|
|
var xStart, xEnd int
|
|
|
|
if opts.width <= 40 {
|
|
|
|
xStart, xEnd = 1, opts.width-1
|
|
|
|
} else {
|
|
|
|
xStart = rng.Intn(opts.width/10) + 1
|
|
|
|
xEnd = opts.width - rng.Intn(opts.width/10) - 1
|
|
|
|
}
|
|
|
|
curveHeight := float64(rng.Intn(opts.height/6) + opts.height/6)
|
|
|
|
yStart := rng.Intn(opts.height*2/3) + opts.height/6
|
|
|
|
angle := 1.0 + rng.Float64()
|
|
|
|
flip := rng.Intn(2) == 0
|
|
|
|
yFlip := 1.0
|
|
|
|
if flip {
|
|
|
|
yFlip = -1.0
|
|
|
|
}
|
2017-09-19 06:12:03 +01:00
|
|
|
curveColor := randomDarkColor()
|
2017-09-19 03:10:02 +01:00
|
|
|
|
|
|
|
for x1 := xStart; x1 <= xEnd; x1++ {
|
|
|
|
y := math.Sin(math.Pi*angle*float64(x1)/float64(opts.width)) * curveHeight * yFlip
|
|
|
|
img.Set(x1, int(y)+yStart, curveColor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-19 06:12:03 +01:00
|
|
|
func randomDarkColor() hsva {
|
|
|
|
hue := float64(rng.Intn(361)) / 360
|
2017-09-19 07:10:49 +01:00
|
|
|
saturation := 0.6 + rng.Float64()*0.2
|
|
|
|
value := 0.25 + rng.Float64()*0.2
|
2017-09-19 03:10:02 +01:00
|
|
|
|
2017-09-19 07:10:49 +01:00
|
|
|
return hsva{h: hue, s: saturation, v: value, a: uint8(255)}
|
2017-09-19 03:10:02 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func drawText(text string, img *image.NRGBA, opts *Options) error {
|
|
|
|
ctx := freetype.NewContext()
|
2017-09-19 04:52:25 +01:00
|
|
|
ctx.SetDPI(92.0)
|
2017-09-19 03:10:02 +01:00
|
|
|
ctx.SetClip(img.Bounds())
|
|
|
|
ctx.SetDst(img)
|
|
|
|
ctx.SetHinting(font.HintingFull)
|
|
|
|
ctx.SetFont(ttfFont)
|
|
|
|
|
|
|
|
fontSpacing := opts.width / len(text)
|
|
|
|
|
|
|
|
for idx, char := range text {
|
2017-09-19 06:12:03 +01:00
|
|
|
fontScale := 1 + rng.Float64()*0.5
|
2017-09-19 03:10:02 +01:00
|
|
|
fontSize := float64(opts.height) / fontScale
|
|
|
|
ctx.SetFontSize(fontSize)
|
2017-09-19 06:12:03 +01:00
|
|
|
ctx.SetSrc(image.NewUniform(randomDarkColor()))
|
2017-09-19 03:10:02 +01:00
|
|
|
x := fontSpacing*idx + fontSpacing/int(fontSize)
|
|
|
|
y := opts.height/6 + rng.Intn(opts.height/3) + int(fontSize/2)
|
|
|
|
pt := freetype.Pt(x, y)
|
|
|
|
if _, err := ctx.DrawString(string(char), pt); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|