This repository has been archived by the owner on Apr 13, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 70
/
redis_utils.py
80 lines (61 loc) · 2.73 KB
/
redis_utils.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
import ipaddress
from flask_redis import FlaskRedis
from redis.exceptions import LockError
from .db_utils import DBUtils
import docker
class RedisUtils:
def __init__(self, app, user_id=0):
self.redis_client = FlaskRedis(app)
self.key = 'ctfd_whale_lock-' + str(user_id)
self.lock = None
self.global_port_key = "ctfd_whale-port-set"
self.global_network_key = "ctfd_whale-network-set"
def init_redis_port_sets(self):
configs = DBUtils.get_all_configs()
self.redis_client.delete(self.global_port_key)
self.redis_client.delete(self.global_network_key)
containers = DBUtils.get_all_container()
used_port_list = []
for container in containers:
if container.port != 0:
used_port_list.append(container.port)
for port in range(int(configs.get("frp_direct_port_minimum", 29000)), int(configs.get("frp_direct_port_maximum", 28000)) + 1):
if port not in used_port_list:
self.add_available_port(port)
client = docker.DockerClient(base_url=configs.get("docker_api_url"))
docker_subnet = configs.get("docker_subnet", "174.1.0.0/16")
try:
docker_subnet = unicode(docker_subnet)
except:
pass
docker_subnet_new_prefix = int(configs.get("docker_subnet_new_prefix", "24"))
exist_networks = []
available_networks = []
for network in client.networks.list(filters={'label': 'prefix'}):
exist_networks.append(str(network.attrs['Labels']['prefix']))
for network in list(ipaddress.ip_network(docker_subnet).subnets(new_prefix=docker_subnet_new_prefix)):
if str(network) not in exist_networks:
available_networks.append(str(network))
self.redis_client.sadd(self.global_network_key, *set(available_networks))
def add_available_network_range(self, network_range):
self.redis_client.sadd(self.global_network_key, network_range.encode())
def get_available_network_range(self):
return self.redis_client.spop(self.global_network_key).decode()
def add_available_port(self, port):
self.redis_client.sadd(self.global_port_key, str(port))
def get_available_port(self):
return int(self.redis_client.spop(self.global_port_key))
def acquire_lock(self):
lock = self.redis_client.lock(name=self.key, timeout=10)
if not lock.acquire(blocking=True, blocking_timeout=2.0):
return False
self.lock = lock
return True
def release_lock(self):
if self.lock is None:
return False
try:
self.lock.release()
return True
except LockError:
return False