679 lines
30 KiB
Python
679 lines
30 KiB
Python
#!/usr/bin/env python3
|
|
import sys
|
|
import os
|
|
from PyQt5.QtWidgets import (QApplication, QMainWindow, QTabWidget, QTreeWidget, QTreeWidgetItem,
|
|
QVBoxLayout, QHBoxLayout, QWidget, QToolBar, QAction, QMenu,
|
|
QHeaderView, QLabel, QLineEdit, QCheckBox, QMessageBox, QInputDialog)
|
|
from PyQt5.QtGui import QIcon, QColor
|
|
from PyQt5.QtCore import Qt, QTimer, QThread, pyqtSignal
|
|
import subprocess
|
|
import platform
|
|
|
|
class StatusDelegate(QWidget):
|
|
def __init__(self, status, parent=None):
|
|
super().__init__(parent)
|
|
layout = QHBoxLayout(self)
|
|
layout.setContentsMargins(4, 4, 4, 4)
|
|
layout.setSpacing(8)
|
|
|
|
self.status_circle = QWidget()
|
|
self.status_circle.setFixedSize(12, 12)
|
|
color = QColor('green') if 'Up' in status else QColor('red')
|
|
self.status_circle.setStyleSheet(f"background-color: {color.name()}; border-radius: 6px;")
|
|
|
|
self.status_label = QLabel(status)
|
|
|
|
layout.addWidget(self.status_circle)
|
|
layout.addWidget(self.status_label)
|
|
layout.addStretch()
|
|
|
|
class TerminalOpener(QThread):
|
|
error = pyqtSignal(str)
|
|
|
|
def __init__(self, container_id):
|
|
super().__init__()
|
|
self.container_id = container_id
|
|
|
|
def run(self):
|
|
docker_command = f"docker exec -it {self.container_id} sh -c '[ -x /bin/bash ] && exec /bin/bash || exec /bin/sh'"
|
|
system = platform.system()
|
|
|
|
try:
|
|
if system == "Darwin": # macOS
|
|
subprocess.Popen(['open', '-a', 'Terminal', '--', 'sh', '-c', f"{docker_command}"])
|
|
elif system == "Linux":
|
|
# check if inside flatpak
|
|
if 'container' in os.environ:
|
|
subprocess.Popen(['flatpak-spawn', '--host', 'x-terminal-emulator', '-e', f'sh -c "{docker_command}"'])
|
|
else:
|
|
subprocess.Popen(['x-terminal-emulator', '-e', f'sh -c "{docker_command}"'])
|
|
elif system == "Windows":
|
|
subprocess.Popen(['start', 'cmd', '/k', docker_command], shell=True)
|
|
else:
|
|
self.error.emit(f"Opening a terminal is not supported on {system}")
|
|
except Exception as e:
|
|
self.error.emit(f"Failed to open terminal: {str(e)}")
|
|
|
|
class LogsOpener(QThread):
|
|
error = pyqtSignal(str)
|
|
|
|
def __init__(self, container_id):
|
|
super().__init__()
|
|
self.container_id = container_id
|
|
|
|
def run(self):
|
|
docker_command = f"docker logs -f {self.container_id}"
|
|
system = platform.system()
|
|
|
|
try:
|
|
if system == "Darwin": # macOS
|
|
subprocess.Popen(['open', '-a', 'Terminal', '--', 'sh', '-c', f"{docker_command}"])
|
|
elif system == "Linux":
|
|
if 'container' in os.environ:
|
|
subprocess.Popen(['flatpak-spawn', '--host', 'x-terminal-emulator', '-e', f'sh -c "{docker_command}"'])
|
|
else:
|
|
subprocess.Popen(['x-terminal-emulator', '-e', f'sh -c "{docker_command}"'])
|
|
elif system == "Windows":
|
|
subprocess.Popen(['start', 'cmd', '/k', docker_command], shell=True)
|
|
else:
|
|
self.error.emit(f"Opening a terminal is not supported on {system}")
|
|
except Exception as e:
|
|
self.error.emit(f"Failed to open terminal: {str(e)}")
|
|
|
|
class DockerGUI(QMainWindow):
|
|
def __init__(self):
|
|
super().__init__()
|
|
self.setWindowTitle("Qocker - Docker Graphical User Interface")
|
|
self.setGeometry(100, 100, 1000, 600)
|
|
|
|
# Create central widget and layout
|
|
central_widget = QWidget()
|
|
self.setCentralWidget(central_widget)
|
|
main_layout = QVBoxLayout(central_widget)
|
|
|
|
# Create toolbar
|
|
self.create_toolbar()
|
|
|
|
# Create tab widget
|
|
self.tab_widget = QTabWidget()
|
|
main_layout.addWidget(self.tab_widget)
|
|
|
|
# Create tabs
|
|
self.containers_tab = QWidget()
|
|
self.images_tab = QWidget()
|
|
self.networks_tab = QWidget()
|
|
self.volumes_tab = QWidget()
|
|
|
|
self.tab_widget.addTab(self.containers_tab, "Containers")
|
|
self.tab_widget.addTab(self.images_tab, "Images")
|
|
self.tab_widget.addTab(self.networks_tab, "Networks")
|
|
self.tab_widget.addTab(self.volumes_tab, "Volumes")
|
|
|
|
# Connect tab change to toolbar update
|
|
self.tab_widget.currentChanged.connect(self.update_toolbar_buttons)
|
|
|
|
# Create tree widgets for each tab
|
|
self.containers_tree = self.create_tree_widget(["ID", "Name", "Image", "Status", "Ports"])
|
|
self.images_tree = self.create_tree_widget(["ID", "Repository", "Tag", "Size"])
|
|
self.networks_tree = self.create_tree_widget(["ID", "Name", "Driver"])
|
|
self.volumes_tree = self.create_tree_widget(["Name", "Driver", "Mountpoint"])
|
|
|
|
# enable sorting
|
|
self.containers_tree.setSortingEnabled(True)
|
|
self.images_tree.setSortingEnabled(True)
|
|
self.networks_tree.setSortingEnabled(True)
|
|
self.volumes_tree.setSortingEnabled(True)
|
|
|
|
self.containers_tree.itemDoubleClicked.connect(self.open_terminal)
|
|
|
|
# Add tree widgets to tabs
|
|
self.setup_tab(self.containers_tab, self.containers_tree, "Search containers...")
|
|
self.setup_tab(self.images_tab, self.images_tree, "Search images...")
|
|
self.setup_tab(self.networks_tab, self.networks_tree, "Search networks...")
|
|
self.setup_tab(self.volumes_tab, self.volumes_tree, "Search volumes...")
|
|
|
|
# Create menu bar
|
|
self.create_menu_bar()
|
|
|
|
# Setup auto-refresh
|
|
self.setup_auto_refresh()
|
|
|
|
# Populate data
|
|
self.refresh_data()
|
|
|
|
# Update toolbar buttons for initial state
|
|
self.update_toolbar_buttons(0)
|
|
|
|
def create_tree_widget(self, headers):
|
|
tree = QTreeWidget()
|
|
tree.setHeaderLabels(headers)
|
|
tree.setContextMenuPolicy(Qt.CustomContextMenu)
|
|
tree.customContextMenuRequested.connect(self.show_context_menu)
|
|
tree.header().setSectionResizeMode(QHeaderView.Interactive)
|
|
tree.header().setSortIndicator(0, Qt.DescendingOrder)
|
|
tree.header().setSortIndicatorShown(True)
|
|
tree.header().sortIndicatorChanged.connect(lambda col, order: self.sort_tree_widget(tree, col, order))
|
|
return tree
|
|
|
|
def sort_tree_widget(self, tree, column, order):
|
|
tree.sortItems(column, order)
|
|
|
|
def setup_tab(self, tab, tree, search_placeholder):
|
|
layout = QVBoxLayout(tab)
|
|
# Add search bar
|
|
search_bar = QLineEdit()
|
|
search_bar.setPlaceholderText(search_placeholder)
|
|
search_bar.textChanged.connect(lambda text: self.filter_tree(tree, text))
|
|
layout.addWidget(search_bar)
|
|
|
|
layout.addWidget(tree)
|
|
|
|
def filter_tree(self, tree, text):
|
|
for i in range(tree.topLevelItemCount()):
|
|
item = tree.topLevelItem(i)
|
|
match = any(text.lower() in item.text(j).lower() for j in range(item.columnCount()))
|
|
item.setHidden(not match)
|
|
|
|
def create_toolbar(self):
|
|
self.toolbar = QToolBar()
|
|
self.toolbar.setMovable(False) # Make toolbar fixed
|
|
self.addToolBar(Qt.TopToolBarArea, self.toolbar)
|
|
|
|
# Common actions
|
|
self.refresh_action = QAction(QIcon.fromTheme("view-refresh"), "Refresh", self)
|
|
self.refresh_action.triggered.connect(self.refresh_data)
|
|
self.toolbar.addAction(self.refresh_action)
|
|
|
|
# Add auto-refresh checkbox
|
|
self.auto_refresh_checkbox = QCheckBox("Auto-refresh")
|
|
self.auto_refresh_checkbox.setChecked(True)
|
|
self.auto_refresh_checkbox.stateChanged.connect(self.toggle_auto_refresh)
|
|
self.toolbar.addWidget(self.auto_refresh_checkbox)
|
|
|
|
# Add separator
|
|
self.toolbar.addSeparator()
|
|
|
|
# Container-specific actions
|
|
self.start_action = QAction(QIcon.fromTheme("media-playback-start"), "Start", self)
|
|
self.start_action.triggered.connect(self.start_container)
|
|
self.toolbar.addAction(self.start_action)
|
|
|
|
self.stop_action = QAction(QIcon.fromTheme("media-playback-stop"), "Stop", self)
|
|
self.stop_action.triggered.connect(self.stop_container)
|
|
self.toolbar.addAction(self.stop_action)
|
|
|
|
self.remove_action = QAction(QIcon.fromTheme("edit-delete"), "Remove", self)
|
|
self.remove_action.triggered.connect(self.remove_container)
|
|
self.toolbar.addAction(self.remove_action)
|
|
|
|
# Image-specific actions
|
|
self.pull_image_action = QAction(QIcon.fromTheme("download"), "Pull Image", self)
|
|
self.pull_image_action.triggered.connect(self.pull_image)
|
|
self.toolbar.addAction(self.pull_image_action)
|
|
|
|
self.remove_image_action = QAction(QIcon.fromTheme("edit-delete"), "Remove Image", self)
|
|
self.remove_image_action.triggered.connect(self.remove_image)
|
|
self.toolbar.addAction(self.remove_image_action)
|
|
|
|
# Network-specific actions
|
|
self.create_network_action = QAction(QIcon.fromTheme("list-add"), "Create Network", self)
|
|
self.create_network_action.triggered.connect(self.create_network)
|
|
self.toolbar.addAction(self.create_network_action)
|
|
|
|
self.remove_network_action = QAction(QIcon.fromTheme("edit-delete"), "Remove Network", self)
|
|
self.remove_network_action.triggered.connect(self.remove_network)
|
|
self.toolbar.addAction(self.remove_network_action)
|
|
|
|
# Volume-specific actions
|
|
self.create_volume_action = QAction(QIcon.fromTheme("list-add"), "Create Volume", self)
|
|
self.create_volume_action.triggered.connect(self.create_volume)
|
|
self.toolbar.addAction(self.create_volume_action)
|
|
|
|
self.remove_volume_action = QAction(QIcon.fromTheme("edit-delete"), "Remove Volume", self)
|
|
self.remove_volume_action.triggered.connect(self.remove_volume)
|
|
self.toolbar.addAction(self.remove_volume_action)
|
|
|
|
# Add terminal action
|
|
self.terminal_action = QAction(QIcon.fromTheme("utilities-terminal"), "Open Terminal", self)
|
|
self.terminal_action.triggered.connect(self.open_terminal)
|
|
self.toolbar.addAction(self.terminal_action)
|
|
|
|
# Add logs action
|
|
self.logs_action = QAction(QIcon.fromTheme("document-open"), "Open Logs", self)
|
|
self.logs_action.triggered.connect(self.open_logs)
|
|
self.toolbar.addAction(self.logs_action)
|
|
|
|
def update_toolbar_buttons(self, index):
|
|
# Hide all specific actions
|
|
self.start_action.setVisible(False)
|
|
self.stop_action.setVisible(False)
|
|
self.remove_action.setVisible(False)
|
|
self.create_network_action.setVisible(False)
|
|
self.remove_network_action.setVisible(False)
|
|
self.create_volume_action.setVisible(False)
|
|
self.remove_volume_action.setVisible(False)
|
|
self.terminal_action.setVisible(False)
|
|
self.pull_image_action.setVisible(False)
|
|
self.remove_image_action.setVisible(False)
|
|
self.logs_action.setVisible(False)
|
|
|
|
# Show actions based on the current tab
|
|
if index == 0: # Containers tab
|
|
self.start_action.setVisible(True)
|
|
self.stop_action.setVisible(True)
|
|
self.remove_action.setVisible(True)
|
|
self.terminal_action.setVisible(True)
|
|
self.logs_action.setVisible(True)
|
|
elif index == 1: # Images tab
|
|
self.pull_image_action.setVisible(True)
|
|
self.remove_image_action.setVisible(True)
|
|
elif index == 2: # Networks tab
|
|
self.create_network_action.setVisible(True)
|
|
self.remove_network_action.setVisible(True)
|
|
elif index == 3: # Volumes tab
|
|
self.create_volume_action.setVisible(True)
|
|
self.remove_volume_action.setVisible(True)
|
|
|
|
# override QMainWindow.createPopupMenu
|
|
def createPopupMenu(self):
|
|
filtered_menu = super(DockerGUI, self).createPopupMenu()
|
|
filtered_menu.removeAction(self.toolbar.toggleViewAction())
|
|
|
|
return filtered_menu
|
|
|
|
def update_visible_tabs(self, index):
|
|
for i in range(self.tab_widget.count()):
|
|
if i == index:
|
|
self.tab_widget.tabBar().setTabVisible(i, True)
|
|
else:
|
|
self.tab_widget.tabBar().setTabVisible(i, False)
|
|
|
|
def create_menu_bar(self):
|
|
menubar = self.menuBar()
|
|
|
|
file_menu = menubar.addMenu("File")
|
|
exit_action = QAction("Exit", self)
|
|
exit_action.triggered.connect(self.close)
|
|
file_menu.addAction(exit_action)
|
|
|
|
docker_menu = menubar.addMenu("Docker")
|
|
refresh_action = QAction("Refresh", self)
|
|
refresh_action.triggered.connect(self.refresh_data)
|
|
docker_menu.addAction(refresh_action)
|
|
|
|
def show_context_menu(self, position):
|
|
context_menu = QMenu()
|
|
current_tab = self.tab_widget.currentWidget()
|
|
|
|
# Add refresh action to context menu
|
|
refresh_action = QAction("Refresh", self)
|
|
refresh_action.triggered.connect(self.refresh_data)
|
|
context_menu.addAction(refresh_action)
|
|
|
|
context_menu.addSeparator()
|
|
|
|
if current_tab == self.containers_tab:
|
|
terminal_action = QAction("Terminal", self)
|
|
terminal_action.triggered.connect(lambda: self.handle_action("Terminal"))
|
|
logs_action = QAction("Logs", self)
|
|
logs_action.triggered.connect(lambda: self.open_logs())
|
|
start_action = QAction("Start", self)
|
|
start_action.triggered.connect(lambda: self.handle_action("Start"))
|
|
stop_action = QAction("Stop", self)
|
|
stop_action.triggered.connect(lambda: self.handle_action("Stop"))
|
|
remove_action = QAction("Remove", self)
|
|
remove_action.triggered.connect(lambda: self.handle_action("Remove"))
|
|
context_menu.addAction(terminal_action)
|
|
context_menu.addAction(logs_action)
|
|
context_menu.addSeparator()
|
|
context_menu.addAction(start_action)
|
|
context_menu.addAction(stop_action)
|
|
context_menu.addAction(remove_action)
|
|
elif current_tab == self.images_tab:
|
|
pull_action = QAction("Pull", self)
|
|
pull_action.triggered.connect(self.pull_image)
|
|
remove_action = QAction("Remove", self)
|
|
remove_action.triggered.connect(self.remove_image)
|
|
context_menu.addAction(pull_action)
|
|
context_menu.addSeparator()
|
|
context_menu.addAction(remove_action)
|
|
elif current_tab == self.networks_tab:
|
|
remove_action = QAction("Remove", self)
|
|
remove_action.triggered.connect(lambda: self.handle_action("Remove"))
|
|
context_menu.addAction(remove_action)
|
|
elif current_tab == self.volumes_tab:
|
|
remove_action = QAction("Remove", self)
|
|
remove_action.triggered.connect(lambda: self.handle_action("Remove"))
|
|
context_menu.addAction(remove_action)
|
|
|
|
context_menu.exec_(current_tab.mapToGlobal(position))
|
|
|
|
def setup_auto_refresh(self):
|
|
self.refresh_timer = QTimer(self)
|
|
self.refresh_timer.timeout.connect(self.refresh_data)
|
|
if self.auto_refresh_checkbox.isChecked():
|
|
self.refresh_timer.start(1000) # 1000 ms = 1 second
|
|
|
|
def toggle_auto_refresh(self, state):
|
|
if state == Qt.Checked:
|
|
self.refresh_timer.start(1000)
|
|
else:
|
|
self.refresh_timer.stop()
|
|
|
|
def handle_action(self, action):
|
|
current_tab = self.tab_widget.currentWidget()
|
|
selected_items = current_tab.findChild(QTreeWidget).selectedItems()
|
|
|
|
if not selected_items:
|
|
return
|
|
|
|
if current_tab == self.containers_tab:
|
|
container_id = selected_items[0].text(0)
|
|
if action == "Terminal":
|
|
self.open_terminal()
|
|
elif action == "Start":
|
|
subprocess.run(["docker", "start", container_id])
|
|
elif action == "Stop":
|
|
subprocess.run(["docker", "stop", container_id])
|
|
elif action == "Remove":
|
|
subprocess.run(["docker", "rm", "-f", container_id])
|
|
elif current_tab == self.networks_tab:
|
|
network_id = selected_items[0].text(0)
|
|
if action == "Remove":
|
|
subprocess.run(["docker", "network", "rm", network_id])
|
|
elif current_tab == self.volumes_tab:
|
|
volume_name = selected_items[0].text(0)
|
|
if action == "Remove":
|
|
subprocess.run(["docker", "volume", "rm", volume_name])
|
|
|
|
self.refresh_data()
|
|
|
|
def refresh_data(self):
|
|
self.refresh_containers()
|
|
self.refresh_images()
|
|
self.refresh_networks()
|
|
self.refresh_volumes()
|
|
|
|
def refresh_containers(self):
|
|
scroll_position = self.containers_tree.verticalScrollBar().value()
|
|
selected_items = self.get_selected_items(self.containers_tree)
|
|
self.containers_tree.clear()
|
|
try:
|
|
output = subprocess.check_output(["docker", "ps", "-a", "--format", "{{.ID}}\\t{{.Names}}\\t{{.Image}}\\t{{.Status}}\\t{{.Ports}}"], stderr=subprocess.STDOUT)
|
|
if output.strip():
|
|
containers = output.decode().strip().split("\n")
|
|
for container in containers:
|
|
parts = container.split("\t")
|
|
id, name, image, status = parts[:4]
|
|
ports = parts[4] if len(parts) > 4 else ""
|
|
item = QTreeWidgetItem([id, name, image, "", ports]) # Empty string for status column
|
|
status_widget = StatusDelegate(status)
|
|
self.containers_tree.addTopLevelItem(item)
|
|
self.containers_tree.setItemWidget(item, 3, status_widget)
|
|
self.filter_tree(self.containers_tree, self.containers_tab.findChild(QLineEdit).text())
|
|
self.restore_selection(self.containers_tree, selected_items)
|
|
except subprocess.CalledProcessError as e:
|
|
print(f"Error refreshing containers: {e.output.decode()}")
|
|
except ValueError:
|
|
print(f"Error parsing container list {repr(containers)}")
|
|
except Exception as e:
|
|
print(f"Unexpected error refreshing containers: {str(e)}")
|
|
QTimer.singleShot(0, lambda: self.containers_tree.verticalScrollBar().setValue(scroll_position))
|
|
|
|
def refresh_images(self):
|
|
scroll_position = self.images_tree.verticalScrollBar().value()
|
|
selected_items = self.get_selected_items(self.images_tree)
|
|
self.images_tree.clear()
|
|
try:
|
|
output = subprocess.check_output(["docker", "images", "--format", "{{.ID}}\\t{{.Repository}}\\t{{.Tag}}\\t{{.Size}}"], stderr=subprocess.STDOUT)
|
|
if output.strip():
|
|
images = output.decode().strip().split("\n")
|
|
for image in images:
|
|
id, repository, tag, size = image.split("\t")
|
|
item = QTreeWidgetItem([id, repository, tag, size])
|
|
self.images_tree.addTopLevelItem(item)
|
|
|
|
self.filter_tree(self.images_tree, self.images_tab.findChild(QLineEdit).text())
|
|
self.restore_selection(self.images_tree, selected_items)
|
|
except subprocess.CalledProcessError as e:
|
|
print(f"Error refreshing images: {e.output.decode()}")
|
|
except Exception as e:
|
|
print(f"Unexpected error refreshing images: {str(e)}")
|
|
QTimer.singleShot(0, lambda: self.images_tree.verticalScrollBar().setValue(scroll_position))
|
|
|
|
def refresh_networks(self):
|
|
scroll_position = self.networks_tree.verticalScrollBar().value()
|
|
selected_items = self.get_selected_items(self.networks_tree)
|
|
self.networks_tree.clear()
|
|
try:
|
|
output = subprocess.check_output(["docker", "network", "ls", "--format", "{{.ID}}\\t{{.Name}}\\t{{.Driver}}"], stderr=subprocess.STDOUT)
|
|
if output.strip():
|
|
networks = output.decode().strip().split("\n")
|
|
for network in networks:
|
|
id, name, driver = network.split("\t")
|
|
item = QTreeWidgetItem([id, name, driver])
|
|
self.networks_tree.addTopLevelItem(item)
|
|
self.filter_tree(self.networks_tree, self.networks_tab.findChild(QLineEdit).text())
|
|
self.restore_selection(self.networks_tree, selected_items)
|
|
except subprocess.CalledProcessError as e:
|
|
print(f"Error refreshing networks: {e.output.decode()}")
|
|
except Exception as e:
|
|
print(f"Unexpected error refreshing networks: {str(e)}")
|
|
QTimer.singleShot(0, lambda: self.networks_tree.verticalScrollBar().setValue(scroll_position))
|
|
|
|
def refresh_volumes(self):
|
|
scroll_position = self.volumes_tree.verticalScrollBar().value()
|
|
selected_items = self.get_selected_items(self.volumes_tree)
|
|
self.volumes_tree.clear()
|
|
try:
|
|
output = subprocess.check_output(["docker", "volume", "ls", "--format", "{{.Name}}\\t{{.Driver}}\\t{{.Mountpoint}}"], stderr=subprocess.STDOUT)
|
|
if output.strip():
|
|
volumes = output.decode().strip().split("\n")
|
|
for volume in volumes:
|
|
name, driver, mountpoint = volume.split("\t")
|
|
item = QTreeWidgetItem([name, driver, mountpoint])
|
|
self.volumes_tree.addTopLevelItem(item)
|
|
self.filter_tree(self.volumes_tree, self.volumes_tab.findChild(QLineEdit).text())
|
|
self.restore_selection(self.volumes_tree, selected_items)
|
|
except subprocess.CalledProcessError as e:
|
|
print(f"Error refreshing volumes: {e.output.decode()}")
|
|
except Exception as e:
|
|
print(f"Unexpected error refreshing volumes: {str(e)}")
|
|
QTimer.singleShot(0, lambda: self.volumes_tree.verticalScrollBar().setValue(scroll_position))
|
|
|
|
def get_selected_items(self, tree):
|
|
return [item.text(0) for item in tree.selectedItems()]
|
|
|
|
def restore_selection(self, tree, selected_items):
|
|
for i in range(tree.topLevelItemCount()):
|
|
item = tree.topLevelItem(i)
|
|
if item.text(0) in selected_items:
|
|
item.setSelected(True)
|
|
|
|
def start_container(self):
|
|
selected_items = self.containers_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a container to start.")
|
|
return
|
|
|
|
for item in selected_items:
|
|
container_id = item.text(0)
|
|
try:
|
|
subprocess.run(["docker", "start", container_id], check=True)
|
|
print(f"Started container: {container_id}")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to start container {container_id}: {e}")
|
|
|
|
self.refresh_containers()
|
|
|
|
def stop_container(self):
|
|
selected_items = self.containers_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a container to stop.")
|
|
return
|
|
|
|
for item in selected_items:
|
|
container_id = item.text(0)
|
|
try:
|
|
subprocess.run(["docker", "stop", container_id], check=True)
|
|
print(f"Stopped container: {container_id}")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to stop container {container_id}: {e}")
|
|
|
|
self.refresh_containers()
|
|
|
|
def remove_container(self):
|
|
selected_items = self.containers_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a container to remove.")
|
|
return
|
|
|
|
for item in selected_items:
|
|
container_id = item.text(0)
|
|
reply = QMessageBox.question(self, "Confirm Removal",
|
|
f"Are you sure you want to remove container {container_id}?",
|
|
QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
|
|
if reply == QMessageBox.Yes:
|
|
try:
|
|
subprocess.run(["docker", "rm", "-f", container_id], check=True)
|
|
print(f"Removed container: {container_id}")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to remove container {container_id}: {e}")
|
|
|
|
self.refresh_containers()
|
|
|
|
def pull_image(self):
|
|
image_name, ok = QInputDialog.getText(self, "Pull Image", "Enter image name (e.g., ubuntu:latest):")
|
|
if ok and image_name:
|
|
try:
|
|
subprocess.run(["docker", "pull", image_name], check=True)
|
|
print(f"Pulled image: {image_name}")
|
|
QMessageBox.information(self, "Success", f"Image '{image_name}' pulled successfully.")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to pull image: {e}")
|
|
|
|
self.refresh_images()
|
|
|
|
def remove_image(self):
|
|
selected_items = self.images_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select an image to remove.")
|
|
return
|
|
|
|
for item in selected_items:
|
|
image_id = item.text(0)
|
|
reply = QMessageBox.question(self, "Confirm Removal",
|
|
f"Are you sure you want to remove image {image_id}?",
|
|
QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
|
|
if reply == QMessageBox.Yes:
|
|
try:
|
|
subprocess.run(["docker", "rmi", image_id], check=True)
|
|
print(f"Removed image: {image_id}")
|
|
QMessageBox.information(self, "Success", f"Image '{image_id}' removed successfully.")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to remove image {image_id}: {e}")
|
|
|
|
self.refresh_images()
|
|
|
|
def create_network(self):
|
|
network_name, ok = QInputDialog.getText(self, "Create Network", "Enter network name:")
|
|
if ok and network_name:
|
|
try:
|
|
subprocess.run(["docker", "network", "create", network_name], check=True)
|
|
print(f"Created network: {network_name}")
|
|
QMessageBox.information(self, "Success", f"Network '{network_name}' created successfully.")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to create network: {e}")
|
|
|
|
self.refresh_networks()
|
|
|
|
def remove_network(self):
|
|
selected_items = self.networks_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a network to remove.")
|
|
return
|
|
|
|
for item in selected_items:
|
|
network_name = item.text(1) # Assuming the network name is in the second column
|
|
reply = QMessageBox.question(self, "Confirm Removal",
|
|
f"Are you sure you want to remove network {network_name}?",
|
|
QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
|
|
if reply == QMessageBox.Yes:
|
|
try:
|
|
subprocess.run(["docker", "network", "rm", network_name], check=True)
|
|
print(f"Removed network: {network_name}")
|
|
QMessageBox.information(self, "Success", f"Network '{network_name}' removed successfully.")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to remove network {network_name}: {e}")
|
|
|
|
self.refresh_networks()
|
|
|
|
def create_volume(self):
|
|
volume_name, ok = QInputDialog.getText(self, "Create Volume", "Enter volume name:")
|
|
if ok and volume_name:
|
|
try:
|
|
subprocess.run(["docker", "volume", "create", volume_name], check=True)
|
|
print(f"Created volume: {volume_name}")
|
|
QMessageBox.information(self, "Success", f"Volume '{volume_name}' created successfully.")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to create volume: {e}")
|
|
|
|
self.refresh_volumes()
|
|
|
|
def remove_volume(self):
|
|
selected_items = self.volumes_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a volume to remove.")
|
|
return
|
|
|
|
for item in selected_items:
|
|
volume_name = item.text(0) # Assuming the volume name is in the first column
|
|
reply = QMessageBox.question(self, "Confirm Removal",
|
|
f"Are you sure you want to remove volume {volume_name}?",
|
|
QMessageBox.Yes | QMessageBox.No, QMessageBox.No)
|
|
if reply == QMessageBox.Yes:
|
|
try:
|
|
subprocess.run(["docker", "volume", "rm", volume_name], check=True)
|
|
print(f"Removed volume: {volume_name}")
|
|
QMessageBox.information(self, "Success", f"Volume '{volume_name}' removed successfully.")
|
|
except subprocess.CalledProcessError as e:
|
|
QMessageBox.critical(self, "Error", f"Failed to remove volume {volume_name}: {e}")
|
|
|
|
self.refresh_volumes()
|
|
|
|
def open_terminal(self):
|
|
selected_items = self.containers_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a container to open terminal.")
|
|
return
|
|
|
|
container_id = selected_items[0].text(0)
|
|
|
|
self.terminal_opener = TerminalOpener(container_id)
|
|
self.terminal_opener.error.connect(self.show_terminal_error)
|
|
self.terminal_opener.start()
|
|
|
|
def show_terminal_error(self, error_message):
|
|
QMessageBox.critical(self, "Error", error_message)
|
|
|
|
def open_logs(self):
|
|
selected_items = self.containers_tree.selectedItems()
|
|
if not selected_items:
|
|
QMessageBox.warning(self, "No Selection", "Please select a container to open logs.")
|
|
return
|
|
|
|
container_id = selected_items[0].text(0)
|
|
|
|
self.logs_opener = LogsOpener(container_id)
|
|
self.logs_opener.error.connect(self.show_logs_error)
|
|
self.logs_opener.start()
|
|
|
|
def show_logs_error(self, error_message):
|
|
QMessageBox.critical(self, "Error", error_message)
|
|
|
|
if __name__ == "__main__":
|
|
app = QApplication(sys.argv)
|
|
window = DockerGUI()
|
|
window.show()
|
|
sys.exit(app.exec_())
|