Golang

关注公众号 jb51net

关闭
首页 > 脚本专栏 > Golang > go ip访问限制

golang实现ip访问限制及提交次数

作者:Ai编码助手

在 Web 应用中,通常会需要对 IP 访问进行限制以及控制提交次数,本文将使用中间件或者基于 Redis 这样的缓存服务来实现,感兴趣的可以了解下

在 Web 应用中,通常会需要对 IP 访问进行限制以及控制提交次数,以防止恶意攻击(例如暴力破解、DoS攻击、API滥用等)。为了实现这一功能,我们可以结合 Golang 的特性,使用中间件或者基于 Redis 这样的缓存服务来实现 IP 限制和提交次数的控制。

实现步骤

核心概念

使用 Golang 及 Redis 实现 IP 访问限制和提交次数限制

这里我们使用 Redis 来存储和控制访问次数,并结合 Go 实现一个简单的 IP 访问限制中间件。

依赖库

你可以使用 Redis 官方的 Go 客户端 go-redis 来连接 Redis 进行操作。先安装这个库:

go get github.com/go-redis/redis/v8

实现代码

下面的代码演示了如何使用 Redis 来实现 IP 访问限制和提交次数限制。

package main

import (
	"context"
	"fmt"
	"log"
	"net/http"
	"strconv"
	"time"

	"github.com/go-redis/redis/v8"
)

// Redis client
var rdb *redis.Client

// 初始化 Redis 客户端
func initRedis() {
	rdb = redis.NewClient(&redis.Options{
		Addr:     "localhost:6379", // Redis 地址
		Password: "",               // Redis 密码(如果有)
		DB:       0,                // 使用的 Redis 数据库
	})
}

// 获取客户端的 IP 地址
func getIP(r *http.Request) string {
	// 尝试从 X-Forwarded-For 或 X-Real-IP 获取真实 IP
	ip := r.Header.Get("X-Forwarded-For")
	if ip == "" {
		ip = r.Header.Get("X-Real-IP")
	}
	if ip == "" {
		ip = r.RemoteAddr
	}
	return ip
}

// 中间件:IP 访问限制
func rateLimitMiddleware(next http.HandlerFunc) http.HandlerFunc {
	return func(w http.ResponseWriter, r *http.Request) {
		ctx := context.Background()
		ip := getIP(r)
		key := "rate_limit:" + ip

		// 获取 Redis 中的访问次数
		count, err := rdb.Get(ctx, key).Result()
		if err == redis.Nil {
			// 如果没有记录,设置计数为1,并设置过期时间
			err := rdb.Set(ctx, key, 1, time.Minute).Err() // 1 分钟限制
			if err != nil {
				http.Error(w, "Redis error", http.StatusInternalServerError)
				return
			}
		} else if err != nil {
			http.Error(w, "Redis error", http.StatusInternalServerError)
			return
		} else {
			// 将访问次数转换为整数
			countInt, _ := strconv.Atoi(count)
			if countInt >= 10 { // 假设限制为每分钟最多10次
				http.Error(w, "Too Many Requests", http.StatusTooManyRequests)
				return
			}

			// 递增计数
			rdb.Incr(ctx, key)
		}

		next.ServeHTTP(w, r)
	}
}

// 示例处理器:提交处理
func submitHandler(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "Request successful")
}

func main() {
	// 初始化 Redis
	initRedis()

	// 创建 HTTP 服务器并添加中间件
	http.HandleFunc("/submit", rateLimitMiddleware(submitHandler))

	log.Println("Server is running on port 8080...")
	http.ListenAndServe(":8080", nil)
}

代码解析

Redis 客户端初始化

IP 获取

通过 getIP() 函数获取请求的客户端 IP 地址。该函数尝试从请求头中的 X-Forwarded-ForX-Real-IP 获取真实的 IP。如果没有,则使用 RemoteAddr

Rate Limiting 中间件

处理请求

改进与扩展

通过 Golang 和 Redis 的结合,可以轻松实现 IP 访问限制和提交次数控制。Redis 的高性能特性使其非常适合用作限流计数器的存储。在实际应用中,可以根据需要扩展该方案,例如使用不同的限流算法、结合 IP 黑名单等。

到此这篇关于golang实现ip访问限制及提交次数的文章就介绍到这了,更多相关go ip访问限制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文