summaryrefslogtreecommitdiff
path: root/hscript
diff options
context:
space:
mode:
Diffstat (limited to 'hscript')
-rw-r--r--hscript/user.cc50
1 files changed, 48 insertions, 2 deletions
diff --git a/hscript/user.cc b/hscript/user.cc
index 2b89167..21db88a 100644
--- a/hscript/user.cc
+++ b/hscript/user.cc
@@ -10,6 +10,10 @@
* SPDX-License-Identifier: AGPL-3.0-only
*/
+#include <assert.h>
+#include <cstring>
+#include <fstream>
+#include <time.h>
#include "user.hh"
#include "util/output.hh"
@@ -32,6 +36,48 @@ bool RootPassphrase::validate(ScriptOptions) const {
return true;
}
-bool RootPassphrase::execute(ScriptOptions) const {
- return false;
+bool RootPassphrase::execute(ScriptOptions options) const {
+ const std::string root_line = "root:" + this->_value + ":" +
+ std::to_string(time(nullptr) / 86400) + ":0:::::";
+
+ if(options.test(Simulate)) {
+ std::cout << "(printf '" << root_line << "\\"
+ << "n'; cat /target/etc/shadow | sed '1d') > /tmp/shadow"
+ << std::endl
+ << "mv /tmp/shadow /target/etc/shadow" << std::endl
+ << "chown root:shadow /target/etc/shadow" << std::endl
+ << "chmod 640 /target/etc/shadow" << std::endl;
+ return true;
+ }
+
+ std::ifstream old_shadow("/target/etc/shadow");
+ if(!old_shadow) {
+ output_error("installfile:" + std::to_string(this->lineno()),
+ "rootpw: cannot open existing shadow file");
+ return false;
+ }
+
+ std::stringstream shadow_stream;
+ char shadow_line[200];
+ /* Discard root. */
+ old_shadow.getline(shadow_line, sizeof(shadow_line));
+ assert(strncmp(shadow_line, "root", 4) == 0);
+
+ /* Insert the new root line... */
+ shadow_stream << root_line << std::endl;
+ /* ...and copy the rest of the old shadow file. */
+ while(old_shadow.getline(shadow_line, sizeof(shadow_line))) {
+ shadow_stream << shadow_line << std::endl;
+ }
+
+ old_shadow.close();
+
+ std::ofstream new_shadow("/target/etc/shadow", std::ios_base::trunc);
+ if(!new_shadow) {
+ output_error("installfile:" + std::to_string(this->lineno()),
+ "rootpw: cannot replace target shadow file");
+ return false;
+ }
+ new_shadow << shadow_stream.str();
+ return true;
}