Merge pull request #151 from Vanilla-OS/user-creation

User creation
main
Mirko Brombin 1 year ago committed by GitHub
commit e598640849
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

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

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

@ -0,0 +1,150 @@
# 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
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.__on_btn_next_clicked)
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 __on_btn_next_clicked(self, widget):
self.__window.set_user(self.username)
self.__window.next()
def get_finals(self):
return {
"vars": {
"create": True
},
"funcs": [
{
"if": "create",
"type": "command",
"commands": [
f"adduser --quiet --disabled-password --shell /bin/bash --gecos \"{self.fullname}\" {self.username}",
f"echo \"{self.username}:{self.password_entry.get_text()}\" | chpasswd"
]
}
]
}
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="libadwaita" version="1.0" />
<template class="VanillaWindow" parent="AdwApplicationWindow">
<property name="default-width">750</property>
<property name="default-height">640</property>
<property name="default-width">800</property>
<property name="default-height">750</property>
<property name="title">First Setup</property>
<child>
<object class="GtkBox">

@ -20,6 +20,7 @@ conf.set('pkgdatadir', pkgdatadir)
configure_file(
input: 'vanilla-first-setup.in',
output: 'vanilla-first-setup',
install_mode: 'rwxr-xr-x',
configuration: conf,
install: true,
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.welcome import VanillaDefaultWelcome
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.yes_no import VanillaLayoutYesNo
@ -37,6 +38,7 @@ templates = {
"conn-check": VanillaDefaultConnCheck,
"welcome": VanillaDefaultWelcome,
"theme": VanillaDefaultTheme,
"user": VanillaDefaultUser,
"preferences": VanillaLayoutPreferences,
"yes-no": VanillaLayoutYesNo,
"applications": VanillaLayoutApplications

@ -141,7 +141,7 @@ class Processor:
if os.path.exists(autostart_file):
os.remove(autostart_file)
with open(desktop_file, "w") as f:
with open(desktop_file, "w+") as f:
f.write("[Desktop Entry]\n")
f.write("Name=FirstSetup\n")
f.write("Comment=FirstSetup\n")

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

@ -119,12 +119,14 @@ class VanillaProgress(Gtk.Box):
status = not bool(status)
if self.__success_fn is not None and status:
self.__success_fn()
self.__success_fn(*self.__success_fn_args)
self.__window.set_installation_result(status, self.__terminal)
def start(self, setup_commands, success_fn):
def start(self, setup_commands, success_fn, *fn_args):
self.__success_fn = success_fn
self.__success_fn_args = fn_args
self.__terminal.spawn_async(
Vte.PtyFlags.DEFAULT,
".",

Loading…
Cancel
Save