在Redis資料庫中實現分布式速率限制的方法_Redis

來源:互聯網
上載者:User

問題

在許多應用中,對昂貴的資源的訪問必須加以限制,此時速率限制是必不可少的。許多現代網路應用程式在多個進程和伺服器上運行,狀態需要被共用。一個理想的解決方案應該是高效、 快捷的,而不是依賴於被綁定到特定用戶端的單個應用程式伺服器(由於Server Load Balancer) 或本身持有任何狀態。
解決方案

實現這一目標的一個簡單有效方法就是使用 Redis, 它有很多有用的資料結構和功能, 儘管實現速率限制只需要2個功能用: 一、在某個具體的索引值上遞增一個整數,二、給這個索引值設定到期時間。

因為redis 有個單一的事件迴圈系統 (每個人每次在同一個時間只能執行一個操作),這是個原子操作, 也就是說無論有多少個用戶端同時互動操作,對於同一個索引值總有一個確定的數值。

這在對同一個資源進行多個速率限制的情況下通常是有利的, 因為這允許少量的破裂,以及更長的期限限制。例如每秒鐘請求3次,沒分鐘請求20次。因為每個限制都是相對獨立的,這就需要與其它限制分開進行單獨的遞增。

因為速率限制通常用在回應時間比較重要的資源(比如網頁應用),所以盡量縮短速率限制的使用時間是非常有必要的。redis的最基本的應用就是發出命令,等待響應,然後發出另一個命令,如此往複。 這個花費是昂貴的,因為需要通過網路在應用程式和redis伺服器之間多次往返。由於在這個用例中,沒有命令依賴其它命令的執行結果,這使得redis的一個叫做流水線技術的使用成為可能。這就是用戶端緩衝所有redis請求,然後把這寫請求發送給redis,redis一次性返回所有的結果。

Redis不會維護用戶端需要的限制的,因為redis會根據用戶端設定的到期時間刪除舊的記數。這消除了用戶端統籌協調的需要,和刪除競爭條件的可能性。

The Code
 

import redisimport time def rate_limit_check(r, key, limits):  period_lengths = [_[0] for _ in sorted(limits.items())]  period_limits = [_[1] for _ in sorted(limits.items())]  pipe = r.pipeline()  for period_length in period_lengths:     current_period = int(time.time() / period_length)     redis_key = 'rate_limit:{key}:{period_length}:{current_period}'.format(key=key, period_length=period_length, current_period=current_period)     pipe.incr(redis_key).expire(redis_key, period_length*3)  return not any(hits > period_limit for period_limit, hits in zip(period_limits, pipe.execute()[::2])) if __name__ == '__main__':  r = redis.Redis()  print rate_limit_check(r, '127.0.0.1', {1: 3, 60: 20})

{1: 3, 60: 20} 意味著每秒鐘3次的命中率是允許的,在任何限制下,都允許20次的命中。'127.0.0.1'在這裡用作索引值,儘管在真實的情況下,可能作為IP地址。更進階的用例將有一個全應用程式的速率限制,索引值只有用戶端的IP地址,以及一個為昂貴的終結點設定的特定終結點限制,這將用到用戶端的IP地址和終結點,例如127.0.0.1+/login/。這些限制可以獨立地設定。
 

return rate_limit_check(r, '127.0.0.1', {1: 3, 60: 20}) and rate_limit_check(r, '127.0.0.1+/login/', {1: 2, 60: 5})

這是一個用Python寫的例子,它可以簡單地移植到任何語言,只要這門語言套件含Redis用戶端庫。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.