大家好~我是
米洛
!
我正在從0到1打造一個開源的介面測試平臺, 也在編寫一套與之對應的完整教程
,希望大家多多支援。
歡迎關注我的公眾號測試開發坑貨
,獲取最新文章教程!
回顧
上一節我們編寫了Redis的相關配置編輯頁面,博主這裡也趁熱打鐵,把前端頁面
完善了。(可能會有一點點小問題,但應該主流程都正常)
其實和其他配置管理頁面差不多,前端優化了一下麵包屑
,頂部的選單也放回到左側了。看看mac下的效果:
搜尋選項改動了一些,所見即所得,如果搜尋項發生變化,那麼內容也會隨之切換
關於Redis客戶端的選用
其實在這個問題上我是比較糾結
的,redis有star很多的py客戶端,也有與之對應的叢集版本。但他們並不支援asyncio
。
而支援asyncio的aioredis,本身是個很好的選擇,但人家沒有支援redis叢集的計劃。orz
所以今天想的是要不就用個同步的redis-cluster-py庫算了,不過在我翻了github一段時間,發現了個叫aredis的非同步庫。大概瞅了下,他基本上是保持了和redis-cluster-py接近的api,可能也是為了吸引使用者
。
所以我們們就先試驗一下,小白鼠嘛,總得有人來做。
安裝aredis
看官網是要安裝aredis[hiredis]
,但我好像不適合這樣方式,於是我分開裝:
pip3 install aredis hiredis
編寫RedisManager
其實這裡還是和MySQL
比較接近的,也是通過一個字典存放各個redis的連線配置。
不過由於Redis的叢集和單例項還有一點區別(好在我們編寫配置的時候就準備好了),所以我們最好是針對單例項和叢集分別編寫2個map存放他們的client,當然1個也是ok的。
整體流程: 從字典獲取客戶端,如果沒有則新開一個客戶端,並放入快取,有則返回。
-
可能存在的問題
程式碼不是執行緒安全的,需要觀察是否需要加鎖
快取不像LRU會降頻,也不能自動過期
對我來說第一個肯定是個大問題,如果出現了就必須得解決。至於第二個問題,由於redis配置很少變動,而且我們本身是連線池的形式,所以影響不算大。
話不多說,現在我們就來編寫吧:
"""
redis客戶端,基於aredis(支援叢集,aioredis不支援叢集)
"""
from aredis import StrictRedisCluster, ClusterConnectionPool, ConnectionPool, StrictRedis
from app.excpetions.RedisException import RedisException
class PityRedisManager(object):
"""非執行緒安全,可能存在問題
"""
_cluster_pool = dict()
_pool = dict()
@staticmethod
def get_cluster_client(redis_id: int, addr: str):
"""
獲取redis叢集客戶端
:param redis_id:
:param addr:
:return:
"""
cluster = PityRedisManager._cluster_pool.get(redis_id)
if cluster is not None:
return cluster
client = PityRedisManager.get_cluster(addr)
PityRedisManager._cluster_pool[redis_id] = client
return client
@staticmethod
def get_single_node_client(redis_id: int, addr: str, password: str, db: str):
"""
獲取redis單例項客戶端
:param redis_id:
:param addr:
:param password:
:param db:
:return:
"""
node = PityRedisManager._cluster_pool.get(redis_id)
if node is not None:
return node
host, port = addr.split(":")
pool = ConnectionPool(host=host, port=port, db=db, max_connections=100, password=password,
decode_responses=True)
client = StrictRedis(connection_pool=pool)
PityRedisManager._pool[redis_id] = PityRedisManager.get_cluster(addr)
return client
@staticmethod
def refresh_redis_client(redis_id: int, addr: str, password: str, db: str):
"""
重新整理redis客戶端
:param redis_id:
:param addr:
:param password:
:param db:
:return:
"""
host, port = addr.split(":")
pool = ConnectionPool(host=host, port=port, db=db, max_connections=100, password=password,
decode_responses=True)
client = StrictRedis(connection_pool=pool, decode_responses=True)
PityRedisManager._pool[redis_id] = client
@staticmethod
def refresh_redis_cluster(redis_id: int, addr: str):
PityRedisManager._cluster_pool[redis_id] = PityRedisManager.get_cluster(addr)
@staticmethod
def get_cluster(addr: str):
"""
獲取叢集連線池
:param addr:
:return:
"""
try:
nodes = addr.split(',')
startup_nodes = [{"host": n.split(":")[0], "port": n.split(":")[1]} for n in nodes]
pool = ClusterConnectionPool(startup_nodes=startup_nodes, max_connections=100, decode_responses=True)
client = StrictRedisCluster(connection_pool=pool, decode_responses=True)
return client
except Exception as e:
raise RedisException(f"獲取Redis連線失敗, {e}")
我們以資料庫的唯一id為key,快取redis的連線池
。
由於連線池會自動開啟/關閉連線,所以我們不需要手動關閉客戶端,非常方便。
可以明顯看到我們分別用了ClusterConnectionPool和ConnectionPool,分別對應叢集和例項。引數基本上算是一致。
至於refresh,是給改動redis以後做的重新整理連線
的工作。
以上就是RedisManager的內容,到這只是能夠獲取Redis客戶端了。
嘗試一下
有條件的同學可以本次安裝redis:
$ wget https://download.redis.io/releases/redis-6.2.6.tar.gz
$ tar xzf redis-6.2.6.tar.gz
$ cd redis-6.2.6
$ make
make了以後,修改redis-6.2.6目錄下的redis.conf, 接著取消這一行的註釋:
使用密碼模式(redis最好是加密碼,埠號也儘量不要用原生的6379,本寶寶有臺機器被人通過redis植入了挖礦程式,苦不堪言
)
- 在redis-6.2.6目錄下啟動
src/redis-server redis.conf
這樣本地redis的例項就啟動了~
編寫個線上測試redis的介面
-
先通過id拿到redis的配置資訊
-
然後通過manager拿到連線池
-
對redis發動命令
我們在http://localhost:7777/docs開啟swagger除錯:
-
讀取faker
- 設定faker為s12
- 再次取faker
可以看到redis的相關操作已經是可以用了,那我們今天的內容就到這了,愉快的週末總是辣麼短暫
。
下一節我們就得編寫線上執行Redis的命令及相關頁面了!