manjaro-hello/src/app-utility

372 lines
14 KiB
Python
Executable File

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# MIT License
#
# Copyright (c) 2018 Fredes Computer Service
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.
import collections
import json
import glob
import os
import subprocess
import sys
import urllib.request
import gi
gi.require_version('Gtk', '3.0')
from gi.repository import Gtk, GLib
VERSION = "0.8"
TITLE = "Manjaro Application Utility {}".format(VERSION)
DEBUG = True
GROUP = 0
ICON = 1
APPLICATION = 2
DESCRIPTION = 3
ACTIVE = 4
PACKAGE = 5
INSTALLED = 6
class AppWindow(Gtk.Window):
def __init__(self):
Gtk.Window.__init__(self, title=TITLE, border_width=6)
self.app = "app-utility"
self.pref = {
"data_set": "default",
"conf_dir": "~/.config",
"share_dir": "/usr/share/{}".format(self.app),
"data_sets": ["default", "advanced"],
"url": "https://gitlab.manjaro.org/fhdk/application-utility/raw/master"
}
self.dev = "--dev" in sys.argv
if self.dev:
self.pref["share_dir"] = "."
self.set_position(Gtk.WindowPosition.CENTER_ALWAYS)
GLib.set_prgname("{}".format(self.app))
icon="system-software-install"
pixbuf24 = Gtk.IconTheme.get_default().load_icon(icon, 24, 0)
pixbuf32 = Gtk.IconTheme.get_default().load_icon(icon, 32, 0)
pixbuf48 = Gtk.IconTheme.get_default().load_icon(icon, 48, 0)
pixbuf64 = Gtk.IconTheme.get_default().load_icon(icon, 64, 0)
pixbuf96 = Gtk.IconTheme.get_default().load_icon(icon, 96, 0)
self.set_icon_list([pixbuf24, pixbuf32, pixbuf48, pixbuf64, pixbuf96])
# set data
self.app_store = None
self.pkg_selected = None
self.pkg_installed = None
self.pkg_list_install = []
self.pkg_list_removal = []
# setup main box
self.set_default_size(800, 650)
self.application_box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
self.add(self.application_box)
# create title box
self.title_box = Gtk.Box()
self.title_image = Gtk.Image()
self.title_image.set_size_request(100, 100)
self.title_image.set_from_file("/usr/share/icons/manjaro/maia/96x96.png")
self.title_label = Gtk.Label()
self.title_label.set_markup("<big>Manjaro Application Maintenance</big>\n"
"Select/Deselect apps you want to install/remove.\n"
"Click <b>UPDATE SYSTEM</b> button when ready.")
self.title_box.pack_start(self.title_image, expand=False, fill=False, padding=0)
self.title_box.pack_start(self.title_label, expand=True, fill=True, padding=0)
# pack title box to main box
self.application_box.pack_start(self.title_box, expand=False, fill=False, padding=0)
# setup grid
self.grid = Gtk.Grid()
self.grid.set_column_homogeneous(True)
self.grid.set_row_homogeneous(True)
self.application_box.add(self.grid)
# setup list store model
self.app_store = self.load_app_data(self.pref["data_set"])
# create a tree view with the model store
self.tree_view = Gtk.TreeView.new_with_model(self.app_store)
self.tree_view.set_activate_on_single_click(True)
# column model: icon
icon = Gtk.CellRendererPixbuf()
column = Gtk.TreeViewColumn("", icon, icon_name=ICON)
self.tree_view.append_column(column)
# column model: group name column
renderer = Gtk.CellRendererText()
column = Gtk.TreeViewColumn("Group", renderer, text=GROUP)
self.tree_view.append_column(column)
# column model: app name column
renderer = Gtk.CellRendererText()
column = Gtk.TreeViewColumn("Application", renderer, text=APPLICATION)
self.tree_view.append_column(column)
# column model: description column
renderer = Gtk.CellRendererText()
column = Gtk.TreeViewColumn("Description", renderer, text=DESCRIPTION)
self.tree_view.append_column(column)
# column model: install column
toggle = Gtk.CellRendererToggle()
toggle.connect("toggled", self.on_app_toggle)
column = Gtk.TreeViewColumn("Installed", toggle, active=ACTIVE)
self.tree_view.append_column(column)
# button box
self.button_box = Gtk.Box(spacing=10)
self.advanced = Gtk.ToggleButton(label="Advanced")
self.advanced.connect("clicked", self.on_expert_clicked)
self.download = Gtk.Button(label="download")
self.download.connect("clicked", self.on_download_clicked)
self.reload_button = Gtk.Button(label="reload")
self.reload_button.connect("clicked", self.on_reload_clicked)
self.update_system_button = Gtk.Button(label="UPDATE SYSTEM")
self.update_system_button.connect("clicked", self.on_update_system_clicked)
self.close_button = Gtk.Button(label="close")
self.close_button.connect("clicked", Gtk.main_quit)
self.button_box.pack_start(self.advanced, expand=False, fill=False, padding=10)
self.button_box.pack_end(self.update_system_button, expand=False, fill=False, padding=10)
self.button_box.pack_end(self.close_button, expand=False, fill=False, padding=10)
self.button_box.pack_end(self.reload_button, expand=False, fill=False, padding=10)
self.button_box.pack_end(self.download, expand=False, fill=False, padding=10)
self.application_box.pack_end(self.button_box, expand=False, fill=False, padding=10)
# create a scrollable window
self.app_window = Gtk.ScrolledWindow()
self.app_window.set_vexpand(True)
self.app_window.add(self.tree_view)
self.grid.attach(self.app_window, 0, 0, 5, len(self.app_store))
# show start
self.show_all()
def load_app_data(self, data_set):
if os.path.isfile("{}/{}.json".format(self.pref["conf_dir"], data_set)):
app_data = self.read_json_file("{}/{}.json".format(self.pref["conf_dir"], data_set))
else:
app_data = self.read_json_file("{}/{}.json".format(self.pref["share_dir"], data_set))
store = Gtk.TreeStore(str, str, str, str, bool, str, bool)
for group in app_data:
index = store.append(None,
[group["name"],
group["icon"],
None, group["description"], None, None, None])
for app in group["apps"]:
status = self.app_installed(app["pkg"])
tree_item = (None,
app["icon"],
app["name"],
app["description"],
status,
app["pkg"],
status)
store.append(index, tree_item)
return store
def reload_app_data(self, dataset):
self.pkg_selected = None
self.pkg_installed = None
self.pkg_list_install = []
self.pkg_list_removal = []
self.app_store.clear()
self.app_store = self.load_app_data(dataset)
self.tree_view.set_model(self.app_store)
def on_reload_clicked(self, widget):
self.reload_app_data(self.pref["data_set"])
def on_expert_clicked(self, widget):
if widget.get_active():
self.pref["data_set"] = "advanced"
else:
self.pref["data_set"] = "default"
self.reload_app_data(self.pref["data_set"])
def on_download_clicked(self, widget):
if self.net_check():
# noinspection PyBroadException
try:
for download in self.pref["data_sets"]:
url = "{}/{}.json".format(self.pref["url"], download)
file = self.fix_path("{}/{}.json".format(self.pref["conf_dir"], download))
req = urllib.request.Request(url=url)
with urllib.request.urlopen(req, timeout=2) as response:
data = json.loads(response.read().decode("utf8"))
self.write_json_file(data, file)
except Exception as e:
print(e)
else:
dialog = Gtk.MessageDialog(self, 0,
Gtk.MessageType.ERROR,
Gtk.ButtonsType.CANCEL,
"Download not available")
dialog.format_secondary.text("The server 'gitlab.manjaro.org' could not be reached")
dialog.run()
def on_app_toggle(self, cell, path):
# a group has no package attached and we don't install groups
if self.app_store[path][PACKAGE] is not None:
self.app_store[path][ACTIVE] = not self.app_store[path][ACTIVE]
self.pkg_selected = self.app_store[path][PACKAGE]
self.pkg_installed = self.app_store[path][INSTALLED]
if self.app_store[path][ACTIVE] is False:
if self.pkg_installed is True:
# to uninstall
self.pkg_list_removal.append(self.pkg_selected)
if self.dev:
print("for removal : {}".format(self.pkg_selected))
if self.pkg_selected in self.pkg_list_install:
# cancel install
self.pkg_list_install.remove(self.pkg_selected)
if self.dev:
print("cancel install: {}".format(self.pkg_selected))
else:
# don't reinstall
if self.pkg_installed is False:
# only install
if self.pkg_selected not in self.pkg_list_install:
self.pkg_list_install.append(self.pkg_selected)
if self.dev:
print("to install : {}".format(self.pkg_selected))
if self.pkg_selected in self.pkg_list_removal:
# cancel uninstall
self.pkg_list_removal.remove(self.pkg_selected)
if self.dev:
print("pkg list install: {}".format(self.pkg_list_install))
print("pkg list removal: {}".format(self.pkg_list_removal))
def on_update_system_clicked(self, widget):
file_install = "/tmp/.install-packages.txt"
file_uninstall = "/tmp/.remove-packages.txt"
os.environ["APP_UTILITY"] = "PACKAGES"
shell_fallback = False
if self.pkg_list_install:
if os.path.isfile("/usr/bin/pamac-installer"):
subprocess.run(["pamac-installer"] + self.pkg_list_install)
else:
shell_fallback = True
with open(file_install, "w") as outfile:
for p in self.pkg_list_install:
outfile.write("{} ".format(p))
if self.pkg_list_removal:
shell_fallback = True
with open(file_uninstall, "w") as outfile:
for p in self.pkg_list_removal:
outfile.write("{} ".format(p))
if shell_fallback:
if self.dev:
os.system('gksu-polkit ./app-install')
else:
os.system('gksu-polkit app-install')
self.reload_app_data(self.pref["data_set"])
@staticmethod
def app_installed(package):
if glob.glob("/var/lib/pacman/local/{}-[0-9]*".format(package)):
return True
return False
@staticmethod
def fix_path(path):
"""Make good paths.
:param path: path to fix
:type path: str
:return: fixed path
:rtype: str
"""
if "~" in path:
path = path.replace("~", os.path.expanduser("~"))
return path
@staticmethod
def net_check():
"""Check for internet connection"""
resp = None
host = "https://gitlab.manjaro.org"
# noinspection PyBroadException
try:
resp = urllib.request.urlopen(host, timeout=2)
except Exception:
pass
return bool(resp)
@staticmethod
def read_json_file(filename, dictionary=True):
"""Read json data from file"""
result = list()
try:
if dictionary:
with open(filename, "rb") as infile:
result = json.loads(
infile.read().decode("utf8"),
object_pairs_hook=collections.OrderedDict)
else:
with open(filename, "r") as infile:
result = json.load(infile)
except OSError:
pass
return result
@staticmethod
def write_json_file(data, filename, dictionary=False):
"""Writes data to file as json
:param data
:param filename:
:param dictionary:
"""
try:
if dictionary:
with open(filename, "wb") as outfile:
json.dump(data, outfile)
else:
with open(filename, "w") as outfile:
json.dump(data, outfile, indent=2)
return True
except OSError:
return False
if __name__ == "__main__":
win = AppWindow()
win.connect("delete-event", Gtk.main_quit)
win.connect("destroy", Gtk.main_quit)
win.show_all()
Gtk.main()