User creation

main
Mateus Melchiades 2 years ago
parent 92625f914e
commit e8885a144d

@ -50,6 +50,9 @@
"theme": { "theme": {
"template": "theme" "template": "theme"
}, },
"user": {
"template": "user"
},
"packages": { "packages": {
"template": "preferences", "template": "preferences",
"is-advanced": true, "is-advanced": true,

@ -39,7 +39,7 @@ class VanillaLayoutApplications(Adw.Bin):
# signals # signals
self.btn_next.connect("clicked", self.__next_step) self.btn_next.connect("clicked", self.__next_step)
self.__window.connect("page-changed", self.__on_page_changed) self.__window.connect("page-changed", self.__on_page_changed)
@property @property
def step_id(self): def step_id(self):
return self.__key return self.__key
@ -92,7 +92,7 @@ class VanillaLayoutApplications(Adw.Bin):
def close_customize(widget, dialog): def close_customize(widget, dialog):
dialog.hide() dialog.hide()
def apply_preferences(widget, dialog, apps_list, item): def apply_preferences(widget, dialog, apps_list, item):
for app in item["applications"]: for app in item["applications"]:
app["active"] = app["switch"].get_active() app["active"] = app["switch"].get_active()
@ -148,12 +148,12 @@ class VanillaLayoutApplications(Adw.Bin):
_customize.connect("clicked", present_customize, selection_dialogs[-1], _apps_list, item) _customize.connect("clicked", present_customize, selection_dialogs[-1], _apps_list, item)
_cancel_button.connect("clicked", close_customize, selection_dialogs[-1]) _cancel_button.connect("clicked", close_customize, selection_dialogs[-1])
_apply_button.connect("clicked", apply_preferences, selection_dialogs[-1], _apps_list, item) _apply_button.connect("clicked", apply_preferences, selection_dialogs[-1], _apps_list, item)
self.bundles_list.add(_action_row) self.bundles_list.add(_action_row)
self.__register_widgets.append((item["id"], _switcher, _index)) self.__register_widgets.append((item["id"], _switcher, _index))
_index += 1 _index += 1
def __on_page_changed(self, widget, page): def __on_page_changed(self, widget, page):
if page == self.__key: if page == self.__key:
if True not in [ if True not in [
@ -163,7 +163,7 @@ class VanillaLayoutApplications(Adw.Bin):
self.bundles_list.set_sensitive(False) self.bundles_list.set_sensitive(False)
else: else:
self.bundles_list.set_sensitive(True) self.bundles_list.set_sensitive(True)
def __next_step(self, *args): def __next_step(self, *args):
self.__window.next() self.__window.next()
@ -192,4 +192,4 @@ class VanillaLayoutApplications(Adw.Bin):
for app in self.__step["bundles"][index]["applications"]: for app in self.__step["bundles"][index]["applications"]:
finals["vars"][app["name"]] = False finals["vars"][app["name"]] = False
return finals return finals

@ -5,6 +5,7 @@ sources = [
'__init__.py', '__init__.py',
'welcome.py', 'welcome.py',
'theme.py', 'theme.py',
'user.py',
'applications.py', 'applications.py',
'conn_check.py' 'conn_check.py'
] ]

@ -0,0 +1,146 @@
# user.py
#
# Copyright 2022 mirkobrombin
# Copyright 2022 muqtadir
#
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundationat version 3 of the License.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import sys
import time
import re, subprocess, shutil
import crypt
from gi.repository import Gtk, Gio, GLib, Adw
@Gtk.Template(resource_path='/io/github/vanilla-os/FirstSetup/gtk/default-user.ui')
class VanillaDefaultUser(Adw.Bin):
__gtype_name__ = 'VanillaDefaultUser'
btn_next = Gtk.Template.Child()
fullname_entry = Gtk.Template.Child()
username_entry = Gtk.Template.Child()
password_entry = Gtk.Template.Child()
password_confirmation = Gtk.Template.Child()
fullname = ""
fullname_filled = False
username = ""
username_filled = False
password_filled = False
def __init__(self, window, distro_info, key, step, **kwargs):
super().__init__(**kwargs)
self.__window = window
self.__distro_info = distro_info
self.__key = key
self.__step = step
# signals
self.btn_next.connect("clicked", self.__window.next)
self.fullname_entry.connect('changed', self.__on_fullname_entry_changed)
self.username_entry.connect('changed', self.__on_username_entry_changed)
self.password_entry.connect('changed', self.__on_password_changed)
self.password_confirmation.connect('changed', self.__on_password_changed)
@property
def step_id(self):
return self.__key
def get_finals(self):
return {
"vars": {
"create": True
},
"funcs": [
{
"if": "create",
"type": "command",
"commands": [
f"useradd -m {self.username} -c \"{self.fullname}\" -G sudo,adm,lpadmin -p {crypt.crypt(self.password_entry.get_text())}",
]
}
]
}
def __on_fullname_entry_changed(self, *args):
_fullname = self.fullname_entry.get_text()
if len(_fullname) > 32:
self.fullname_entry.set_text(_fullname[:32])
self.fullname_entry.set_position(-1)
_fullname = self.fullname_entry.get_text()
self.fullname_filled = True
self.__verify_continue()
self.fullname = _fullname
def __on_username_entry_changed(self, *args):
_input = self.username_entry.get_text()
_status = True
# cannot be longer than 32 characters
if len(_input) > 32:
self.username_entry.set_text(_input[:32])
self.username_entry.set_position(-1)
_input = self.username_entry.get_text()
# cannot contain special characters
if re.search(r'[^a-z0-9]', _input):
_status = False
self.__window.toast("Username cannot contain special characters or uppercase letters. Please choose another username.")
# cannot be empty
elif not _input:
_status = False
self.__window.toast("Username cannot be empty. Please type a username.")
# cannot be root
elif _input == "root":
_status = False
self.__window.toast("root user is reserved. Please choose another username.")
if not _status:
self.username_entry.add_css_class('error')
self.username_filled = False
self.__verify_continue()
else:
self.username_entry.remove_css_class('error')
self.username_filled = True
self.__verify_continue()
self.username = _input
def __on_password_changed(self, *args):
password = self.password_entry.get_text()
if password == self.password_confirmation.get_text() \
and password.strip():
self.password_filled = True;
self.password_confirmation.remove_css_class('error')
self.password = self.__encrypt_password(password)
else:
self.password_filled = False;
self.password_confirmation.add_css_class('error')
self.__verify_continue();
def __verify_continue(self):
self.btn_next.set_sensitive(self.fullname_filled and self.password_filled and self.username_filled)
def __encrypt_password(self, password):
command = subprocess.run(
[shutil.which("openssl"), "passwd", "-crypt", password],
capture_output=True
)
password_encrypted = command.stdout.decode('utf-8').strip('\n')
return password_encrypted

@ -0,0 +1,70 @@
<?xml version="1.0" encoding="UTF-8"?>
<interface>
<requires lib="gtk" version="4.0"/>
<template class="VanillaDefaultUser" parent="AdwBin">
<property name="halign">fill</property>
<property name="valign">center</property>
<property name="hexpand">true</property>
<child>
<object class="GtkBox">
<property name="orientation">vertical</property>
<child>
<object class="AdwAvatar">
<property name="valign">center</property>
<property name="size">128</property>
<property name="show-initials">true</property>
<property name="text" bind-source="fullname_entry" bind-property="text" bind-flags="sync-create"/>
</object>
</child>
<child>
<object class="AdwStatusPage" id="status_page">
<property name="title" translatable="true">Create User</property>
<property name="description" translatable="true">Provide details for your user account</property>
<child>
<object class="AdwPreferencesPage">
<child>
<object class="AdwPreferencesGroup">
<child>
<object class="AdwEntryRow" id="fullname_entry">
<property name="title" translatable="true">Name</property>
<property name="input-purpose">name</property>
</object>
</child>
<child>
<object class="AdwEntryRow" id="username_entry">
<property name="title" translatable="true">Username</property>
<property name="input-purpose">name</property>
</object>
</child>
<child>
<object class="AdwPasswordEntryRow" id="password_entry">
<property name="title" translatable="true">Password</property>
<property name="input-purpose">password</property>
</object>
</child>
<child>
<object class="AdwPasswordEntryRow" id="password_confirmation">
<property name="title" translatable="true">Confirm Password</property>
<property name="input-purpose">password</property>
</object>
</child>
</object>
</child>
</object>
</child>
</object>
</child>
<child>
<object class="GtkButton" id="btn_next">
<property name="halign">center</property>
<property name="label" translatable="true">Next</property>
<style>
<class name="pill"/>
<class name="suggested-action"/>
</style>
</object>
</child>
</object>
</child>
</template>
</interface>

@ -3,8 +3,8 @@
<requires lib="gtk" version="4.0"/> <requires lib="gtk" version="4.0"/>
<requires lib="libadwaita" version="1.0" /> <requires lib="libadwaita" version="1.0" />
<template class="VanillaWindow" parent="AdwApplicationWindow"> <template class="VanillaWindow" parent="AdwApplicationWindow">
<property name="default-width">750</property> <property name="default-width">800</property>
<property name="default-height">640</property> <property name="default-height">750</property>
<property name="title">First Setup</property> <property name="title">First Setup</property>
<child> <child>
<object class="GtkBox"> <object class="GtkBox">
@ -45,4 +45,4 @@
</object> </object>
</child> </child>
</template> </template>
</interface> </interface>

@ -20,6 +20,7 @@ conf.set('pkgdatadir', pkgdatadir)
configure_file( configure_file(
input: 'vanilla-first-setup.in', input: 'vanilla-first-setup.in',
output: 'vanilla-first-setup', output: 'vanilla-first-setup',
install_mode: 'rwxr-xr-x',
configuration: conf, configuration: conf,
install: true, install: true,
install_dir: get_option('bindir') install_dir: get_option('bindir')

@ -24,6 +24,7 @@ from vanilla_first_setup.utils.recipe import RecipeLoader
from vanilla_first_setup.defaults.conn_check import VanillaDefaultConnCheck from vanilla_first_setup.defaults.conn_check import VanillaDefaultConnCheck
from vanilla_first_setup.defaults.welcome import VanillaDefaultWelcome from vanilla_first_setup.defaults.welcome import VanillaDefaultWelcome
from vanilla_first_setup.defaults.theme import VanillaDefaultTheme from vanilla_first_setup.defaults.theme import VanillaDefaultTheme
from vanilla_first_setup.defaults.user import VanillaDefaultUser
from vanilla_first_setup.layouts.preferences import VanillaLayoutPreferences from vanilla_first_setup.layouts.preferences import VanillaLayoutPreferences
from vanilla_first_setup.layouts.yes_no import VanillaLayoutYesNo from vanilla_first_setup.layouts.yes_no import VanillaLayoutYesNo
@ -37,6 +38,7 @@ templates = {
"conn-check": VanillaDefaultConnCheck, "conn-check": VanillaDefaultConnCheck,
"welcome": VanillaDefaultWelcome, "welcome": VanillaDefaultWelcome,
"theme": VanillaDefaultTheme, "theme": VanillaDefaultTheme,
"user": VanillaDefaultUser,
"preferences": VanillaLayoutPreferences, "preferences": VanillaLayoutPreferences,
"yes-no": VanillaLayoutYesNo, "yes-no": VanillaLayoutYesNo,
"applications": VanillaLayoutApplications "applications": VanillaLayoutApplications
@ -102,7 +104,7 @@ class Builder:
if step["template"] in templates: if step["template"] in templates:
_widget = templates[step["template"]](self.__window, self.distro_info, key, step) _widget = templates[step["template"]](self.__window, self.distro_info, key, step)
self.__register_widgets.append((_widget, _status, _protected)) self.__register_widgets.append((_widget, _status, _protected))
def get_temp_finals(self, step_id: str): def get_temp_finals(self, step_id: str):
for widget, _, _ in self.__register_widgets: for widget, _, _ in self.__register_widgets:
if widget.step_id == step_id: if widget.step_id == step_id:
@ -125,7 +127,7 @@ class Builder:
@property @property
def recipe(self): def recipe(self):
return self.__recipe.raw return self.__recipe.raw
@property @property
def distro_info(self): def distro_info(self):
return { return {

@ -74,7 +74,7 @@ class Parser:
# check if the condition is met # check if the condition is met
if _condition == _vars[_func["if"]]: if _condition == _vars[_func["if"]]:
commands += _func["commands"] commands += _func["commands"]
# set-up warps if any # set-up warps if any
for warp in warps: for warp in warps:
_vars = warp["vars"] _vars = warp["vars"]

@ -113,7 +113,7 @@ class Processor:
f.write("if [ $? -eq 0 ]; then") f.write("if [ $? -eq 0 ]; then")
f.write(f"{out_run}\n") f.write(f"{out_run}\n")
f.write("fi") f.write("fi")
# create the done file # create the done file
f.write("if [ $? -eq 0 ]; then\n") f.write("if [ $? -eq 0 ]; then\n")
f.write(f"touch {done_file}\n") f.write(f"touch {done_file}\n")

@ -11,6 +11,7 @@
<file>gtk/default-conn-check.ui</file> <file>gtk/default-conn-check.ui</file>
<file>gtk/default-theme.ui</file> <file>gtk/default-theme.ui</file>
<file>gtk/default-welcome.ui</file> <file>gtk/default-welcome.ui</file>
<file>gtk/default-user.ui</file>
<file>gtk/layout-preferences.ui</file> <file>gtk/layout-preferences.ui</file>
<file>gtk/layout-yes-no.ui</file> <file>gtk/layout-yes-no.ui</file>

@ -172,8 +172,8 @@ class VanillaWindow(Adw.ApplicationWindow):
commands commands
) )
self.__view_progress.start(res, Processor.hide_first_setup, self.__user) self.__view_progress.start(res, Processor.hide_first_setup)
def set_installation_result(self, result, terminal): def set_installation_result(self, result, terminal):
self.__view_done.set_result(result, terminal) self.__view_done.set_result(result, terminal)
self.next() self.next()

Loading…
Cancel
Save