/*************************************************************************** * Copyright (C) 2012 by Timothy Pearson * * kb9vqf@pearsoncomputing.net * * * * 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 Foundation; either version 2 of the License, or * * (at your option) any later version. * * * * 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, write to the * * Free Software Foundation, Inc., * * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * ***************************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ldapmgr.h" #include "libtdeldap.h" #include "ldappasswddlg.h" #include "userconfigdlg.h" #include "groupconfigdlg.h" // FIXME // Connect this to CMake/Automake #define KDE_CONFDIR "/etc/trinity" typedef KGenericFactory LDAPConfigFactory; K_EXPORT_COMPONENT_FACTORY( kcm_ldapmanager, LDAPConfigFactory("kcmldapmanager")) LDAPConfig::LDAPConfig(TQWidget *parent, const char *name, const TQStringList&) : KCModule(parent, name), myAboutData(0), m_ldapmanager(0) { m_systemconfig = new KSimpleConfig( TQString::fromLatin1( KDE_CONFDIR "/ldap/ldapconfigrc" )); TQVBoxLayout *layout = new TQVBoxLayout(this, KDialog::marginHint(), KDialog::spacingHint()); base = new LDAPConfigBase(this); layout->add(base); base->user_list->setAllColumnsShowFocus(true); base->user_list->setFullWidth(true); base->group_list->setAllColumnsShowFocus(true); base->group_list->setFullWidth(true); base->group_memberList->setAllColumnsShowFocus(true); base->group_memberList->setFullWidth(true); base->machine_list->setAllColumnsShowFocus(true); base->machine_list->setFullWidth(true); base->user_loginName->setEnabled(false); base->user_uid->setEnabled(false); base->user_primaryGroup->setEnabled(false); base->user_realName->setEnabled(false); base->user_status->setEnabled(false); base->user_secondaryGroups->setEnabled(false); base->machine_name->setEnabled(false); base->machine_author->setEnabled(false); connect(base->user_ldapRealm, TQT_SIGNAL(activated(const TQString&)), this, TQT_SLOT(connectToRealm(const TQString&))); connect(base->group_ldapRealm, TQT_SIGNAL(activated(const TQString&)), this, TQT_SLOT(connectToRealm(const TQString&))); connect(base->machine_ldapRealm, TQT_SIGNAL(activated(const TQString&)), this, TQT_SLOT(connectToRealm(const TQString&))); connect(base->user_list, TQT_SIGNAL(selectionChanged()), this, TQT_SLOT(userHighlighted())); connect(base->group_list, TQT_SIGNAL(selectionChanged()), this, TQT_SLOT(groupHighlighted())); connect(base->machine_list, TQT_SIGNAL(selectionChanged()), this, TQT_SLOT(machineHighlighted())); connect(base->user_buttonAdd, TQT_SIGNAL(clicked()), this, TQT_SLOT(addNewUser())); connect(base->group_buttonAdd, TQT_SIGNAL(clicked()), this, TQT_SLOT(addNewGroup())); connect(base->user_buttonModify, TQT_SIGNAL(clicked()), this, TQT_SLOT(modifySelectedUser())); connect(base->group_buttonModify, TQT_SIGNAL(clicked()), this, TQT_SLOT(modifySelectedGroup())); connect(base->user_buttonDelete, TQT_SIGNAL(clicked()), this, TQT_SLOT(removeSelectedUser())); connect(base->group_buttonDelete, TQT_SIGNAL(clicked()), this, TQT_SLOT(removeSelectedGroup())); connect(base->machine_buttonDelete, TQT_SIGNAL(clicked()), this, TQT_SLOT(removeSelectedMachine())); load(); KAboutData* about = new KAboutData("ldap", I18N_NOOP("TDE LDAP Realm Manager"), "0.1", I18N_NOOP("TDE LDAP Realm Manager Control Panel Module"), KAboutData::License_GPL, I18N_NOOP("(c) 2012 Timothy Pearson"), 0, 0); about->addAuthor("Timothy Pearson", 0, "kb9vqf@pearsoncomputing.net"); setAboutData( about ); processLockouts(); }; LDAPConfig::~LDAPConfig() { delete m_systemconfig; } void LDAPConfig::load() { // Load realms int i; base->user_ldapRealm->clear(); base->group_ldapRealm->clear(); base->machine_ldapRealm->clear(); TQStringList cfgRealms = m_systemconfig->groupList(); for (TQStringList::Iterator it(cfgRealms.begin()); it != cfgRealms.end(); ++it) { if ((*it).startsWith("LDAPRealm-")) { m_systemconfig->setGroup(*it); TQString realmName=*it; realmName.remove(0,strlen("LDAPRealm-")); base->user_ldapRealm->insertItem(realmName); base->group_ldapRealm->insertItem(realmName); base->machine_ldapRealm->insertItem(realmName); } } TQString defaultRealm = m_systemconfig->readEntry("DefaultRealm", TQString::null); if (defaultRealm != "") { for (i=0; iuser_ldapRealm->count(); i++) { if (base->user_ldapRealm->text(i).lower() == defaultRealm.lower()) { base->user_ldapRealm->setCurrentItem(i); base->group_ldapRealm->setCurrentItem(i); base->machine_ldapRealm->setCurrentItem(i); break; } } } connectToRealm(base->user_ldapRealm->currentText().upper()); } void LDAPConfig::defaults() { // } void LDAPConfig::save() { // } void LDAPConfig::processLockouts() { TQListViewItem* lvi = base->user_list->selectedItem(); if (lvi) { base->user_buttonModify->setEnabled(true); base->user_buttonDelete->setEnabled(true); } else { base->user_buttonModify->setEnabled(false); base->user_buttonDelete->setEnabled(false); } base->user_buttonAdd->setEnabled(true); lvi = base->group_list->selectedItem(); if (lvi) { base->group_buttonModify->setEnabled(true); base->group_buttonDelete->setEnabled(true); } else { base->group_buttonModify->setEnabled(false); base->group_buttonDelete->setEnabled(false); } base->group_buttonAdd->setEnabled(true); lvi = base->machine_list->selectedItem(); if (lvi) { base->machine_buttonDelete->setEnabled(true); } else { base->machine_buttonDelete->setEnabled(false); } // FIXME // Disable machine add/modify as they are not implemented // In fact, I don't know if I CAN implement them! base->machine_buttonAdd->setEnabled(false); base->machine_buttonModify->setEnabled(false); } void LDAPConfig::connectToRealm(const TQString& realm) { // Update all drop down lists base->user_ldapRealm->setCurrentItem(realm, false, -1); base->group_ldapRealm->setCurrentItem(realm, false, -1); base->machine_ldapRealm->setCurrentItem(realm, false, -1); if (m_ldapmanager) { if (m_ldapmanager->realm() == realm) { return; } delete m_ldapmanager; } m_systemconfig->setGroup("LDAPRealm-" + realm); TQString host = m_systemconfig->readEntry("admin_server"); m_ldapmanager = new LDAPManager(realm, host); updateAllInformation(); } void LDAPConfig::updateAllInformation() { populateUsers(); populateGroups(); populateMachines(); updateUsersList(); updateGroupsList(); updateMachinesList(); } void LDAPConfig::populateUsers() { m_userInfoList = m_ldapmanager->users(); } void LDAPConfig::populateGroups() { m_groupInfoList = m_ldapmanager->groups(); } void LDAPConfig::populateMachines() { m_machineInfoList = m_ldapmanager->machines(); } void LDAPConfig::updateUsersList() { TQListViewItem* itm = base->user_list->selectedItem(); TQString prevSelectedItemText; if (itm) { prevSelectedItemText = itm->text(0); } base->user_list->clear(); LDAPUserInfoList::Iterator it; for (it = m_userInfoList.begin(); it != m_userInfoList.end(); ++it) { LDAPUserInfo user = *it; itm = new TQListViewItem(base->user_list, user.name, user.commonName, TQString("%1").arg(user.uid)); if (prevSelectedItemText != "") { if (user.name == prevSelectedItemText) { base->user_list->setSelected(itm, true); } } } processLockouts(); } void LDAPConfig::updateGroupsList() { TQListViewItem* itm = base->group_list->selectedItem(); TQString prevSelectedItemText; if (itm) { prevSelectedItemText = itm->text(0); } base->group_list->clear(); LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; itm = new TQListViewItem(base->group_list, group.name, TQString("%1").arg(group.gid)); if (prevSelectedItemText != "") { if (group.name == prevSelectedItemText) { base->group_list->setSelected(itm, true); } } } processLockouts(); } void LDAPConfig::updateMachinesList() { TQListViewItem* itm = base->machine_list->selectedItem(); TQString prevSelectedItemText; if (itm) { prevSelectedItemText = itm->text(0); } base->machine_list->clear(); LDAPMachineInfoList::Iterator it; for (it = m_machineInfoList.begin(); it != m_machineInfoList.end(); ++it) { LDAPMachineInfo machine = *it; itm = new TQListViewItem(base->machine_list, machine.name); if (prevSelectedItemText != "") { if (machine.name == prevSelectedItemText) { base->machine_list->setSelected(itm, true); } } } processLockouts(); } LDAPUserInfo LDAPConfig::findUserInfoByName(TQString name) { // Figure out which user is selected LDAPUserInfoList::Iterator it; for (it = m_userInfoList.begin(); it != m_userInfoList.end(); ++it) { LDAPUserInfo user = *it; if (user.name == name) { return user; } } return LDAPUserInfo(); } LDAPGroupInfo LDAPConfig::findGroupInfoByName(TQString name) { // Figure out which group is selected LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if (group.name == name) { return group; } } return LDAPGroupInfo(); } LDAPMachineInfo LDAPConfig::findMachineInfoByName(TQString name) { // Figure out which machine is selected LDAPMachineInfoList::Iterator it; for (it = m_machineInfoList.begin(); it != m_machineInfoList.end(); ++it) { LDAPMachineInfo machine = *it; if (machine.name == name) { return machine; } } return LDAPMachineInfo(); } LDAPUserInfo LDAPConfig::findUserInfoByNameAndUID(TQString name, TQString uid) { // Figure out which user is selected LDAPUserInfoList::Iterator it; for (it = m_userInfoList.begin(); it != m_userInfoList.end(); ++it) { LDAPUserInfo user = *it; if ((user.name == name) && (TQString("%1").arg(user.uid) == uid)) { return user; } } return LDAPUserInfo(); } LDAPGroupInfo LDAPConfig::findGroupInfoByNameAndGID(TQString name, TQString gid) { // Figure out which group is selected LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if ((group.name == name) && (TQString("%1").arg(group.gid) == gid)) { return group; } } return LDAPGroupInfo(); } LDAPGroupInfo LDAPConfig::findGroupInfoByGID(TQString gid) { // Figure out which group is selected LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if (TQString("%1").arg(group.gid) == gid) { return group; } } return LDAPGroupInfo(); } LDAPUserInfo LDAPConfig::selectedUser() { TQListViewItem* lvi = base->user_list->selectedItem(); if (!lvi) { return LDAPUserInfo(); } return findUserInfoByNameAndUID(lvi->text(0), lvi->text(2)); } LDAPGroupInfo LDAPConfig::selectedGroup() { TQListViewItem* lvi = base->group_list->selectedItem(); if (!lvi) { return LDAPGroupInfo(); } return findGroupInfoByNameAndGID(lvi->text(0), lvi->text(1)); } LDAPMachineInfo LDAPConfig::selectedMachine() { TQListViewItem* lvi = base->machine_list->selectedItem(); if (!lvi) { return LDAPMachineInfo(); } return findMachineInfoByName(lvi->text(0)); } LDAPUserInfo LDAPConfig::findUserByDistinguishedName(TQString dn) { LDAPUserInfoList::Iterator it; for (it = m_userInfoList.begin(); it != m_userInfoList.end(); ++it) { LDAPUserInfo user = *it; if (user.distinguishedName == dn) { return user; } } return LDAPUserInfo(); } LDAPGroupInfoList LDAPConfig::findGroupsForUserByDistinguishedName(TQString dn) { LDAPGroupInfoList groups; LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if (group.userlist.contains(dn)) { groups.append(group); } } return groups; } LDAPUserInfoList LDAPConfig::userList() { return m_userInfoList; } LDAPGroupInfoList LDAPConfig::groupList() { return m_groupInfoList; } void LDAPConfig::userHighlighted() { // Show information in the quick view area LDAPUserInfo user = selectedUser(); base->user_loginName->setText(user.name); base->user_uid->setText(TQString("%1").arg(user.uid)); base->user_primaryGroup->setText(findGroupInfoByGID(TQString("%1").arg(user.primary_gid)).name); base->user_realName->setText(user.commonName); base->user_status->setText((user.status == KRB5_DISABLED_ACCOUNT)?"Disabled":"Enabled"); LDAPGroupInfoList groupsForUser = findGroupsForUserByDistinguishedName(user.distinguishedName); TQString groupsForUserText; LDAPGroupInfoList::Iterator it; for (it = groupsForUser.begin(); it != groupsForUser.end(); ++it) { if (it != groupsForUser.begin()) { groupsForUserText.append(","); } groupsForUserText.append((*it).name); } base->user_secondaryGroups->setText(groupsForUserText); processLockouts(); } void LDAPConfig::groupHighlighted() { // Show information in the quick view area LDAPGroupInfo group = selectedGroup(); base->group_memberList->clear(); for ( TQStringList::Iterator it = group.userlist.begin(); it != group.userlist.end(); ++it ) { LDAPUserInfo user = findUserByDistinguishedName(*it); (void)new TQListViewItem(base->group_memberList, user.name, user.commonName, TQString("%1").arg(user.uid)); } processLockouts(); } void LDAPConfig::machineHighlighted() { // Show information in the quick view area LDAPMachineInfo machine = selectedMachine(); base->machine_name->setText(machine.name); base->machine_author->setText(findUserByDistinguishedName(machine.creatorsName).name); processLockouts(); } void LDAPConfig::addNewUser() { // Launch a dialog to add the user LDAPUserInfo user; // Find the next available, reasonable UID uid_t uid = 100; LDAPUserInfoList::Iterator it; for (it = m_userInfoList.begin(); it != m_userInfoList.end(); ++it) { LDAPUserInfo user = *it; if (user.uid >= uid) { uid = user.uid + 1; } } user.uid = uid; UserConfigDialog userconfigdlg(user, this); if (userconfigdlg.exec() == TQDialog::Accepted) { user = userconfigdlg.m_user; if (user.name != "") { // Try to find a reasonable place to stuff the new entry // Do any users exist right now? if (m_userInfoList.begin() != m_userInfoList.end()) { user.distinguishedName = (*m_userInfoList.begin()).distinguishedName; int eqpos = user.distinguishedName.find("=")+1; int cmpos = user.distinguishedName.find(",", eqpos); user.distinguishedName.remove(eqpos, cmpos-eqpos); user.distinguishedName.insert(eqpos, user.name); } else { user.distinguishedName = "uid=" + user.name + "," + m_ldapmanager->basedn(); } if (m_ldapmanager->addUserInfo(user) == 0) { if (user.new_password != "") { // If a new password was set, use Kerberos to set it on the server TQString errorString; if (setPasswordForUser(user, &errorString) != 0) { KMessageBox::error(0, i18n("Unable to set password for user!

%1").arg(errorString), i18n("Kerberos Failure")); } } // Modify group(s) as needed populateGroups(); LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if (userconfigdlg.selectedGroups.contains(group.name)) { // Make sure that we are in this group! if (!group.userlist.contains(user.distinguishedName)) { group.userlist.append(user.distinguishedName); m_ldapmanager->updateGroupInfo(group); } } else { // Make sure that we are NOT in this group! if (group.userlist.contains(user.distinguishedName)) { group.userlist.remove(user.distinguishedName); m_ldapmanager->updateGroupInfo(group); } } } } } else { // PEBKAC KMessageBox::error(0, i18n("Unable to add new user with no name!

Enter a name and try again"), i18n("Illegal Operation")); } } updateAllInformation(); } void LDAPConfig::addNewGroup() { // Launch a dialog to add the group LDAPGroupInfo group; // Find the next available, reasonable GID gid_t gid = 100; LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if (group.gid >= gid) { gid = group.gid + 1; } } group.gid = gid; GroupConfigDialog groupconfigdlg(group, this); if (groupconfigdlg.exec() == TQDialog::Accepted) { group = groupconfigdlg.m_group; if (group.name != "") { // Try to find a reasonable place to stuff the new entry // Do any groups exist right now? if (m_groupInfoList.begin() != m_groupInfoList.end()) { group.distinguishedName = (*m_groupInfoList.begin()).distinguishedName; int eqpos = group.distinguishedName.find("=")+1; int cmpos = group.distinguishedName.find(",", eqpos); group.distinguishedName.remove(eqpos, cmpos-eqpos); group.distinguishedName.insert(eqpos, group.name); } else { group.distinguishedName = "cn=" + group.name + "," + m_ldapmanager->basedn(); } m_ldapmanager->addGroupInfo(group); } else { // PEBKAC KMessageBox::error(0, i18n("Unable to add new group with no name!

Enter a name and try again"), i18n("Illegal Operation")); } } updateAllInformation(); } void LDAPConfig::modifySelectedUser() { // Launch a dialog to edit the user LDAPUserInfo user = selectedUser(); // Reload user data from LDAP before launching dialog user = m_ldapmanager->getUserByDistinguishedName(user.distinguishedName); UserConfigDialog userconfigdlg(user, this); if (userconfigdlg.exec() == TQDialog::Accepted) { user = userconfigdlg.m_user; if (m_ldapmanager->updateUserInfo(user) == 0) { if (user.new_password != "") { // If a new password was set, use Kerberos to set it on the server TQString errorString; if (setPasswordForUser(user, &errorString) != 0) { KMessageBox::error(0, i18n("Unable to set password for user!

%1").arg(errorString), i18n("Kerberos Failure")); } } // Modify group(s) as needed populateGroups(); LDAPGroupInfoList::Iterator it; for (it = m_groupInfoList.begin(); it != m_groupInfoList.end(); ++it) { LDAPGroupInfo group = *it; if (userconfigdlg.selectedGroups.contains(group.name)) { // Make sure that we are in this group! if (!group.userlist.contains(user.distinguishedName)) { group.userlist.append(user.distinguishedName); m_ldapmanager->updateGroupInfo(group); } } else { // Make sure that we are NOT in this group! if (group.userlist.contains(user.distinguishedName)) { group.userlist.remove(user.distinguishedName); m_ldapmanager->updateGroupInfo(group); } } } } } updateAllInformation(); } void LDAPConfig::modifySelectedGroup() { // Launch a dialog to edit the group LDAPGroupInfo group = selectedGroup(); // Reload group data from LDAP before launching dialog group = m_ldapmanager->getGroupByDistinguishedName(group.distinguishedName); GroupConfigDialog groupconfigdlg(group, this); if (groupconfigdlg.exec() == TQDialog::Accepted) { group = groupconfigdlg.m_group; m_ldapmanager->updateGroupInfo(group); } updateAllInformation(); } void LDAPConfig::removeSelectedUser() { LDAPUserInfo user = selectedUser(); if (KMessageBox::warningYesNo(this, i18n("You are about to delete the user %1
This action cannot be undone

Are you sure you want to proceed?").arg(user.name), i18n("Confirmation Required")) == KMessageBox::Yes) { m_ldapmanager->deleteUserInfo(user); } updateAllInformation(); } void LDAPConfig::removeSelectedGroup() { LDAPGroupInfo group = selectedGroup(); if (KMessageBox::warningYesNo(this, i18n("You are about to delete the group %1
This action cannot be undone

Are you sure you want to proceed?").arg(group.name), i18n("Confirmation Required")) == KMessageBox::Yes) { m_ldapmanager->deleteGroupInfo(group); } updateAllInformation(); } void LDAPConfig::removeSelectedMachine() { LDAPMachineInfo machine = selectedMachine(); if (KMessageBox::warningYesNo(this, i18n("You are about to delete the machine %1
This action cannot be undone

Are you sure you want to proceed?").arg(machine.name), i18n("Confirmation Required")) == KMessageBox::Yes) { m_ldapmanager->deleteMachineInfo(machine); } updateAllInformation(); } TQString readFullLineFromPtyProcess(PtyProcess* proc) { TQString result = ""; while ((!result.contains("\n")) && (!result.contains(":")) && (!result.contains(">"))) { result = result + TQString(proc->readLine(false)); tqApp->processEvents(); } return result; } int LDAPConfig::setPasswordForUser(LDAPUserInfo user, TQString *errstr) { if (user.new_password == "") { return 0; } LDAPCredentials admincreds = m_ldapmanager->currentLDAPCredentials(); TQCString command = "kadmin"; QCStringList args; args << TQCString("-p") << TQCString(admincreds.username.lower()+"@"+(admincreds.realm.upper())) << TQCString("-r") << TQCString(admincreds.realm.upper()); TQString prompt; PtyProcess kadminProc; kadminProc.exec(command, args); prompt = kadminProc.readLine(true); prompt = prompt.stripWhiteSpace(); if (prompt == "kadmin>") { kadminProc.writeLine(TQCString("passwd "+user.name), true); prompt = kadminProc.readLine(true); // Discard our own input prompt = readFullLineFromPtyProcess(&kadminProc); prompt = prompt.stripWhiteSpace(); if ((prompt.endsWith(" Password:")) && (!prompt.startsWith(TQString(user.name + "@")))) { kadminProc.writeLine(admincreds.password, true); prompt = kadminProc.readLine(true); // Discard our own input prompt = kadminProc.readLine(true); prompt = prompt.stripWhiteSpace(); } if (prompt.contains("authentication failed")) { if (errstr) *errstr = prompt; kadminProc.writeLine("quit", true); return 1; } else if ((prompt.endsWith(" Password:")) && (prompt.startsWith(TQString(user.name + "@")))) { kadminProc.writeLine(user.new_password, true); prompt = kadminProc.readLine(true); // Discard our own input prompt = kadminProc.readLine(true); prompt = prompt.stripWhiteSpace(); if ((prompt.endsWith(" Password:")) && (prompt.startsWith("Verify"))) { kadminProc.writeLine(user.new_password, true); prompt = kadminProc.readLine(true); // Discard our own input prompt = kadminProc.readLine(true); prompt = prompt.stripWhiteSpace(); } if ((prompt.endsWith(" Password:")) && (!prompt.startsWith(TQString(user.name + "@")))) { kadminProc.writeLine(admincreds.password, true); prompt = kadminProc.readLine(true); // Discard our own input prompt = kadminProc.readLine(true); prompt = prompt.stripWhiteSpace(); } if (prompt != "kadmin>") { if (errstr) *errstr = prompt; kadminProc.writeLine("quit", true); return 1; } // Success! kadminProc.writeLine("quit", true); return 0; } else if (prompt == "kadmin>") { // Success! kadminProc.writeLine("quit", true); return 0; } // Failure if (errstr) *errstr = prompt; kadminProc.writeLine("quit", true); return 1; } if (errstr) *errstr = "Internal error. Verify that kadmin exists and can be executed."; return 1; // Failure } int LDAPConfig::buttons() { return KCModule::Apply|KCModule::Help; } TQString LDAPConfig::quickHelp() const { return i18n("This module manages users, groups, and machines in LDAP realms."); }