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

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

实现步骤

  • IP 访问限制:对每个 IP 的访问频次进行限制,比如每个 IP 每分钟只能访问某个接口 10 次。超过限制后,返回错误信息(例如 429 Too Many Requests)。
  • 提交次数限制:通过限制某个时间段内某个 IP 的提交次数,防止暴力破解或者滥用接口。
  • Redis(或其他存储系统)作为计数器:为了更好地实现这种限制,可以使用 Redis 等缓存系统来存储 IP 的访问记录、提交次数等,因为 Redis 的性能和易用性使它成为理想的选择。

核心概念

  • Rate Limiting(限流):根据 IP 限制某个时间段内的访问次数。
  • 请求次数计数:对每个 IP 进行计数,并基于计数来判断是否超过限制。
  • 时间窗口:设置一定的时间窗口(例如一分钟或五分钟),在这个时间段内统计 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 客户端初始化

  • 使用 redis.NewClient() 初始化 Redis 客户端。
  • 通过 rdb.Set()rdb.Get() 来操作 Redis 中的计数器。

IP 获取

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

Rate Limiting 中间件

  • rateLimitMiddleware() 是核心的中间件函数,负责限制每个 IP 的访问次数。它使用 Redis 来存储每个 IP 的访问计数和限流时间窗口(这里设置为 1 分钟)。
  • 当 IP 的访问次数超过限制时,返回 HTTP 状态码 429 Too Many Requests

处理请求

  • submitHandler() 是一个简单的示例处理器,处理成功的请求。
  • 访问 /submit 时,经过中间件限制后,正常情况下返回 "Request successful"。

改进与扩展

  • 动态调整限流策略: 可以根据不同的用户类型、不同的 API 路径动态调整限流策略。例如,VIP 用户可能会有更高的访问频次。
  • IP 黑名单: 通过 Redis 或其他存储系统维护一个黑名单,遇到黑名单中的 IP 可以直接拒绝请求。
  • 按时间窗口的限流算法: 你可以采用滑动窗口、漏桶算法、令牌桶算法等更复杂的限流算法来实现更灵活的控制。
  • 使用 Redis Expire 特性: 在 Redis 中使用 SetEX(带过期时间的键设置)或 TTL 来确保计数器可以自动重置,避免手动管理。
  • 日志记录与报警: 可以结合日志系统,在某个 IP 频繁触发限制时记录日志或发送报警信息。

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

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

原创文章,作者:DKNGX,如若转载,请注明出处:http://www.wangzhanshi.com/n/5420.html

(0)
DKNGX的头像DKNGX
上一篇 2024年12月17日 19:27:33
下一篇 2024年12月17日 19:27:35

相关推荐

  • 使用Go语言实现LRU缓存的代码详解

    引言 在日常开发中,缓存是提高系统性能的重要手段。LRU(Least Recently Used)缓存是一种基于“最近最少使用”策略的缓存系统,其目的是在空…

    2024年12月17日
  • GO语言中ni,零值与空结构体的使用

    go语言的初学者,特别是java开发者新学习go语言,对于一些和java类似但是又有差异的概念很容易混淆,比如说go中的零值,nil 和 空结构体。本文就来详细探讨一下go中这些特…

    Golang 2024年12月17日
  • Golang使用gorm实现分页功能的示例代码

    1、背景 在提供列表接口时一般要用到分页,对于存储在某些数据库中的数据进行分页起来非常的方便,下文给出一个通过gorm进行分页并通过http返回数据的例子。 2、go库下载 gor…

    Golang 2024年12月17日
  • Go语言结合validator包实现表单验证

    在现代 Web 开发中,表单验证和错误处理是至关重要的环节,尤其是在多语言环境下。 本文将通过一个实际的示例,演示如何使用 Go 语言的 Gin 框架结合 validator 包,…

    Golang 2024年12月17日
  • golang高并发之本地缓存详解

    一、使用场景 试想一个场景,有一个配置服务系统,里面存储着各种各样的配置,比如直播间的直播信息、点赞、签到、红包、带货等等。这些配置信息有两个特点: 并发量可能会特别特别大,试想一…

    2024年12月17日
  • 如何在Go语言中高效使用Redis的Pipeline

    在构建高性能应用时,Redis 经常成为开发者的首选工具。作为一个内存数据库,Redis 可以处理大量的数据操作,但如果每个命令都单独发送,网络延迟会成为瓶颈,影响性能。 这时,R…

    Golang 2024年12月17日
  • Go集成swagger实现在线接口文档的教程指南

    安装swaggo go install github.com/swaggo/swag/cmd/swag@latest 编写swag import ( “github.com/gin…

    2024年12月17日
  • Go语言中如何进行包管理

    Go语言是一种编译型、静态类型、并发型、并具有垃圾回收功能的编程语言。在Go语言中,包(package)是函数和数据的集合,用于组织代码,实现模块化开发。通过包,开发者可以将相关的…

    Golang 2024年12月17日
  • Golang解析yaml文件的方法小结

    yaml 文件是目前最常用的配置文件,使用go语言编写代码和工具时,也会用到yaml文件,将服务配置及中间件等信息定义到yaml文件中,后续可根据实际场景来选用。 //先下载外部包…

    Golang 2024年12月17日
  • 一起聊聊Go语言中的语法糖的使用

    前言 由于工作变动,我现在已经开始使用Golang了。用了一段时间之后,我发现Golang(后面简称Go)中的语法糖还蛮多的,有些语法糖还让会让人很懵逼。那么接下来,让我以一个曾经…

    Golang 2024年12月29日

发表回复

登录后才能评论