diff options
Diffstat (limited to 'hscript')
-rw-r--r-- | hscript/network.cc | 69 |
1 files changed, 67 insertions, 2 deletions
diff --git a/hscript/network.cc b/hscript/network.cc index 224e979..5808730 100644 --- a/hscript/network.cc +++ b/hscript/network.cc @@ -20,6 +20,7 @@ # include <string.h> /* strerror */ # include <sys/ioctl.h> /* ioctl, ioctl numbers */ # include <unistd.h> /* close */ +# include "util/filesystem.hh" /* fs for pppoe eni */ #else /*! The size of Linux interface names. */ # define IFNAMSIZ 16 @@ -509,8 +510,72 @@ bool execute_pppoe_netifrc(const PPPoE &link) { return true; } -bool execute_pppoe_eni(const PPPoE &link) { +bool execute_pppoe_eni(const PPPoE &link, const Horizon::Script *script) { +#ifndef HAS_INSTALL_ENV + output_error(link.where(), "pppoe: ENI cannot be simulated"); return false; +#else + const auto ¶ms = link.params(); + const std::string &pppdir{script->targetDirectory() + "/etc/ppp"}; + const std::string &linkiface{"ppp" + std::to_string(ppp_link_count)}; + error_code ec; + + fs::create_directories(pppdir + "/peers", ec); + if(ec && ec.value() != EEXIST) { + output_error(link.where(), "pppoe: cannot create PPP directory", + ec.message()); + return false; + } + + std::ofstream config("/tmp/horizon/eni/" + link.iface(), + std::ios_base::trunc); + if(!config) { + output_error(link.where(), "pppoe: couldn't write network " + "configuration for " + link.iface()); + return false; + } + config << "iface " << linkiface << " inet ppp" << std::endl + << "pre-up /sbin/ifconfig " << link.iface() << " up" << std::endl + << "provider " << linkiface; + + std::ofstream pppconf(pppdir + "/peers/" + linkiface); + if(!pppconf) { + output_error(link.where(), "pppoe: couldn't write peer information"); + return false; + } + + pppconf << "plugin rp-pppoe.so" << std::endl + << link.iface() << std::endl + << "defaultroute" << std::endl + << "noauth" << std::endl + << "+ipv6" << std::endl; + + for(const auto &cpair : params) { + if(cpair.first == "password") continue; + + std::string key{cpair.first}; + /* rewrite our keys -> pppd keys */ + if(key == "username") key = "user"; + + pppconf << key; + if(cpair.second.size() > 0) pppconf << " " << cpair.second; + pppconf << std::endl; + } + + if(params.find("password") != params.end()) { + if(params.find("username") == params.end()) { + output_error(link.where(), "pppoe: password without username " + "not supported in ENI mode"); + return false; + } + std::ofstream secrets(pppdir + "/chap-secrets"); + secrets << params.at("username") << "\t*\t" << params.at("password") + << std::endl; + } + + ppp_link_count++; + return true; +#endif } bool PPPoE::execute() const { @@ -521,7 +586,7 @@ bool PPPoE::execute() const { default: return execute_pppoe_netifrc(*this); case NetConfigType::ENI: - return execute_pppoe_eni(*this); + return execute_pppoe_eni(*this, script); } } |