generated from container/tmpl
Julin
2 years ago
5 changed files with 599 additions and 0 deletions
@ -0,0 +1,21 @@ |
|||||
|
## 配置文件 |
||||
|
`config.ini`为项目配置文件。 |
||||
|
- `[clickhouse]`配置节为 ClickHouse 数据库的配置信息: |
||||
|
- `port`为数据库的`TCP`端口。 |
||||
|
- `password`为数据库密码,如果没有密码,则不需要配置任何值。 |
||||
|
- `[qiniu]`配置节为七牛云的配置信息: |
||||
|
- `access-key`和`secret-key`为七牛云密钥。 |
||||
|
- `bucket`为七牛云存储空间。 |
||||
|
- `domain`为存储空间对应的外链域名。 |
||||
|
|
||||
|
## requirements.txt |
||||
|
项目所需的依赖库。 |
||||
|
``` |
||||
|
pip freeze > requirements.txt |
||||
|
``` |
||||
|
|
||||
|
## 打包`.exe`文件 |
||||
|
使用`--distpath`选项指定输出目录的路径。例如,将`.exe`文件输出至当前文件夹下: |
||||
|
``` |
||||
|
(venv) E:\WorkSpace\FS-Git\fs-pep-stats-report>pyinstaller -F main.py --distpath=. |
||||
|
``` |
@ -0,0 +1,184 @@ |
|||||
|
def querystring_user_procinst_duration(start, end, except_senior, department): |
||||
|
filter_user = "and u.name not in ('刘文峰', '金亮', '姜珍', '余莎莎', '张阳根', '唐国华', '刘国勇', '刘会连', '肖琥')" if except_senior else "" |
||||
|
filter_department = "and department_name='{}'".format(department) if department else "" |
||||
|
return ''' |
||||
|
select department_name, |
||||
|
user_name, |
||||
|
procinst_id, |
||||
|
sum(duration_in_minutes) as procinst_duration_in_minutes |
||||
|
from ( |
||||
|
select distinct d.id as department_id, |
||||
|
d.name as department_name, |
||||
|
u.id as user_id, |
||||
|
u.name as user_name, |
||||
|
wp.name as process_name, |
||||
|
wpa.procinst_id as procinst_id, |
||||
|
wpa.task_id as taskinst_id, |
||||
|
wpa.task_name as task_name, |
||||
|
wpa.start_time as start_time, |
||||
|
wpa.end_time as end_time, |
||||
|
wpa.total_minutes as duration_in_minutes |
||||
|
from workflow_process_achievements as wpa |
||||
|
inner join workflow_process_history as wph on wpa.procinst_id=wph.procinst_id |
||||
|
inner join workflow_process_version as wpv on wph.version_id=wpv.id |
||||
|
inner join workflow_process as wp on wpv.process_id=wp.id |
||||
|
inner join user as u on wpa.deal_user_id=u.id |
||||
|
inner join department_user as du on u.id=du."user" |
||||
|
inner join department as d on du.department=d.id |
||||
|
where wpa.end_time >='{}' and wpa.end_time < '{}' and wp.deleted=false and wp.is_enable=true and d.delete=0 and u.delete=0 and u.state=1 and u.active_status=1 |
||||
|
{} {} |
||||
|
) as r |
||||
|
group by department_name, user_name, procinst_id |
||||
|
'''.format(start, end, filter_user, filter_department) |
||||
|
|
||||
|
|
||||
|
def querystring_senior_procinst_duration(start, end): |
||||
|
return ''' |
||||
|
SELECT user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes) /60, 2) as procinst_duration_in_hours_by_user |
||||
|
from ( |
||||
|
select user_name, |
||||
|
procinst_id, |
||||
|
sum(duration_in_minutes) as procinst_duration_in_minutes |
||||
|
from ( |
||||
|
select distinct u.id as user_id, |
||||
|
u.name as user_name, |
||||
|
wp.name as process_name, |
||||
|
wpa.procinst_id as procinst_id, |
||||
|
wpa.task_id as taskinst_id, |
||||
|
wpa.task_name as task_name, |
||||
|
wpa.start_time as start_time, |
||||
|
wpa.end_time as end_time, |
||||
|
wpa.total_minutes as duration_in_minutes |
||||
|
from workflow_process_achievements as wpa |
||||
|
inner join workflow_process_history as wph on wpa.procinst_id=wph.procinst_id |
||||
|
inner join workflow_process_version as wpv on wph.version_id=wpv.id |
||||
|
inner join workflow_process as wp on wpv.process_id=wp.id |
||||
|
inner join user as u on wpa.deal_user_id=u.id |
||||
|
where wpa.end_time >='{}' and wpa.end_time < '{}' and wp.deleted=false and wp.is_enable=true and u.delete=0 and u.state=1 and u.active_status=1 |
||||
|
and u.name in ('刘文峰', '金亮', '姜珍', '余莎莎', '张阳根', '唐国华', '刘国勇', '刘会连', '肖琥') |
||||
|
) as r |
||||
|
group by user_name, procinst_id |
||||
|
) as r2 |
||||
|
GROUP by r2.user_name |
||||
|
'''.format(start, end) |
||||
|
|
||||
|
|
||||
|
# 按月统计项企平台使用人数 |
||||
|
def querystring_user_count(start, end): |
||||
|
return ''' |
||||
|
select count(distinct deal_user_id) as pep_using_user_count |
||||
|
from workflow_process_achievements as wpa |
||||
|
inner join user as u on wpa.deal_user_id=u.id |
||||
|
where wpa.end_time >='{}' and wpa.end_time < '{}' and u.delete=0 and u.state=1 and u.active_status=1 |
||||
|
'''.format(start, end) |
||||
|
|
||||
|
|
||||
|
# 个人处理流程平均耗时 |
||||
|
def querystring_procinst_duration_by_company(start, end): |
||||
|
return ''' |
||||
|
SELECT ROUND(AVG(procinst_duration_in_minutes_by_user) /60, 2) as procinst_duration_in_hours from ( |
||||
|
SELECT user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes), 2) as procinst_duration_in_minutes_by_user |
||||
|
from ({}) as r2 |
||||
|
GROUP by r2.user_name |
||||
|
) as r3 |
||||
|
'''.format(querystring_user_procinst_duration(start, end, False, '')) |
||||
|
|
||||
|
|
||||
|
# 部门数(高于公司平均数) |
||||
|
def querystring_department_count_gt_avg(start, end, avg): |
||||
|
return ''' |
||||
|
SELECT count(*) from ( |
||||
|
SELECT department_name, |
||||
|
ROUND(AVG(procinst_duration_in_hours_by_user), 2) as procinst_duration_in_hours_by_department |
||||
|
from ( |
||||
|
SELECT department_name, |
||||
|
user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes) /60, 2) as procinst_duration_in_hours_by_user |
||||
|
from ({}) as r2 |
||||
|
GROUP by r2.department_name, r2.user_name |
||||
|
) as r3 |
||||
|
GROUP by r3.department_name |
||||
|
) as r4 |
||||
|
WHERE procinst_duration_in_hours_by_department > {} |
||||
|
'''.format(querystring_user_procinst_duration(start, end, True, ''), avg) |
||||
|
|
||||
|
|
||||
|
# 用户数(高于公司平均数) |
||||
|
def querystring_user_count_gt_avg(start, end, avg): |
||||
|
return ''' |
||||
|
SELECT count(*) from ( |
||||
|
SELECT user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes) /60, 2) as procinst_duration_in_hours_by_user |
||||
|
from ({}) as r2 |
||||
|
group by r2.user_name |
||||
|
) as r3 |
||||
|
WHERE r3.procinst_duration_in_hours_by_user > {} |
||||
|
'''.format(querystring_user_procinst_duration(start, end, False, ''), avg) |
||||
|
|
||||
|
|
||||
|
# 部门处理流程平均耗时 |
||||
|
def querystring_procinst_duration_by_department(start, end): |
||||
|
return """ |
||||
|
SELECT department_name, |
||||
|
ROUND(AVG(procinst_duration_in_hours_by_user), 2) as procinst_duration_in_hours_by_department |
||||
|
from ( |
||||
|
SELECT department_name, |
||||
|
user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes) /60, 2) as procinst_duration_in_hours_by_user |
||||
|
from ({}) as r2 |
||||
|
GROUP by r2.department_name, r2.user_name |
||||
|
) as r3 |
||||
|
GROUP by r3.department_name |
||||
|
order by procinst_duration_in_hours_by_department DESC |
||||
|
""".format(querystring_user_procinst_duration(start, end, True, '')) |
||||
|
|
||||
|
|
||||
|
# 部门处理流程平均耗时(高管) |
||||
|
def querystring_procinst_duration_by_senior(start, end): |
||||
|
return ''' |
||||
|
SELECT ROUND(AVG(procinst_duration_in_hours_by_user), 2) as procinst_duration_in_hours_by_department |
||||
|
from ({}) as r3 |
||||
|
'''.format(querystring_senior_procinst_duration(start, end)) |
||||
|
|
||||
|
|
||||
|
# 用户单流程处理耗时(高管) |
||||
|
def querystring_procinst_duration_by_user_senior(start, end): |
||||
|
return ''' |
||||
|
{} |
||||
|
order by procinst_duration_in_hours_by_user DESC |
||||
|
'''.format(querystring_senior_procinst_duration(start, end)) |
||||
|
|
||||
|
|
||||
|
# 各部门耗时较长用户(高管) |
||||
|
def querystring_user_gt_senior_avg(start, end, avg): |
||||
|
return ''' |
||||
|
SELECT * from ({}) as r3 |
||||
|
WHERE r3.procinst_duration_in_hours_by_user > {} |
||||
|
order by procinst_duration_in_hours_by_user DESC |
||||
|
'''.format(querystring_senior_procinst_duration(start, end), avg) |
||||
|
|
||||
|
|
||||
|
# 各部门耗时较长用户 |
||||
|
def querystring_user_gt_department_avg(start, end, department, avg): |
||||
|
return """ |
||||
|
SELECT * from ( |
||||
|
SELECT user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes) /60, 2) as procinst_duration_in_hours_by_user |
||||
|
from ({}) as r2 |
||||
|
group by r2.user_name |
||||
|
) as r3 |
||||
|
WHERE r3.procinst_duration_in_hours_by_user > {} |
||||
|
order by procinst_duration_in_hours_by_user desc |
||||
|
""".format(querystring_user_procinst_duration(start, end, True, department), avg) |
||||
|
|
||||
|
|
||||
|
# 按部门统计用户单流程处理耗时 |
||||
|
def querystring_procinst_duration_by_user(start, end, department): |
||||
|
return """ |
||||
|
SELECT user_name, |
||||
|
ROUND(AVG(procinst_duration_in_minutes) /60, 2) as procinst_duration_in_hours_by_user |
||||
|
from ({}) as r2 |
||||
|
group by r2.user_name |
||||
|
""".format(querystring_user_procinst_duration(start, end, True, department)) |
@ -0,0 +1,12 @@ |
|||||
|
[clickhouse] |
||||
|
host = 10.8.30.161 |
||||
|
port = 30900 |
||||
|
username = default |
||||
|
password = |
||||
|
database = pepca8 |
||||
|
|
||||
|
[qiniu] |
||||
|
access-key=5XrM4wEB9YU6RQwT64sPzzE6cYFKZgssdP5Kj3uu |
||||
|
secret-key=w6j2ixR_i-aelc6I7S3HotKIX-ukMzcKmDfH6-M5 |
||||
|
bucket=pep-resource |
||||
|
domain=https://pepsource.anxinyun.cn |
@ -0,0 +1,346 @@ |
|||||
|
import os |
||||
|
import datetime |
||||
|
import configparser |
||||
|
from clickhouse_driver import Client |
||||
|
from docx import Document |
||||
|
from docx.shared import Inches, Pt, RGBColor |
||||
|
from docx.enum.text import WD_PARAGRAPH_ALIGNMENT, WD_LINE_SPACING |
||||
|
from docx.oxml.ns import qn |
||||
|
import matplotlib.pyplot as plt |
||||
|
from app import db_helper |
||||
|
import qiniu |
||||
|
import logging |
||||
|
|
||||
|
REPORT_DIR = 'output' |
||||
|
LOG_DIR = 'log' |
||||
|
|
||||
|
|
||||
|
def get_report_year_month(): |
||||
|
# 获取上月的最后一天 |
||||
|
last_day = datetime.datetime.now().replace(day=1) - datetime.timedelta(days=1) |
||||
|
return last_day.year, last_day.month |
||||
|
|
||||
|
|
||||
|
def get_report_start_end(): |
||||
|
# 获取当前日期 |
||||
|
now = datetime.datetime.now() |
||||
|
# 获取本月第一天 |
||||
|
this_month_start = datetime.datetime(now.year, now.month, 1) |
||||
|
# 获取上个月最后一天 |
||||
|
last_month_end = this_month_start - datetime.timedelta(days=1) |
||||
|
# 获取上个月第一天 |
||||
|
last_month_start = datetime.datetime(last_month_end.year, last_month_end.month, 1) |
||||
|
return last_month_start, this_month_start |
||||
|
|
||||
|
|
||||
|
def set_paragraph_space(p): |
||||
|
# 设置间距 |
||||
|
p.paragraph_format.space_before = 0 # 段前 |
||||
|
p.paragraph_format.space_after = 0 # 段后 |
||||
|
p.paragraph_format.line_spacing = Pt(20) # 设置段落行距为20磅 |
||||
|
|
||||
|
|
||||
|
def set_paragraph_format(p): |
||||
|
set_paragraph_space(p) |
||||
|
# 设置首行缩进2个字符 |
||||
|
p.paragraph_format.first_line_indent = Inches(0.3) |
||||
|
|
||||
|
|
||||
|
def set_heading_format(h, r): |
||||
|
set_paragraph_space(h) |
||||
|
r.font.name = 'Times New Roman' |
||||
|
r._element.rPr.rFonts.set(qn('w:eastAsia'), u'宋体') |
||||
|
r.font.size = Pt(10.5) # 五号 |
||||
|
r.font.color.rgb = RGBColor(0, 0, 0) |
||||
|
r.bold = True |
||||
|
|
||||
|
|
||||
|
def set_title_format(h, r): |
||||
|
h.paragraph_format.space_before = Pt(12) # 段前1行 |
||||
|
h.paragraph_format.space_after = Pt(12) # 段后1行 |
||||
|
h.paragraph_format.line_spacing = Pt(20) # 设置段落行距为20磅 |
||||
|
h.alignment = WD_PARAGRAPH_ALIGNMENT.CENTER |
||||
|
r.font.name = u'Times New Roman' |
||||
|
r._element.rPr.rFonts.set(qn('w:eastAsia'), u'宋体') |
||||
|
r.font.size = Pt(10.5) # 五号 |
||||
|
r.font.color.rgb = RGBColor(0, 0, 0) |
||||
|
r.bold = True |
||||
|
|
||||
|
|
||||
|
def set_global_normal_style(doc): |
||||
|
style = doc.styles['Normal'] |
||||
|
style.font.name = 'Times New Roman' # 英文、数字字体 |
||||
|
style._element.rPr.rFonts.set(qn('w:eastAsia'), u'宋体') # 中文字体 |
||||
|
style.font.size = Pt(10.5) # 五号 |
||||
|
style.paragraph_format.space_before = 0 # 段前 |
||||
|
style.paragraph_format.space_after = 0 # 段后 |
||||
|
style.paragraph_format.line_spacing_rule = WD_LINE_SPACING.SINGLE # 单倍行距 |
||||
|
|
||||
|
|
||||
|
def create_bar_chart(doc, x, y, mean, title, filename): |
||||
|
plt.rcParams["font.sans-serif"] = ["SimHei"] # 设置字体 |
||||
|
plt.rcParams["axes.unicode_minus"] = False # 正常显示负号 |
||||
|
|
||||
|
if title == '部门流程处理平均耗时': |
||||
|
plt.figure(figsize=(12, 6), layout='tight') |
||||
|
plt.margins(x=0.01) |
||||
|
else: |
||||
|
plt.figure(figsize=(6.4, 3.4), layout='tight') |
||||
|
|
||||
|
# 绘制柱状图 |
||||
|
plt.bar(x, y, color=(31 / 255, 168 / 255, 201 / 255)) |
||||
|
# 绘制均值线 |
||||
|
plt.axhline(mean, linestyle='dashed', color='#FF8C00') |
||||
|
# 添加标题 |
||||
|
plt.title(title) |
||||
|
# 设置x轴标签旋转角度 |
||||
|
plt.xticks(rotation=-30, ha='left') |
||||
|
# 添加图例 |
||||
|
plt.legend(['均值:{}'.format(mean)], loc='upper right') |
||||
|
# 保存图形 |
||||
|
plt.savefig('{}/{}'.format(REPORT_DIR, filename)) |
||||
|
plt.close() |
||||
|
# 插入图形到 Word 文档中 |
||||
|
p = doc.add_paragraph() |
||||
|
p.alignment = WD_PARAGRAPH_ALIGNMENT.CENTER |
||||
|
r = p.add_run() |
||||
|
r.add_picture('{}/{}'.format(REPORT_DIR, filename), width=Inches(6)) |
||||
|
|
||||
|
|
||||
|
def add_section(doc, rank, department, d_elapse): |
||||
|
h = doc.add_heading(level=2) |
||||
|
r = h.add_run('{}. {}'.format(rank, department)) |
||||
|
set_heading_format(h, r) |
||||
|
|
||||
|
try: |
||||
|
qs = db_helper.querystring_user_gt_senior_avg( |
||||
|
start_time, end_time, d_elapse |
||||
|
) if department == '高管' else db_helper.querystring_user_gt_department_avg( |
||||
|
start_time, end_time, department, d_elapse) |
||||
|
user_gt_department_avg = client.execute(qs) |
||||
|
|
||||
|
qs = db_helper.querystring_procinst_duration_by_user_senior( |
||||
|
start_time, end_time |
||||
|
) if department == '高管' else db_helper.querystring_procinst_duration_by_user( |
||||
|
start_time, end_time, department) |
||||
|
procinst_duration_by_user = client.execute(qs) |
||||
|
except Exception as error: |
||||
|
print('数据库查询错误:', error) |
||||
|
logging.error('数据库查询错误:{}'.format(error)) |
||||
|
raise RuntimeError('数据库查询错误:', error) |
||||
|
|
||||
|
users = [] |
||||
|
for user, _ in user_gt_department_avg: |
||||
|
users.append(user) |
||||
|
|
||||
|
p = doc.add_paragraph() |
||||
|
p.add_run('{}{}月份个人处理流程平均耗时时长在公司排名为'.format(department, last_month)) |
||||
|
p.add_run('{}'.format(rank)).underline = True |
||||
|
if len(procinst_duration_by_user) == 1: |
||||
|
p.add_run(',{}公司平均水平。'.format( |
||||
|
'高于' if d_elapse > procinst_duration_by_company else |
||||
|
'低于' if d_elapse < procinst_duration_by_company else '等于')) |
||||
|
else: |
||||
|
p.add_run(',其中耗时较长的有') |
||||
|
p.add_run('{}'.format('、'.join(users))).underline = True |
||||
|
p.add_run(',{}({}小时/流程)。' |
||||
|
.format('高于公司平均水平和部门平均水平' if d_elapse >= procinst_duration_by_company else '低于公司平均水平但高于部门平均水平', |
||||
|
d_elapse)) |
||||
|
set_paragraph_format(p) |
||||
|
|
||||
|
x = [] |
||||
|
y = [] |
||||
|
for user, elapse in procinst_duration_by_user: |
||||
|
x.append(user) |
||||
|
y.append(elapse) |
||||
|
create_bar_chart(doc, x, y, d_elapse, "个人流程处理平均耗时", "{}-流程处理平均耗时.png".format(department)) |
||||
|
caption = doc.add_paragraph('图 2-{} {}个人处理流程平均耗时'.format(rank, department)) |
||||
|
caption.alignment = WD_PARAGRAPH_ALIGNMENT.CENTER |
||||
|
|
||||
|
|
||||
|
def add_chapter_2(doc): |
||||
|
h = doc.add_heading(level=1) |
||||
|
r = h.add_run('二、各部门成员详情') |
||||
|
set_heading_format(h, r) |
||||
|
|
||||
|
departments_excluded = ['汇派-质量部', '汇派-生产部', '汇派-计划部', '汇派-人事部', '汇派-采购部', '党建工会', '工程项目中心', '北京技术中心'] |
||||
|
|
||||
|
for index, (department, elapse) in enumerate(procinst_duration_by_department): |
||||
|
if department not in departments_excluded: |
||||
|
add_section(doc, index + 1, department, elapse) |
||||
|
|
||||
|
|
||||
|
def add_chapter_1(doc): |
||||
|
h = doc.add_heading(level=1) |
||||
|
r = h.add_run('一、公司各部门横向比较情况') |
||||
|
set_heading_format(h, r) |
||||
|
|
||||
|
try: |
||||
|
qs = db_helper.querystring_user_count(start_time, end_time) |
||||
|
user_count = client.execute(qs)[0][0] |
||||
|
|
||||
|
qs = db_helper.querystring_department_count_gt_avg(start_time, end_time, procinst_duration_by_company) |
||||
|
department_count_gt_avg = client.execute(qs)[0][0] |
||||
|
|
||||
|
qs = db_helper.querystring_user_count_gt_avg(start_time, end_time, procinst_duration_by_company) |
||||
|
user_count_gt_avg = client.execute(qs)[0][0] |
||||
|
except Exception as error: |
||||
|
print('数据库查询错误:', error) |
||||
|
logging.error('数据库查询错误:{}'.format(error)) |
||||
|
raise RuntimeError('数据库查询错误:', error) |
||||
|
|
||||
|
if procinst_duration_by_senior > procinst_duration_by_company: |
||||
|
department_count_gt_avg += 1 |
||||
|
|
||||
|
p = doc.add_paragraph() |
||||
|
p.add_run('{}年{}月,公司共计'.format(last_year, last_month)) |
||||
|
p.add_run('{}'.format(user_count)).underline = True |
||||
|
p.add_run('人使用项企平台,个人处理流程平均耗时') |
||||
|
p.add_run('{}'.format(procinst_duration_by_company)).underline = True |
||||
|
p.add_run('小时/流程,其中,平均单一流程处理耗时高于公司平均数的有') |
||||
|
p.add_run('{}'.format(department_count_gt_avg)).underline = True |
||||
|
p.add_run('个部门,') |
||||
|
p.add_run('{}'.format(user_count_gt_avg)).underline = True |
||||
|
p.add_run('人。') |
||||
|
set_paragraph_format(p) |
||||
|
|
||||
|
# 准备数据 |
||||
|
x = [] |
||||
|
y = [] |
||||
|
for department, elapse in procinst_duration_by_department: |
||||
|
x.append(department) |
||||
|
y.append(elapse) |
||||
|
|
||||
|
# 创建柱状图并保存为文件 |
||||
|
create_bar_chart(doc, x, y, procinst_duration_by_company, "部门流程处理平均耗时", "部门流程处理平均耗时.png") |
||||
|
caption = doc.add_paragraph('图 1 部门流程处理平均耗时') |
||||
|
caption.alignment = WD_PARAGRAPH_ALIGNMENT.CENTER |
||||
|
|
||||
|
|
||||
|
def upload_to_qiniu(filename): |
||||
|
logging.info('上传报表至七牛云...') |
||||
|
|
||||
|
# 获取 qiniu 配置参数 |
||||
|
qn_cfg = config['qiniu'] |
||||
|
access_key = qn_cfg['access-key'] |
||||
|
secret_key = qn_cfg['secret-key'] |
||||
|
bucket_name = qn_cfg['bucket'] |
||||
|
|
||||
|
q = qiniu.Auth(access_key, secret_key) |
||||
|
bucket = qiniu.BucketManager(q) |
||||
|
|
||||
|
key = 'pep-stats-report/{}'.format(filename) |
||||
|
localfile = '{}/{}'.format(REPORT_DIR, filename) |
||||
|
|
||||
|
ret, info = bucket.stat(bucket_name, key) |
||||
|
if info.status_code == 200: |
||||
|
print('文件已存在,删除文件...') |
||||
|
ret, info = bucket.delete(bucket_name, key) |
||||
|
if info.status_code == 200: |
||||
|
print('删除成功') |
||||
|
else: |
||||
|
print('删除失败') |
||||
|
|
||||
|
print('上传文件...') |
||||
|
token = q.upload_token(bucket_name, key) |
||||
|
ret, info = qiniu.put_file(token, key, localfile) |
||||
|
if info.status_code == 200: |
||||
|
print('上传成功') |
||||
|
logging.info('上传成功') |
||||
|
# 获取文件访问 URL |
||||
|
url = '{}/{}?attname='.format(qn_cfg['domain'], ret['key']) |
||||
|
print('文件访问 URL:', url) |
||||
|
logging.info('文件访问 URL:{}'.format(url)) |
||||
|
else: |
||||
|
print('上传失败,错误信息:', info.error) |
||||
|
logging.error('上传失败,错误信息:{}'.format(error)) |
||||
|
|
||||
|
|
||||
|
def generate_word_report(): |
||||
|
# 创建一个新的Word文档 |
||||
|
doc = Document() |
||||
|
set_global_normal_style(doc) |
||||
|
|
||||
|
# 添加标题 |
||||
|
h = doc.add_heading(level=0) |
||||
|
r = h.add_run('项企流程效能分析结果公示') |
||||
|
set_title_format(h, r) |
||||
|
|
||||
|
# 添加段落 |
||||
|
p1 = doc.add_paragraph( |
||||
|
'为了提升公司整体工作效率,确保跨部门工作高效推进,充分发挥我司自研平台的优势,现基于项企系统数据,利用Superset工具打造了项企流程效能分析平台,能够对我司项企平台的数据进行分析,进而通过量化数据发现我司办公流程流转中的问题,' \ |
||||
|
'现将{}年{}月公司员工处理流程平均响应时长(响应时长为从流程流转到当前节点到当前节点处理完成的耗时,单位:小时)情况结果进行公示如下:' |
||||
|
.format(last_year, last_month)) |
||||
|
set_paragraph_format(p1) |
||||
|
|
||||
|
# 一、公司各部门横向比较情况 |
||||
|
add_chapter_1(doc) |
||||
|
|
||||
|
# 二、各部门成员详情 |
||||
|
add_chapter_2(doc) |
||||
|
|
||||
|
# 保存文档 |
||||
|
report_path = '{}/项企流程效能分析结果公示(全员)-{}年{}月.docx'.format(REPORT_DIR, last_year, last_month) |
||||
|
doc.save(report_path) |
||||
|
logging.info('本地报表创建成功:{}'.format(report_path)) |
||||
|
|
||||
|
# 上传七牛云 |
||||
|
upload_to_qiniu('项企流程效能分析结果公示(全员)-{}年{}月.docx'.format(last_year, last_month)) |
||||
|
|
||||
|
|
||||
|
def create_clickhouse_client(): |
||||
|
# 获取 clickhouse 配置参数 |
||||
|
ch_cfg = config['clickhouse'] |
||||
|
host = ch_cfg['host'] |
||||
|
port = ch_cfg.getint('port') |
||||
|
username = ch_cfg['username'] |
||||
|
password = ch_cfg['password'] |
||||
|
database = ch_cfg['database'] |
||||
|
# 创建 ClickHouse 客户端对象 |
||||
|
return Client(host=host, port=port, user=username, password=password, database=database) |
||||
|
|
||||
|
|
||||
|
def try_create_dir(dir_name): |
||||
|
# 判断目录是否存在,如果不存在,则创建目录 |
||||
|
if not os.path.exists(dir_name): |
||||
|
print(f"目录'{dir_name}'不存在,即将创建...") |
||||
|
os.makedirs(dir_name) |
||||
|
print(f"目录'{dir_name}'创建成功!") |
||||
|
|
||||
|
|
||||
|
if __name__ == '__main__': |
||||
|
try: |
||||
|
try_create_dir(REPORT_DIR) |
||||
|
try_create_dir(LOG_DIR) |
||||
|
logging.basicConfig(filename='{}/runtime.log'.format(LOG_DIR), level=logging.INFO, |
||||
|
format='%(asctime)s.%(msecs)03d - %(levelname)s: %(message)s', |
||||
|
datefmt='%Y-%m-%d %H:%M:%S') |
||||
|
|
||||
|
config = configparser.ConfigParser() |
||||
|
config.read('config.ini') |
||||
|
|
||||
|
client = create_clickhouse_client() |
||||
|
|
||||
|
last_year, last_month = get_report_year_month() |
||||
|
start_time, end_time = get_report_start_end() |
||||
|
|
||||
|
qs1 = db_helper.querystring_procinst_duration_by_company(start_time, end_time) |
||||
|
procinst_duration_by_company = client.execute(qs1)[0][0] |
||||
|
|
||||
|
qs2 = db_helper.querystring_procinst_duration_by_department(start_time, end_time) |
||||
|
procinst_duration_by_department = client.execute(qs2) |
||||
|
|
||||
|
qs3 = db_helper.querystring_procinst_duration_by_senior(start_time, end_time) |
||||
|
procinst_duration_by_senior = client.execute(qs3)[0][0] |
||||
|
|
||||
|
for index, (department, elapse) in enumerate(procinst_duration_by_department): |
||||
|
if procinst_duration_by_senior >= elapse: |
||||
|
procinst_duration_by_department.insert(index, ('高管', procinst_duration_by_senior)) |
||||
|
break |
||||
|
|
||||
|
generate_word_report() # 生成报表文件 |
||||
|
except Exception as error: |
||||
|
print('程序运行出错:', error) |
||||
|
logging.error('程序运行出错:{}'.format(error)) |
||||
|
|
||||
|
# input("请按任意键退出...") |
@ -0,0 +1,36 @@ |
|||||
|
altgraph==0.17.3 |
||||
|
APScheduler==3.10.1 |
||||
|
backports.zoneinfo==0.2.1 |
||||
|
certifi==2022.12.7 |
||||
|
charset-normalizer==3.1.0 |
||||
|
clickhouse-driver==0.2.5 |
||||
|
contourpy==1.0.7 |
||||
|
cycler==0.11.0 |
||||
|
decorator==5.1.1 |
||||
|
fonttools==4.39.3 |
||||
|
idna==3.4 |
||||
|
importlib-resources==5.12.0 |
||||
|
kiwisolver==1.4.4 |
||||
|
logging==0.4.9.6 |
||||
|
lxml==4.9.2 |
||||
|
matplotlib==3.7.1 |
||||
|
numpy==1.24.2 |
||||
|
packaging==23.0 |
||||
|
pefile==2023.2.7 |
||||
|
Pillow==9.5.0 |
||||
|
pyinstaller==5.10.1 |
||||
|
pyinstaller-hooks-contrib==2023.2 |
||||
|
pyparsing==3.0.9 |
||||
|
python-dateutil==2.8.2 |
||||
|
python-docx==0.8.11 |
||||
|
pytz==2023.3 |
||||
|
pytz-deprecation-shim==0.1.0.post0 |
||||
|
pywin32-ctypes==0.2.0 |
||||
|
qiniu==7.10.0 |
||||
|
requests==2.28.2 |
||||
|
self==2020.12.3 |
||||
|
six==1.16.0 |
||||
|
tzdata==2023.3 |
||||
|
tzlocal==4.3 |
||||
|
urllib3==1.26.15 |
||||
|
zipp==3.15.0 |
Loading…
Reference in new issue