|
|
- import logging
- import uuid
- import random
- import threading
- from fe.access import book
- from fe.access.new_seller import register_new_seller
- from fe.access.new_buyer import register_new_buyer
- from fe.access.buyer import Buyer
- from fe import conf
-
-
- class NewOrder:
- def __init__(self, buyer: Buyer, store_id, book_id_and_count):
- self.buyer = buyer
- self.store_id = store_id
- self.book_id_and_count = book_id_and_count
-
- def run(self) -> (bool, str):
- code, order_id = self.buyer.new_order(self.store_id, self.book_id_and_count)
- return code == 200, order_id
-
-
- class Payment:
- def __init__(self, buyer:Buyer, order_id):
- self.buyer = buyer
- self.order_id = order_id
-
- def run(self) -> bool:
- code = self.buyer.payment(self.order_id)
- return code == 200
-
-
- class Workload:
- def __init__(self):
- self.uuid = str(uuid.uuid1())
- self.book_ids = {}
- self.buyer_ids = []
- self.store_ids = []
- self.book_db = book.BookDB(conf.Use_Large_DB)
- self.row_count = self.book_db.get_book_count()
-
- self.book_num_per_store = conf.Book_Num_Per_Store
- if self.row_count < self.book_num_per_store:
- self.book_num_per_store = self.row_count
- self.store_num_per_user = conf.Store_Num_Per_User
- self.seller_num = conf.Seller_Num
- self.buyer_num = conf.Buyer_Num
- self.session = conf.Session
- self.stock_level = conf.Default_Stock_Level
- self.user_funds = conf.Default_User_Funds
- self.batch_size = conf.Data_Batch_Size
- self.procedure_per_session = conf.Request_Per_Session
-
- self.n_new_order = 0
- self.n_payment = 0
- self.n_new_order_ok = 0
- self.n_payment_ok = 0
- self.time_new_order = 0
- self.time_payment = 0
- self.lock = threading.Lock()
- # 存储上一次的值,用于两次做差
- self.n_new_order_past = 0
- self.n_payment_past = 0
- self.n_new_order_ok_past = 0
- self.n_payment_ok_past = 0
-
- def to_seller_id_and_password(self, no: int) -> (str, str):
- return "seller_{}_{}".format(no, self.uuid), "password_seller_{}_{}".format(no, self.uuid)
-
- def to_buyer_id_and_password(self, no: int) -> (str, str):
- return "buyer_{}_{}".format(no, self.uuid), "buyer_seller_{}_{}".format(no, self.uuid)
-
- def to_store_id(self, seller_no: int, i):
- return "store_s_{}_{}_{}".format(seller_no, i, self.uuid)
-
- def gen_database(self):
- logging.info("load data")
- for i in range(1, self.seller_num + 1):
- user_id, password = self.to_seller_id_and_password(i)
- seller = register_new_seller(user_id, password)
- for j in range(1, self.store_num_per_user + 1):
- store_id = self.to_store_id(i, j)
- code = seller.create_store(store_id)
- assert code == 200
- self.store_ids.append(store_id)
- row_no = 0
-
- while row_no < self.book_num_per_store:
- books = self.book_db.get_book_info(row_no, self.batch_size)
- if len(books) == 0:
- break
- for bk in books:
- code = seller.add_book(store_id, self.stock_level, bk)
- assert code == 200
- self.book_ids[store_ids].append(bk.id)
- row_no = row_no + len(books)
- logging.info("seller data loaded.")
- for k in range(1, self.buyer_num + 1):
- user_id, password = self.to_buyer_id_and_password(k)
- buyer = register_new_buyer(user_id, password)
- buyer.add_funds(self.user_funds)
- self.buyer_ids.append(user_id)
- logging.info("buyer data loaded.")
-
- def get_new_order(self) -> NewOrder:
- n = random.randint(1, self.buyer_num)
- buyer_id, buyer_password = self.to_buyer_id_and_password(n)
- store_no = int(random.uniform(0, len(self.store_ids) - 1))
- store_id = self.store_ids[store_no]
- books = random.randint(1, 10)
- book_id_and_count = []
- book_temp = []
- for i in range(0, books):
- book_no = int(random.uniform(0, len(self.book_ids[store_id]) - 1))
- book_id = self.book_ids[store_id][book_no]
- if book_id in book_temp:
- continue
- else:
- book_temp.append(book_id)
- count = random.randint(1, 10)
- book_id_and_count.append((book_id, count))
- b = Buyer(url_prefix=conf.URL, user_id=buyer_id, password=buyer_password)
- new_ord = NewOrder(b, store_id, book_id_and_count)
- return new_ord
-
- def update_stat(self, n_new_order, n_payment,
- n_new_order_ok, n_payment_ok,
- time_new_order, time_payment):
- # 获取当前并发数
- thread_num = len(threading.enumerate())
- # 加索
- self.lock.acquire()
- self.n_new_order = self.n_new_order + n_new_order
- self.n_payment = self.n_payment + n_payment
- self.n_new_order_ok = self.n_new_order_ok + n_new_order_ok
- self.n_payment_ok = self.n_payment_ok + n_payment_ok
- self.time_new_order = self.time_new_order + time_new_order
- self.time_payment = self.time_payment + time_payment
- # 计算这段时间内新创建订单的总数目
- n_new_order_diff = self.n_new_order - self.n_new_order_past
- # 计算这段时间内新付款订单的总数目
- n_payment_diff = self.n_payment - self.n_payment_past
-
- if self.n_payment != 0 and self. n_new_order != 0 \
- and (self.time_payment + self.time_new_order):
- # TPS_C(吞吐量):成功创建订单数量/(提交订单时间/提交订单并发数 + 提交付款订单时间/提交付款订单并发数)
- # NO=OK:新创建订单数量
- # Thread_num:以新提交订单的数量作为并发数(这一次的TOTAL-上一次的TOTAL)
- # TOTAL:总提交订单数量
- # LATENCY:提交订单时间/处理订单笔数(只考虑该线程延迟,未考虑并发)
- # P=OK:新创建付款订单数量
- # Thread_num:以新提交付款订单的数量作为并发数(这一次的TOTAL-上一次的TOTAL)
- # TOTAL:总付款提交订单数量
- # LATENCY:提交付款订单时间/处理付款订单笔数(只考虑该线程延迟,未考虑并发)
- logging.info("TPS_C={}, NO=OK:{} Thread_num:{} TOTAL:{} LATENCY:{} , P=OK:{} Thread_num:{} TOTAL:{} LATENCY:{}".format(
- int(self.n_new_order_ok / (self.time_payment / n_payment_diff + self.time_new_order / n_new_order_diff)), # 吞吐量:完成订单数/((付款所用时间+订单所用时间)/并发数)
- self.n_new_order_ok, n_new_order_diff, self.n_new_order, self.time_new_order / self.n_new_order, # 订单延迟:(创建订单所用时间/并发数)/新创建订单数
- self.n_payment_ok, n_payment_diff, self.n_payment, self.time_payment / self.n_payment # 付款延迟:(付款所用时间/并发数)/付款订单数
- ))
- self.lock.release()
- # 旧值更新为新值,便于下一轮计算
- self.n_new_order_past = self.n_new_order
- self.n_payment_past = self.n_payment
- self.n_new_order_ok_past = self.n_new_order_ok
- self.n_payment_ok_past = self.n_payment_ok
|