mirror of
https://github.com/outbackdingo/sysadm.git
synced 2026-01-27 10:20:26 +00:00
Add a new action to the sysadm/users class: "groupmod"
This action allows for modifying a given group on the system
REQUIRED: "name"="<name of group to modify>"
and any one of these options is also required:
"users":["array of users"] (will set the list of users for this group)
"add_users":["array of users"] (will add the listed users to the current users)
"remove_users":["array of users"] (will remove the listed users from the current users)
Example API Request (JSON)
{
"id":"sample",
"namespace":"sysadm",
"name":"users",
"args":{
"action":"groupmod",
"name":"operator",
"users":["user1","user2"]
}
}
370 lines
17 KiB
C++
370 lines
17 KiB
C++
//===========================================
|
|
// TrueOS source code
|
|
// Copyright (c) 2016, TrueOS Software/iXsystems
|
|
// Available under the 3-clause BSD license
|
|
// See the LICENSE file for full details
|
|
//===========================================
|
|
#include "sysadm-users.h"
|
|
#include "sysadm-general.h"
|
|
#include "sysadm-global.h"
|
|
#include "globals.h"
|
|
|
|
using namespace sysadm;
|
|
|
|
bool UserManager::listUsers(QJsonObject *out, bool showall, QString user){
|
|
QStringList args; args << "usershow";
|
|
if(showall){ args << "-a"; }
|
|
else{ args << user; }
|
|
bool ok = false;
|
|
QStringList users = sysadm::General::RunCommand(ok, "pw", args, "",QStringList() << "MM_CHARSET=UTF-8").split("\n");
|
|
if(ok){
|
|
//Get a list of all current PersonaCrypt users (if any)
|
|
bool PCinstalled = QFile::exists("/var/db/personacrypt");
|
|
//Go ahead and parse/list all the users
|
|
for(int i=0; i<users.length(); i++){
|
|
QStringList info = users[i].split(":");
|
|
if(info.length() == 10){
|
|
QJsonObject uinfo;
|
|
uinfo.insert("name", info[0]);
|
|
//uinfo.insert("name", info[1]); //Skip Password field (just a "*" in this viewer anyway)
|
|
uinfo.insert("uid", info[2]);
|
|
uinfo.insert("gid", info[3]);
|
|
uinfo.insert("class", info[4]);
|
|
uinfo.insert("change", info[5]);
|
|
uinfo.insert("expire", info[6]);
|
|
uinfo.insert("comment", info[7]);
|
|
uinfo.insert("home_dir", info[8]);
|
|
uinfo.insert("shell", info[9]);
|
|
if(PCinstalled && QFile::exists("/var/db/personacrypt/" + info[0] + ".key") ){ uinfo.insert("personacrypt_enabled","true"); }
|
|
if(info[0]==user){ uinfo.insert("canremove","false"); } //current user requesting info - cannot remove it
|
|
out->insert(info[0], uinfo); //use the username as the unique object name
|
|
}else if(info.length() == 7){
|
|
QJsonObject uinfo;
|
|
uinfo.insert("name", info[0]);
|
|
//uinfo.insert("name", info[1]); //Skip Password field (just a "*" in this viewer anyway)
|
|
uinfo.insert("uid", info[2]);
|
|
uinfo.insert("gid", info[3]);
|
|
uinfo.insert("comment", info[4]);
|
|
uinfo.insert("home_dir", info[5]);
|
|
uinfo.insert("shell", info[6]);
|
|
if(PCinstalled && QFile::exists("/var/db/personacrypt/" + info[0] + ".key") ){ uinfo.insert("personacrypt_enabled","true"); }
|
|
out->insert(info[0], uinfo); //use the username as the unique object name
|
|
}
|
|
}
|
|
}else{
|
|
//Bad result from "pw" - inputs were just fine (just return nothing)
|
|
ok = true;
|
|
}
|
|
if(!ok){ out->insert("error",users.join("\n")); } //should never happen - "pw" is an OS built-in on FreeBSD
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::addUser(QJsonObject* out, QJsonObject obj){
|
|
bool ok = false;
|
|
//REQUIRED: "name" AND "password"
|
|
//OPTIONAL: "uid", "comment", "home_dir", "expire", "change", "shell", "group", "other_groups", "class"
|
|
//OPTIONAL: "personacrypt_init"=<devicename>, "personacrypt_password"
|
|
//OPTIONAL: "personacrypt_import"=<contents of key file - base64 encoded>
|
|
if(obj.contains("password") && obj.contains("name") ){
|
|
QString username = obj.value("name").toString();
|
|
QStringList args; args << "useradd";
|
|
args << "-n" << username;
|
|
if(obj.contains("uid")){ args << "-u" << obj.value("uid").toString(); }
|
|
if(obj.contains("comment")){ args << "-c" << obj.value("comment").toString(); }
|
|
if(obj.contains("home_dir")){ args << "-d" << obj.value("home_dir").toString(); }
|
|
if(obj.contains("expire")){ args << "-e" << obj.value("expire").toString(); }
|
|
if(obj.contains("change")){ args << "-p" << obj.value("change").toString(); }
|
|
if(obj.contains("shell")){ args << "-s" << obj.value("shell").toString(); }
|
|
if(obj.contains("group")){ args << "-g" << obj.value("group").toString(); }
|
|
if(obj.contains("other_groups")){
|
|
if(obj.value("other_groups").isString()){ args << "-G" << obj.value("other_groups").toString(); }
|
|
else if(obj.value("other_groups").isArray()){ args << "-G" << General::JsonArrayToStringList(obj.value("other_groups").toArray()).join(","); }
|
|
}
|
|
if(obj.contains("class")){ args << "-L" << obj.value("class").toString(); }
|
|
//See if PersonaCrypt should be used on this user as well
|
|
QString PCdev;
|
|
if(obj.contains("personacrypt_init")){
|
|
QString dev = obj.value("personacrypt_init").toString();
|
|
if(dev.startsWith("/dev/")){ dev = dev.remove(0,5); }
|
|
//Verify that the given device is valid
|
|
QStringList valid = getAvailablePersonaCryptDevices();
|
|
for(int i=0; i<valid.length(); i++){
|
|
if(valid[i].startsWith(dev+": ")){ PCdev = dev; } //this is a valid device
|
|
}
|
|
if(PCdev!=dev){ return false; } //invalid inputs - invalid device for PersonaCrypt
|
|
}
|
|
QTemporaryFile pwfile("/tmp/.XXXXXXXXXXXXXXX");
|
|
bool usercreated = false;
|
|
if(pwfile.open()){
|
|
pwfile.write( obj.value("password").toString().toUtf8().data() );
|
|
pwfile.close(); //closed but still exists - will go out of scope and get removed in a moment
|
|
args << "-h" << "0"; //read from std input
|
|
args << "-m"; //automatically create users home dir
|
|
ok = (0== system("cat "+pwfile.fileName().toUtf8()+" | pw "+args.join(" ").toUtf8()) );
|
|
usercreated = ok;
|
|
}else{
|
|
out->insert("error","Could not open temporary file for password"); //should never happen
|
|
}
|
|
|
|
if(usercreated && !PCdev.isEmpty()){
|
|
// INIT PERSONACRYPT DEVICE NOW
|
|
//User created fine - go ahead and initialize the PersonaCrypt device for this user now
|
|
QString pass = obj.value("password").toString(); //assume the same password as the user unless specified otherwise
|
|
if(obj.contains("personacrypt_password")){ pass = obj.value("personacrypt_password").toString(); } //separate password for device
|
|
ok = InitializePersonaCryptDevice(username, pass, PCdev);
|
|
if(!ok){ out->insert("error","could not initialize personacrypt device"); }
|
|
}else if(usercreated && obj.contains("personacrypt_import")){
|
|
// IMPORT PERSONACRYPT KEY NOW
|
|
ok = false;
|
|
//Need to save key to a local file temporarily
|
|
QTemporaryFile keyfile("/tmp/.XXXXXXXXXXXXXXX");
|
|
if(keyfile.open()){
|
|
keyfile.write( QByteArray::fromBase64(obj.value("personacrypt_import").toString().toUtf8()).data() );
|
|
keyfile.close();
|
|
ok = ImportPersonaCryptKey(keyfile.fileName());
|
|
}
|
|
if(!ok){ out->insert("error","could not import personacrypt key"); }
|
|
}
|
|
if(usercreated && !ok){
|
|
//One of the personacrypt options failed - treat this as a full failure and also remove the newly-created user
|
|
removeUser(username);
|
|
}
|
|
} //end check for valid inputs
|
|
if(ok){ out->insert("result","success"); }
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::removeUser(QString username, bool deletehomedir){
|
|
bool ok = false;
|
|
QStringList args;
|
|
args << "userdel" << "-n" << username;
|
|
if(deletehomedir){ args << "-r"; }
|
|
ok = General::RunQuickCommand("pw", args);
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::modifyUser(QJsonObject* out, QJsonObject obj){
|
|
bool ok = false;
|
|
// REQUIRED: "name"
|
|
//OPTIONAL: "newname", "uid", "comment", "home_dir", "expire", "change", "shell", "group", "other_groups", "class"
|
|
//OPTIONAL: "personacrypt_init"=<devicename>, "personacrypt_password"
|
|
//OPTIONAL: "personacrypt_import"=<contents of key file - base64 encoded>
|
|
if(obj.contains("name") ){
|
|
QString username = obj.value("name").toString();
|
|
//See if PersonaCrypt should be used on this user as well
|
|
QString PCdev;
|
|
if(obj.contains("personacrypt_init")){
|
|
QString dev = obj.value("personacrypt_init").toString();
|
|
if(dev.startsWith("/dev/")){ dev = dev.remove(0,5); }
|
|
//Verify that the given device is valid
|
|
QStringList valid = getAvailablePersonaCryptDevices();
|
|
for(int i=0; i<valid.length(); i++){
|
|
if(valid[i].startsWith(dev+": ")){ PCdev = dev; } //this is a valid device
|
|
}
|
|
if(PCdev!=dev){ return false; } //invalid inputs - invalid device for PersonaCrypt
|
|
}
|
|
//Now start assembling the external command
|
|
QStringList args; args << "usermod";
|
|
args << "-n" << username;
|
|
if(obj.contains("uid")){ args << "-u" << obj.value("uid").toString(); }
|
|
if(obj.contains("comment")){ args << "-c" << obj.value("comment").toString(); }
|
|
if(obj.contains("home_dir")){ args << "-d" << obj.value("home_dir").toString(); }
|
|
if(obj.contains("expire")){ args << "-e" << obj.value("expire").toString(); }
|
|
if(obj.contains("change")){ args << "-p" << obj.value("change").toString(); }
|
|
if(obj.contains("shell")){ args << "-s" << obj.value("shell").toString(); }
|
|
if(obj.contains("group")){ args << "-g" << obj.value("group").toString(); }
|
|
if(obj.contains("other_groups")){
|
|
if(obj.value("other_groups").isString()){ args << "-G" << obj.value("other_groups").toString(); }
|
|
else if(obj.value("other_groups").isArray()){ args << "-G" << General::JsonArrayToStringList(obj.value("other_groups").toArray()).join(","); }
|
|
}
|
|
if(obj.contains("class")){ args << "-L" << obj.value("class").toString(); }
|
|
if(obj.contains("newname")){ args << "-l" << obj.value("newname").toString(); }
|
|
if(obj.contains("password")){
|
|
//changing password too - need to handle differently
|
|
QTemporaryFile pwfile("/tmp/.XXXXXXXXXXXXXXX");
|
|
bool usercreated = false;
|
|
if(pwfile.open()){
|
|
pwfile.write( obj.value("password").toString().toUtf8().data() );
|
|
pwfile.close(); //closed but still exists - will go out of scope and get removed in a moment
|
|
args << "-h" << "0"; //read from std input
|
|
ok = (0== system("cat "+pwfile.fileName().toUtf8()+" | pw "+args.join(" ").toUtf8()) );
|
|
}else{
|
|
out->insert("error","Could not open temporary file for password"); //should never happen
|
|
}
|
|
}else if(args.length()>3){
|
|
//No password change - simple command
|
|
ok = General::RunQuickCommand("pw", args);
|
|
}else if( !obj.keys().filter("personacrypt_").isEmpty() ){
|
|
ok = true; //no user changes aside from PersonaCrypt changes
|
|
}
|
|
//PERSONACRYPT CHANGES
|
|
if(ok && !PCdev.isEmpty()){
|
|
// INIT PERSONACRYPT DEVICE NOW
|
|
//User created fine - go ahead and initialize the PersonaCrypt device for this user now
|
|
QString pass = obj.value("password").toString(); //assume the same password as the user unless specified otherwise
|
|
if(obj.contains("personacrypt_password")){ pass = obj.value("personacrypt_password").toString(); } //separate password for device
|
|
ok = InitializePersonaCryptDevice(username, pass, PCdev);
|
|
if(!ok){ out->insert("error","could not initialize personacrypt device"); }
|
|
}else if(ok && obj.contains("personacrypt_import")){
|
|
// IMPORT PERSONACRYPT KEY NOW
|
|
ok = false;
|
|
//Need to save key to a local file temporarily
|
|
QTemporaryFile keyfile("/tmp/.XXXXXXXXXXXXXXX");
|
|
if(keyfile.open()){
|
|
keyfile.write( QByteArray::fromBase64(obj.value("personacrypt_import").toString().toUtf8()).data() );
|
|
keyfile.close();
|
|
ok = ImportPersonaCryptKey(keyfile.fileName());
|
|
}
|
|
if(!ok){ out->insert("error","could not import personacrypt key"); }
|
|
}else if(ok && obj.contains("personacrypt_disable")){
|
|
QString pass = obj.value("personacrypt_disable").toString();
|
|
ok = DisablePersonaCryptKey(username, pass);
|
|
}
|
|
}
|
|
if(ok){ out->insert("result","success"); }
|
|
else{ out->insert("error","Could not modify user"); }
|
|
return ok;
|
|
}
|
|
|
|
// === GROUP MANAGEMENT ===
|
|
bool UserManager::listGroups(QJsonObject* out, QString user ){
|
|
bool ok = false;
|
|
QStringList info = General::RunCommand(ok, "pw", QStringList() << "groupshow" << "-a", "",QStringList() << "MM_CHARSET=UTF-8").split("\n");
|
|
if(ok){
|
|
for(int i=0; i<info.length(); i++){
|
|
QStringList ginfo = info[i].split(":");
|
|
if(ginfo.length()<4){ continue; } //invalid line
|
|
QJsonObject obj;
|
|
obj.insert("name",ginfo[0]);
|
|
obj.insert("gid",ginfo[2]);
|
|
QStringList users = ginfo[3].split(",");
|
|
//If restricted to a particular user, only show that user in the users list as needed (don't show other users)
|
|
if(user.isEmpty()){obj.insert("users", QJsonArray::fromStringList( users ) ); }
|
|
else if(users.contains(user)){ obj.insert("users", QJsonArray::fromStringList( QStringList() << user ) ); }
|
|
out->insert(ginfo[0], obj);
|
|
}
|
|
}
|
|
if(!ok){ out->insert("error",info.join("\n")); }
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::addGroup(QJsonObject* out, QJsonObject input){
|
|
bool ok = false;
|
|
// REQUIRED: "name"
|
|
// OPTIONAL: "gid", "users"
|
|
if(input.contains("name")){
|
|
QStringList args;
|
|
args << "groupadd" << "-n" << input.value("name").toString();
|
|
if(input.contains("gid")){ args << "-g" << input.value("gid").toString(); }
|
|
if(input.contains("users")){
|
|
if(input.value("users").isArray()){ args << "-M" << General::JsonArrayToStringList(input.value("users").toArray()).join(","); }
|
|
else if(input.value("users").isString()){ args << "-M" << input.value("users").toString(); }
|
|
}
|
|
QString res = General::RunCommand(ok, "pw", args, "",QStringList() << "MM_CHARSET=UTF-8");
|
|
if(ok){ out->insert("result","success"); }
|
|
else{ out->insert("error", res); }
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::removeGroup(QString name){
|
|
bool ok = false;
|
|
QStringList args; args << "groupdel" << "-n" << name;
|
|
QString res = General::RunCommand(ok, "pw", args);
|
|
if(!ok){ qDebug() << "[ERROR] Could not delete group:"<< name << "\n - Result Message:" << res; }
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::modifyGroup(QJsonObject* out, QJsonObject input){
|
|
bool ok = false;
|
|
// REQUIRED: "name"
|
|
// OPTIONAL: "users", "add_users", "remove_users" (only one of these at a time) - all a string or array of strings
|
|
if(input.contains("name")){
|
|
QStringList args;
|
|
QString group = input.value("name").toString();
|
|
if(input.contains("users")){
|
|
QStringList users;
|
|
ok = true; //got valid inputs
|
|
if(input.value("users").isString()){ users << input.value("users").toString(); } //only one user
|
|
else if(input.value("users").isArray()){ users = General::JsonArrayToStringList(input.value("users").toArray()); }
|
|
else{ ok = false; } //actually invalid - could not read users
|
|
if(ok){
|
|
args << "groupmod" << "-n" << group << "-M" << users.join(",");
|
|
}
|
|
}else if(input.contains("add_users")){
|
|
QStringList users;
|
|
ok = true; //got valid inputs
|
|
if(input.value("add_users").isString()){ users << input.value("add_users").toString(); } //only one user
|
|
else if(input.value("add_users").isArray()){ users =General:: JsonArrayToStringList(input.value("add_users").toArray()); }
|
|
else{ ok = false; } //actually invalid - could not read users
|
|
if(ok){
|
|
args << "groupmod" << "-n" << group << "-m" << users.join(",");
|
|
}
|
|
}else if(input.contains("remove_users")){
|
|
QStringList users;
|
|
ok = true; //got valid inputs
|
|
if(input.value("remove_users").isString()){ users << input.value("remove_users").toString(); } //only one user
|
|
else if(input.value("remove_users").isArray()){ users = General::JsonArrayToStringList(input.value("remove_users").toArray()); }
|
|
else{ ok = false; } //actually invalid - could not read users
|
|
if(ok){
|
|
args << "groupmod" << "-n" << group << "-d" << users.join(",");
|
|
}
|
|
}
|
|
if(!args.isEmpty()){
|
|
QString res = General::RunCommand(ok, "pw", args);
|
|
if(!ok){ out->insert("error",res); }
|
|
else{ out->insert("result","success"); }
|
|
}else{
|
|
ok = false; //nothing to do
|
|
}
|
|
}
|
|
|
|
return ok;
|
|
}
|
|
|
|
// === PERSONACRYPT FUNCTIONS ===
|
|
//List all the devices currently available to be used for a PersonaCrypt User
|
|
QStringList UserManager::getAvailablePersonaCryptDevices(){
|
|
QStringList devs = General::RunCommand("personacrypt", QStringList() << "list" << "-r").split("\n");
|
|
for(int i=0; i<devs.length(); i++){
|
|
//Check validity of each line really quick (no gpart errors output and such)
|
|
if(devs[i].isEmpty() || devs[i].startsWith("gpart")){ devs.removeAt(i); i--; }
|
|
}
|
|
return devs;
|
|
}
|
|
|
|
//PersonaCrypt Modification functions
|
|
bool UserManager::InitializePersonaCryptDevice(QString username, QString pass, QString device){
|
|
QTemporaryFile pfile("/tmp/.XXXXXXXXXXXXXXX");
|
|
bool ok = false;
|
|
if( pfile.open() ){
|
|
pfile.write(pass.toUtf8().data());
|
|
pfile.close();
|
|
QString result = General::RunCommand(ok, "personacrypt", QStringList() << "init" << username << pfile.fileName() << device);
|
|
//qDebug() << "PC init result:" << result;
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::ImportPersonaCryptKey(QString keyfile){
|
|
if(!QFile::exists(keyfile)){ return false; } //quick check first
|
|
bool ok = General::RunQuickCommand("personacrypt", QStringList() << "import" << keyfile);
|
|
return ok;
|
|
}
|
|
|
|
bool UserManager::DisablePersonaCryptKey(QString username, QString pass){
|
|
bool ok = false;
|
|
if(pass.isEmpty()){
|
|
General::RunCommand(ok, "personacrypt", QStringList() << "remove" << username);
|
|
}else{
|
|
QTemporaryFile pfile("/tmp/.XXXXXXXXXXXXXXX");
|
|
if( pfile.open() ){
|
|
pfile.write(pass.toUtf8().data());
|
|
pfile.close();
|
|
QString result = General::RunCommand(ok, "personacrypt", QStringList() << "remove" << username << pfile.fileName());
|
|
//qDebug() << "PC init result:" << result;
|
|
}
|
|
}
|
|
return ok;
|
|
}
|