"""
|
常用工具
|
"""
|
import ctypes
|
import decimal
|
import math
|
import random
|
import threading
|
import time
|
import time as t
|
import datetime
|
from threading import Thread
|
|
import constant
|
|
|
def async_call(fn):
|
def wrapper(*args, **kwargs):
|
Thread(target=fn, args=args, kwargs=kwargs).start()
|
|
return wrapper
|
|
|
def get_expire():
|
now = int(t.time())
|
end = int(t.time()) + 60 * 60 * 24
|
local_time = t.strftime("%Y-%m-%d", t.localtime(end))
|
end = int(t.mktime(t.strptime(local_time, "%Y-%m-%d")))
|
expire = end - now
|
# 加随机数,防止一起销毁数据
|
expire += random.randint(0, 60 * 30)
|
return expire
|
|
|
def get_now_date():
|
date = datetime.datetime.now()
|
return date
|
|
|
def get_now_date_str(format="%Y-%m-%d"):
|
date = datetime.datetime.now().strftime(format)
|
return date
|
|
|
# 日期减去多少天
|
def date_sub(date_str, day, format="%Y-%m-%d"):
|
t_ = time.mktime(time.strptime(date_str, format))
|
t_ -= day * 24 * 60 * 60
|
return time.strftime(format, t.localtime(t_))
|
|
|
def get_now_time_str():
|
time_str = datetime.datetime.now().strftime("%H:%M:%S")
|
return time_str
|
|
|
def get_now_time_with_ms_str():
|
now = datetime.datetime.now()
|
ms = int(now.microsecond / 1000)
|
time_str = now.strftime(f"%H:%M:%S.{ms:03d}")
|
return time_str
|
|
|
def get_now_datetime_str():
|
time_str = datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
return time_str
|
|
|
def get_now_timestamp():
|
return round(time.time() * 1000)
|
|
|
# 转为价格,四舍五入保留2位小数
|
def to_price(_decimal):
|
return _decimal.quantize(decimal.Decimal("0.00"), decimal.ROUND_HALF_UP)
|
|
|
# 是否为交易时间
|
def is_trade_time():
|
# 测试
|
if constant.TEST:
|
return True
|
|
relative_timestamp = t.time() % (24 * 60 * 60) + 8 * 60 * 60
|
start1 = 60 * 60 * 9 + 24 * 60
|
end1 = 60 * 60 * 11 + 31 * 60
|
start2 = 60 * 60 * 12 + 58 * 60
|
end2 = 60 * 60 * 15 + 1 * 60
|
if start1 < relative_timestamp < end1 or start2 < relative_timestamp < end2:
|
return True
|
else:
|
return False
|
|
|
# 是否为报警时间
|
def is_alert_time():
|
relative_timestamp = t.time() % (24 * 60 * 60) + 8 * 60 * 60
|
start1 = 60 * 60 * 9 + 29 * 60
|
end1 = 60 * 60 * 11 + 29 * 60
|
start2 = 60 * 60 * 13
|
end2 = 60 * 60 * 14 + 54 * 60
|
if start1 < relative_timestamp < end1 or start2 < relative_timestamp < end2:
|
return True
|
else:
|
return False
|
|
|
# 是否为修复时间
|
def is_repaire_time():
|
relative_timestamp = t.time() % (24 * 60 * 60) + 8 * 60 * 60
|
start1 = 60 * 60 * 9 + 29 * 60
|
end1 = 60 * 60 * 11 + 29 * 60
|
start2 = 60 * 60 * 13
|
end2 = 60 * 60 * 15
|
if start1 < relative_timestamp < end1 or start2 < relative_timestamp < end2:
|
return True
|
else:
|
return False
|
|
|
# 是否为初始化时间
|
def is_init_time():
|
relative_timestamp = t.time() % (24 * 60 * 60) + 8 * 60 * 60
|
start1 = 60 * 60 * 9 + 30 * 60
|
end1 = 60 * 60 * 15 + 1 * 60
|
if start1 < relative_timestamp < end1:
|
return False
|
else:
|
return True
|
|
|
def is_set_code_time():
|
# 测试
|
if constant.TEST:
|
return True
|
|
relative_timestamp = t.time() % (24 * 60 * 60) + 8 * 60 * 60
|
start1 = 60 * 60 * 9 + 14 * 60
|
end1 = 60 * 60 * 11 + 35 * 60
|
start2 = 60 * 60 * 12 + 50 * 60
|
end2 = 60 * 60 * 15 + 5 * 60
|
if start1 < relative_timestamp < end1 or start2 < relative_timestamp < end2:
|
return True
|
else:
|
return False
|
|
|
def run_time():
|
def decorator(func):
|
def infunc(*args, **kwargs):
|
start = round(time.time() * 1000)
|
result = func(args, **kwargs)
|
print("执行时间", round(time.time() * 1000) - start)
|
return result
|
|
return infunc
|
|
return decorator
|
|
|
def get_time_as_second(time_str):
|
ts = time_str.split(":")
|
return int(ts[0]) * 3600 + int(ts[1]) * 60 + int(ts[2])
|
|
|
def get_time_as_millionsecond(time_str):
|
s_str, ms_str = time_str.split(".")
|
ts = s_str.split(":")
|
return (int(ts[0]) * 3600 + int(ts[1]) * 60 + int(ts[2])) * 1000 + int(ms_str)
|
|
|
# 将秒数格式化为时间
|
def time_seconds_format(seconds):
|
h = seconds // 3600
|
m = seconds % 3600 // 60
|
s = seconds % 60
|
return "{0:0>2}:{1:0>2}:{2:0>2}".format(h, m, s)
|
|
|
def time_millionseconds_format(millionseconds):
|
ms = millionseconds % 1000
|
seconds = millionseconds // 1000
|
h = seconds // 3600
|
m = seconds % 3600 // 60
|
s = seconds % 60
|
return "{0:0>2}:{1:0>2}:{2:0>2}.{3:0>3}".format(h, m, s, ms)
|
|
|
def timestamp_format(timestamp, format):
|
"""
|
时间戳格式化
|
@param timestamp:
|
@param format:
|
@return:
|
"""
|
return datetime.datetime.fromtimestamp(timestamp).strftime(format)
|
|
|
# 交易時間的差值
|
# 如11:29:59 与 13:00:00只相差1s
|
def trade_time_sub(time_str_1, time_str_2):
|
split_time = get_time_as_second("11:30:00")
|
time_1 = get_time_as_second(time_str_1)
|
time_2 = get_time_as_second(time_str_2)
|
if time_1 < split_time < time_2:
|
time_2 = time_2 - 90 * 60
|
elif time_2 < split_time < time_1:
|
time_2 = time_2 + 90 * 60
|
|
return time_1 - time_2
|
|
|
def trade_time_sub_with_ms(time_str_1, time_str_2):
|
split_time = get_time_as_second("11:30:00") * 1000
|
time_1 = get_time_as_millionsecond(time_str_1)
|
time_2 = get_time_as_millionsecond(time_str_2)
|
if time_1 < split_time < time_2:
|
time_2 = time_2 - 90 * 60 * 1000
|
elif time_2 < split_time < time_1:
|
time_2 = time_2 + 90 * 60 * 1000
|
|
return time_1 - time_2
|
|
|
def compare_time(time_1: str, time_2: str):
|
return int(time_1.replace(":", "")) - int(time_2.replace(":", ""))
|
|
|
def compare_time_with_ms(time_1: str, time_2: str):
|
return int(time_1.replace(":", "").replace(".", "")) - int(time_2.replace(":", "").replace(".", ""))
|
|
|
# 交易时间加几s
|
def trade_time_add_second(time_str, second):
|
ts = time_str.split(":")
|
s_ = int(ts[0]) * 3600 + int(ts[1]) * 60 + int(ts[2])
|
s = s_ + second
|
# 是否在11:30:00
|
if s >= 11 * 3600 + 30 * 60 > s_:
|
s += 90 * 60
|
return time_seconds_format(s)
|
|
|
def trade_time_add_millionsecond(time_str, millionsecond):
|
s_str, ms_str = time_str.split(".")
|
ts = s_str.split(":")
|
s_ = int(ts[0]) * 3600 + int(ts[1]) * 60 + int(ts[2])
|
ms_ = s_ * 1000
|
ms_ += int(ms_str)
|
ms = ms_ + millionsecond
|
# 是否在11:30:00
|
if ms >= 11 * 3600 * 1000 + 30 * 60 * 1000 > ms_:
|
ms += 90 * 60 * 1000
|
return time_millionseconds_format(ms)
|
|
|
def compute_buy1_real_time(time_):
|
ts = time_.split(":")
|
s = int(ts[0]) * 3600 + int(ts[1]) * 60 + int(ts[2])
|
cha = (s - 2) % 3
|
return time_seconds_format(s - 2 - cha)
|
|
|
def to_time_with_ms(time_s_str, time_ms):
|
"""
|
将时间+毫秒数 转为字符串
|
@param time_s_str: 时间如:10:00:00
|
@param time_ms: 毫秒如: 12
|
@return: 如:10:00:00.001
|
"""
|
return f"{time_s_str}." + "{0:0>3}".format(time_ms)
|
|
|
# 全角转半角
|
def strQ2B(ustring):
|
rstring = ""
|
for uchar in ustring:
|
inside_code = ord(uchar)
|
if inside_code == 12288: # 全角空格直接转换
|
inside_code = 32
|
elif 65281 <= inside_code <= 65374: # 全角字符(除空格)根据关系转化
|
inside_code -= 65248
|
rstring += chr(inside_code)
|
return rstring
|
|
|
# 将时间戳s格式化
|
def to_time_str(timestamp_second, format_="%H:%M:%S"):
|
return datetime.datetime.fromtimestamp(timestamp_second).strftime(format_)
|
|
|
# 获取买入价格笼子的最低价
|
def get_buy_min_price(price):
|
price1 = price * (1 - 0.02)
|
price1 = math.ceil(price1 * 100) / 100
|
price2 = price - 0.1
|
return max(price1, price2)
|
|
|
# 获取买入价格笼子的最低价
|
def get_shadow_price(price):
|
# fprice = round((100 - random.randint(2, 10)) * price / 100, 2)
|
# if price - 0.1 < fprice:
|
# fprice = price - 0.1
|
# return round(fprice, 2)
|
if price < 20:
|
return round(get_buy_min_price(price) - 0.03, 2)
|
else:
|
# 大股价直接向下取2%
|
return round(price*(1-0.02), 2)
|
|
|
if __name__ == "__main__":
|
print(trade_time_sub("13:00:01",
|
"11:29:55"))
|
|
|
class CodeDataCacheUtil:
|
# --缓存操作--
|
@classmethod
|
def clear_cache(cls, cache_dict, code):
|
if code in cache_dict:
|
cache_dict.pop(code)
|
|
@classmethod
|
def set_cache(cls, cache_dict, code, data):
|
cache_dict[code] = data
|
|
@classmethod
|
def get_cache(cls, cache_dict, code):
|
if code in cache_dict:
|
return True, cache_dict.get(code)
|
return False, None
|
|
|
def is_can_buy_code(code):
|
if code.find("00") == 0 or code.find("60") == 0 or code.find("30") == 0:
|
return True
|
return False
|
|
|
def is_target_code(code):
|
"""
|
是否是目标代码
|
@param code:
|
@return:
|
"""
|
prefixes = ["00", "60", "30"]
|
for prefix in prefixes:
|
if code.find(prefix) == 0:
|
return True
|
return False
|
|
|
def get_limit_up_rate(code):
|
# 获取涨停倍数
|
if code.find("00") == 0 or code.find("60") == 0:
|
return 1.1
|
else:
|
return 1.2
|
|
|
def get_limit_down_rate(code):
|
# 获取涨停倍数
|
if code.find("00") == 0 or code.find("60") == 0:
|
return 0.9
|
else:
|
return 0.8
|
|
|
def get_thread_id():
|
try:
|
if constant.is_windows():
|
return threading.current_thread().ident
|
else:
|
thread_id = ctypes.CDLL('libc.so.6').syscall(186) # Linux下的系统调用号
|
return thread_id
|
except:
|
pass
|
return None
|
|
|
def get_buy_volume(limit_up_price):
|
return get_buy_volume_by_money(limit_up_price, constant.BUY_MONEY_PER_CODE)
|
|
|
def get_buy_volume_by_money(limit_up_price, money):
|
count = (money // int(round(float(limit_up_price) * 100))) * 100
|
if count < 100:
|
count = 100
|
return count
|
|
|
# 深证
|
MARKET_TYPE_SZSE = 1
|
# 上证
|
MARKET_TYPE_SSE = 0
|
# 未知
|
MARKET_TYPE_UNKNOWN = -1
|
|
|
def get_market_type(code):
|
"""
|
根据股票代码
|
:param code:
|
:return:
|
"""
|
if code.find("00") == 0 or code.find("30") == 0 or code.find("12") == 0:
|
return MARKET_TYPE_SZSE
|
elif code.find("60") == 0 or code.find("68") == 0 or code.find("11") == 0:
|
return MARKET_TYPE_SSE
|
else:
|
return MARKET_TYPE_UNKNOWN
|
|
|
def is_sh_code(code):
|
"""
|
是否是上证
|
@param code:
|
@return:
|
"""
|
return get_market_type(code) == MARKET_TYPE_SSE
|
|
|
def is_sz_code(code):
|
"""
|
是否是深证
|
@param code:
|
@return:
|
"""
|
return get_market_type(code) == MARKET_TYPE_SZSE
|
|
|
def is_ge_code(code):
|
"""
|
是否是创业板
|
@param code:
|
@return:
|
"""
|
return code.find("30") == 0
|
|
|
if __name__ == "__main__":
|
print(timestamp_format(1726034271, "%H%M%S"))
|