-
Notifications
You must be signed in to change notification settings - Fork 1
/
user_manager.py
409 lines (348 loc) · 17.7 KB
/
user_manager.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
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
import jaydebeapi
from PySide6.QtWidgets import (QWidget, QVBoxLayout, QHBoxLayout, QPushButton, QTableWidget, QTableWidgetItem,
QMessageBox, QInputDialog, QLineEdit, QComboBox, QDialog, QFormLayout, QCheckBox, QDialogButtonBox, QLabel)
from PySide6.QtCore import Qt
from PySide6.QtGui import QIcon
class PasswordLineEdit(QWidget):
def __init__(self, parent=None):
super().__init__(parent)
layout = QHBoxLayout(self)
layout.setContentsMargins(0, 0, 0, 0)
self.password_input = QLineEdit(self)
self.password_input.setEchoMode(QLineEdit.Password)
layout.addWidget(self.password_input)
self.toggle_button = QPushButton("顯示密碼", self)
self.toggle_button.clicked.connect(self.toggle_password_visibility)
layout.addWidget(self.toggle_button)
def toggle_password_visibility(self):
if self.password_input.echoMode() == QLineEdit.Password:
self.password_input.setEchoMode(QLineEdit.Normal)
self.toggle_button.setText("隱藏密碼")
else:
self.password_input.setEchoMode(QLineEdit.Password)
self.toggle_button.setText("顯示密碼")
def text(self):
return self.password_input.text()
class CreateUserDialog(QDialog):
def __init__(self, parent=None):
super().__init__(parent)
self.setWindowTitle("創建新用戶")
self.layout = QFormLayout(self)
# 用戶名
self.username_input = QLineEdit(self)
self.layout.addRow("用戶名:", self.username_input)
# 密碼
self.password_input = PasswordLineEdit(self)
self.layout.addRow("密碼:", self.password_input)
# 描述
self.description_input = QLineEdit(self)
self.layout.addRow("描述:", self.description_input)
# User Class
self.user_class_combo = QComboBox(self)
self.user_class_combo.addItems(["*USER", "*SYSOPR", "*PGMR", "*SECADM", "*SECOFR"])
self.layout.addRow("User Class:", self.user_class_combo)
# 特殊權限
self.special_auth_checkboxes = []
special_auths = ["*ALLOBJ", "*AUDIT", "*IOSYSCFG", "*JOBCTL", "*SAVSYS", "*SECADM", "*SERVICE", "*SPLCTL"]
auth_layout = QVBoxLayout()
for auth in special_auths:
checkbox = QCheckBox(auth)
self.special_auth_checkboxes.append(checkbox)
auth_layout.addWidget(checkbox)
self.layout.addRow("特殊權限:", auth_layout)
# 確認和取消按鈕
self.button_box = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
self.button_box.accepted.connect(self.accept)
self.button_box.rejected.connect(self.reject)
self.layout.addRow(self.button_box)
def get_user_info(self):
special_auths = [cb.text() for cb in self.special_auth_checkboxes if cb.isChecked()]
return {
"username": self.username_input.text(),
"password": self.password_input.text(),
"description": self.description_input.text(),
"user_class": self.user_class_combo.currentText(),
"special_authorities": special_auths
}
class UserManager:
def __init__(self, connection):
self.connection = connection
def list_users(self):
"""列出所有使用者"""
query = """
SELECT USER_NAME, STATUS, PREVIOUS_SIGNON, PASSWORD_CHANGE_DATE
FROM QSYS2.USER_INFO
ORDER BY USER_NAME
"""
return self._execute_query(query)
def create_user(self, username, password, description="", user_class="*USER", special_authorities=[]):
"""創建新使用者"""
special_auth = " ".join(special_authorities)
cmd = f"CRTUSRPRF USRPRF({username}) PASSWORD({password}) TEXT('{description}') USRCLS({user_class}) SPCAUT({special_auth})"
self._execute_command(cmd)
def delete_user(self, username):
"""刪除使用者"""
cmd = f"DLTUSRPRF USRPRF({username})"
self._execute_command(cmd)
def change_password(self, username, new_password):
"""更改使用者密碼"""
cmd = f"CHGUSRPRF USRPRF({username}) PASSWORD({new_password})"
self._execute_command(cmd)
def disable_user(self, username):
"""停用用戶帳號"""
cmd = f"CHGUSRPRF USRPRF({username}) STATUS(*DISABLED)"
self._execute_command(cmd)
def enable_user(self, username):
"""啟用用戶帳號"""
cmd = f"CHGUSRPRF USRPRF({username}) STATUS(*ENABLED)"
self._execute_command(cmd)
def modify_user_authorities(self, username, user_class, special_authorities):
"""修改用戶的 User Class 和特殊權限"""
special_auth = " ".join(special_authorities)
cmd = f"CHGUSRPRF USRPRF({username}) USRCLS({user_class}) SPCAUT({special_auth})"
self._execute_command(cmd)
def get_user_spool_files(self, username):
"""獲取指定用戶的 spool files"""
query = f"""
SELECT * FROM QSYS2.OUTPUT_QUEUE_ENTRIES_BASIC
WHERE USER_NAME = '{username}'
ORDER BY SIZE DESC
FETCH FIRST 100 ROWS ONLY
"""
return self._execute_query(query)
def _execute_query(self, query):
try:
with self.connection.cursor() as cursor:
cursor.execute(query)
columns = [desc[0] for desc in cursor.description]
result = cursor.fetchall()
return columns, result
except Exception as e:
print(f"執行查詢時發生錯誤: {str(e)}")
return None
def _execute_command(self, cmd):
try:
with self.connection.cursor() as cursor:
cursor.execute("CALL QSYS2.QCMDEXC(?)", (cmd,))
except Exception as e:
print(f"執行命令時發生錯誤: {str(e)}")
raise
class UserManagerGUI(QWidget):
def __init__(self, parent, user_manager):
super().__init__(parent)
self.parent = parent
self.user_manager = user_manager
self.initUI()
def initUI(self):
layout = QVBoxLayout(self)
# 創建按鈕
button_layout = QHBoxLayout()
self.refresh_button = QPushButton('刷新用戶列表')
self.create_button = QPushButton('創建用戶')
self.delete_button = QPushButton('刪除用戶')
self.change_password_button = QPushButton('更改密碼')
self.disable_user_button = QPushButton('停用帳號')
self.enable_user_button = QPushButton('啟用帳號')
self.modify_authorities_button = QPushButton('修改權限')
self.view_spool_files_button = QPushButton('查看 Spool Files')
button_layout.addWidget(self.refresh_button)
button_layout.addWidget(self.create_button)
button_layout.addWidget(self.delete_button)
button_layout.addWidget(self.change_password_button)
button_layout.addWidget(self.disable_user_button)
button_layout.addWidget(self.enable_user_button)
button_layout.addWidget(self.modify_authorities_button)
button_layout.addWidget(self.view_spool_files_button)
layout.addLayout(button_layout)
# 創建表格
self.user_table = QTableWidget()
layout.addWidget(self.user_table)
# 連接按鈕信號
self.refresh_button.clicked.connect(self.refresh_user_list)
self.create_button.clicked.connect(self.create_user_dialog)
self.delete_button.clicked.connect(self.delete_user_dialog)
self.change_password_button.clicked.connect(self.change_password_dialog)
self.disable_user_button.clicked.connect(self.disable_user_dialog)
self.enable_user_button.clicked.connect(self.enable_user_dialog)
self.modify_authorities_button.clicked.connect(self.modify_authorities_dialog)
self.view_spool_files_button.clicked.connect(self.show_user_spool_files)
# 初始化用戶列表
self.refresh_user_list()
def refresh_user_list(self):
result = self.user_manager.list_users()
if result:
columns, data = result
self.user_table.setColumnCount(len(columns))
self.user_table.setRowCount(len(data))
self.user_table.setHorizontalHeaderLabels(columns)
for row, user_data in enumerate(data):
for col, value in enumerate(user_data):
self.user_table.setItem(row, col, QTableWidgetItem(str(value)))
self.user_table.resizeColumnsToContents()
self.user_table.setSelectionBehavior(QTableWidget.SelectRows)
else:
QMessageBox.warning(self, "錯誤", "無法取用戶列表")
def create_user_dialog(self):
dialog = CreateUserDialog(self)
if dialog.exec_():
user_info = dialog.get_user_info()
try:
self.user_manager.create_user(
user_info["username"],
user_info["password"],
user_info["description"],
user_info["user_class"],
user_info["special_authorities"]
)
QMessageBox.information(self, "成功", f"用戶 {user_info['username']} 已成功創建")
self.refresh_user_list()
except Exception as e:
QMessageBox.critical(self, "錯誤", f"創建用戶時發生錯誤:{str(e)}")
def delete_user_dialog(self):
username, ok = QInputDialog.getText(self, "刪除用戶", "輸入要刪除的用戶名:")
if ok and username:
confirm = QMessageBox.question(self, "確認", f"確定要刪除用戶 {username} 嗎?",
QMessageBox.Yes | QMessageBox.No)
if confirm == QMessageBox.Yes:
try:
self.user_manager.delete_user(username)
QMessageBox.information(self, "成功", f"用戶 {username} 已成功刪除")
self.refresh_user_list()
except Exception as e:
QMessageBox.critical(self, "錯誤", f"刪除用戶時發生錯誤:{str(e)}")
def change_password_dialog(self):
username, ok = QInputDialog.getText(self, "更改密碼", "輸入要更改密碼的用戶名:")
if ok and username:
password_dialog = QDialog(self)
password_dialog.setWindowTitle("輸入新密碼")
layout = QVBoxLayout(password_dialog)
password_input = PasswordLineEdit(password_dialog)
layout.addWidget(password_input)
buttons = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel, password_dialog)
buttons.accepted.connect(password_dialog.accept)
buttons.rejected.connect(password_dialog.reject)
layout.addWidget(buttons)
if password_dialog.exec_() == QDialog.Accepted:
new_password = password_input.text()
try:
self.user_manager.change_password(username, new_password)
QMessageBox.information(self, "成功", f"用戶 {username} 的密碼已成功更改")
except Exception as e:
QMessageBox.critical(self, "錯誤", f"更改密碼時發生錯誤:{str(e)}")
def disable_user_dialog(self):
username, ok = QInputDialog.getText(self, "停用帳號", "輸入要停用的用戶名:")
if ok and username:
try:
self.user_manager.disable_user(username)
QMessageBox.information(self, "成功", f"用戶 {username} 的帳號已成功停用")
self.refresh_user_list()
except Exception as e:
QMessageBox.critical(self, "錯誤", f"停用帳號時發生錯誤:{str(e)}")
def enable_user_dialog(self):
username, ok = QInputDialog.getText(self, "啟用帳號", "輸入要啟用的用戶名:")
if ok and username:
try:
self.user_manager.enable_user(username)
QMessageBox.information(self, "成功", f"用戶 {username} 的帳號已成功啟用")
self.refresh_user_list()
except Exception as e:
QMessageBox.critical(self, "錯誤", f"啟用帳號時發生錯誤:{str(e)}")
def modify_authorities_dialog(self):
username, ok = QInputDialog.getText(self, "修改改用戶權限", "輸入要修改權限的用戶名:")
if ok and username:
dialog = QDialog(self)
dialog.setWindowTitle(f"修改 {username} 的權限")
layout = QVBoxLayout(dialog)
user_class_combo = QComboBox()
user_class_combo.addItems(["*USER", "*SYSOPR", "*PGMR", "*SECADM", "*SECOFR"])
layout.addWidget(QLabel("User Class:")) # Now QLabel is defined
layout.addWidget(user_class_combo)
special_auths = ["*ALLOBJ", "*AUDIT", "*IOSYSCFG", "*JOBCTL", "*SAVSYS", "*SECADM", "*SERVICE", "*SPLCTL"]
auth_checkboxes = []
auth_layout = QVBoxLayout()
for auth in special_auths:
checkbox = QCheckBox(auth)
auth_checkboxes.append(checkbox)
auth_layout.addWidget(checkbox)
layout.addWidget(QLabel("特殊權限:"))
layout.addLayout(auth_layout)
button_box = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel)
button_box.accepted.connect(dialog.accept)
button_box.rejected.connect(dialog.reject)
layout.addWidget(button_box)
if dialog.exec_() == QDialog.Accepted:
new_user_class = user_class_combo.currentText()
new_special_auths = [cb.text() for cb in auth_checkboxes if cb.isChecked()]
try:
self.user_manager.modify_user_authorities(username, new_user_class, new_special_auths)
QMessageBox.information(self, "成功", f"用戶 {username} 的權限已成功修改")
self.refresh_user_list()
except Exception as e:
QMessageBox.critical(self, "錯誤", f"修改用戶權限時發生錯誤:{str(e)}")
def show_user_spool_files(self):
username, ok = QInputDialog.getText(self, "查看 Spool Files", "輸入要查看的用戶名:")
if ok and username:
username = username.upper() # 將輸入轉換為大寫
result = self.user_manager.get_user_spool_files(username)
if result:
columns, data = result
dialog = QDialog(self)
dialog.setWindowTitle(f"{username} 的 Spool Files")
dialog.resize(1000, 600) # 設置視窗大小為 1000x600
layout = QVBoxLayout(dialog)
table = QTableWidget()
table.setColumnCount(len(columns) + 1) # 增加一列用於放置"檢視"按鈕
table.setRowCount(len(data))
headers = columns + ["操作"]
table.setHorizontalHeaderLabels(headers)
for row, spool_data in enumerate(data):
for col, value in enumerate(spool_data):
table.setItem(row, col, QTableWidgetItem(str(value)))
view_button = QPushButton("檢視")
view_button.clicked.connect(lambda _, r=row: self.view_spool_file_content(data[r]))
table.setCellWidget(row, len(columns), view_button)
table.resizeColumnsToContents()
layout.addWidget(table)
close_button = QPushButton("關閉")
close_button.clicked.connect(dialog.close)
layout.addWidget(close_button)
dialog.setLayout(layout)
dialog.exec()
else:
QMessageBox.warning(self, "錯誤", f"無法獲取 {username} 的 Spool Files")
def view_spool_file_content(self, spool_data):
spooled_file_name = spool_data[3] # SPOOLED_FILE_NAME 在第4列(索引3)
job_name = spool_data[11] # JOB_NAME 在第12列(索引11)
print(f"Debug: JOB_NAME = {job_name}, SPOOLED_FILE_NAME = {spooled_file_name}") # 添加調試輸出
query = f"""
SELECT * FROM TABLE(SYSTOOLS.SPOOLED_FILE_DATA(
JOB_NAME =>'{job_name}',
SPOOLED_FILE_NAME =>'{spooled_file_name}'))
ORDER BY ORDINAL_POSITION
"""
result = self.user_manager._execute_query(query)
if result:
columns, data = result
if not data:
QMessageBox.warning(self, "警告", "報表內容為空")
return
content_dialog = QDialog(self)
content_dialog.setWindowTitle(f"報表內容: {spooled_file_name}")
content_dialog.resize(1200, 800) # 設置視窗大小為 1200x800
content_layout = QVBoxLayout(content_dialog)
content_table = QTableWidget()
content_table.setColumnCount(len(columns))
content_table.setRowCount(len(data))
content_table.setHorizontalHeaderLabels(columns)
for row, content_data in enumerate(data):
for col, value in enumerate(content_data):
content_table.setItem(row, col, QTableWidgetItem(str(value)))
content_table.resizeColumnsToContents()
content_layout.addWidget(content_table)
close_button = QPushButton("關閉")
close_button.clicked.connect(content_dialog.close)
content_layout.addWidget(close_button)
content_dialog.setLayout(content_layout)
content_dialog.exec()
else:
QMessageBox.warning(self, "錯誤", "無法獲取報表內容")