$pkg) { if ($pkg['name'] == $package_name) { return $idx; } } return -1; } /* Return internal_name when it's defined, otherwise, returns name */ function get_package_internal_name($package_data) { if (isset($package_data['internal_name']) && ($package_data['internal_name'] != "")) { /* e.g. name is Ipguard-dev, internal name is ipguard */ return $package_data['internal_name']; } else { return $package_data['name']; } } /****f* pkg-utils/get_pkg_info * NAME * get_pkg_info - Retrieve package information from package server. * INPUTS * $pkgs - 'all' to retrieve all packages, an array containing package names otherwise * $info - 'all' to retrieve all information, an array containing keys otherwise * RESULT * $raw_versions - Array containing retrieved information, indexed by package name. ******/ function get_pkg_info($pkgs = 'all', $info = 'all') { global $g; $freebsd_machine = php_uname("m"); $params = array( "pkg" => $pkgs, "info" => $info, "freebsd_version" => get_freebsd_version(), "freebsd_machine" => $freebsd_machine ); $resp = call_pfsense_method('pfsense.get_pkgs', $params, 10); return $resp ? $resp : array(); } /* * resync_all_package_configs() Force packages to setup their configuration and rc.d files. * This function may also print output to the terminal indicating progress. */ function resync_all_package_configs($show_message = false) { global $config, $pkg_interface, $g; log_error(gettext("Resyncing configuration for all packages.")); if (!is_array($config['installedpackages']['package'])) { return; } if ($show_message == true) { echo "Syncing packages:"; } conf_mount_rw(); foreach ($config['installedpackages']['package'] as $idx => $package) { if (empty($package['name'])) { continue; } if ($show_message == true) { echo " " . $package['name']; } get_pkg_depends($package['name'], "all"); if (platform_booting() != true) { stop_service(get_package_internal_name($package)); } sync_package($idx, true, true); if ($pkg_interface == "console") { echo "\n" . gettext("Syncing packages:"); } } if ($show_message == true) { echo " done.\n"; } @unlink("/conf/needs_package_sync"); conf_mount_ro(); } /* * get_pkg_depends($pkg_name, $filetype = ".xml", $format = "files", return_nosync = 1): Return a package's dependencies. * * $filetype = "all" || ".xml", ".tgz", etc. * $format = "files" (full filenames) || "names" (stripped / parsed depend names) * $return_nosync = 1 (return depends that have nosync set) | 0 (ignore packages with nosync) * */ function get_pkg_depends($pkg_name, $filetype = ".xml", $format = "files", $return_nosync = 1) { global $config; $pkg_id = get_package_id($pkg_name); if ($pkg_id == -1) { return -1; // This package doesn't really exist - exit the function. } else if (!isset($config['installedpackages']['package'][$pkg_id])) { return; // No package belongs to the pkg_id passed to this function. } $package =& $config['installedpackages']['package'][$pkg_id]; if (!file_exists("/usr/local/pkg/" . $package['configurationfile'])) { log_error(sprintf(gettext('The %1$s package is missing required dependencies and must be reinstalled. %2$s'), $package['name'], $package['configurationfile'])); uninstall_package($package['name']); if (install_package($package['name']) < 0) { log_error("Failed reinstalling package {$package['name']}."); return false; } } $pkg_xml = parse_xml_config_pkg("/usr/local/pkg/" . $package['configurationfile'], "packagegui"); if (!empty($pkg_xml['additional_files_needed'])) { foreach ($pkg_xml['additional_files_needed'] as $item) { if ($return_nosync == 0 && isset($item['nosync'])) { continue; // Do not return depends with nosync set if not required. } $depend_file = substr(strrchr($item['item']['0'], '/'),1); // Strip URLs down to filenames. $depend_name = substr(substr($depend_file,0,strpos($depend_file,".")+1),0,-1); // Strip filename down to dependency name. if (($filetype != "all") && (!preg_match("/{$filetype}/i", $depend_file))) { if (($filetype != "all") && (strtolower(substr($depend_file, -strlen($filetype))) != strtolower($filetype))) { continue; } } if ($item['prefix'] != "") { $prefix = $item['prefix']; } else { $prefix = "/usr/local/pkg/"; } // Ensure that the prefix exists to avoid installation errors. if (!is_dir($prefix)) { exec("/bin/mkdir -p {$prefix}"); } if (!file_exists($prefix . $depend_file)) { log_error(sprintf(gettext("The %s package is missing required dependencies and must be reinstalled."), $package['name'])); } switch ($format) { case "files": $depends[] = $prefix . $depend_file; break; case "names": switch ($filetype) { case "all": if (preg_match("/\.xml/i", $depend_file)) { $depend_xml = parse_xml_config_pkg("/usr/local/pkg/{$depend_file}", "packagegui"); if (!empty($depend_xml)) { $depends[] = $depend_xml['name']; } } else { $depends[] = $depend_name; // If this dependency isn't package XML, use the stripped filename. } break; case ".xml": $depend_xml = parse_xml_config_pkg("/usr/local/pkg/" . $depend_file, "packagegui"); if (!empty($depend_xml)) { $depends[] = $depend_xml['name']; } break; default: $depends[] = $depend_name; // If we aren't looking for XML, use the stripped filename (it's all we have). break; } } } return $depends; } } function uninstall_package($pkg_name) { global $config, $static_output; global $builder_package_install; $id = get_package_id($pkg_name); if ($id >= 0) { stop_service(get_package_internal_name($config['installedpackages']['package'][$id])); $pkg_depends =& $config['installedpackages']['package'][$id]['depends_on_package_pbi']; $static_output .= "Removing package...\n"; update_output_window($static_output); if (is_array($pkg_depends)) { foreach ($pkg_depends as $pkg_depend) { delete_package($pkg_depend); } } else { // The packages (1 or more) are all in one long string. // We need to pass them 1 at a time to delete_package. // Compress any multiple whitespace (sp, tab, cr, lf...) into a single space char. $pkg_dep_str = preg_replace("'\s+'", ' ', $pkg_depends); // Get rid of any leading or trailing space. $pkg_dep_str = trim($pkg_dep_str); // Now we have a space-separated string. Make it into an array and process it. $pkg_dep_array = explode(" ", $pkg_dep_str); foreach ($pkg_dep_array as $pkg_depend) { delete_package($pkg_depend); } } } delete_package_xml($pkg_name); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* * sync_package($pkg_name, $sync_depends = true, $show_message = false) Force a package to setup its configuration and rc.d files. */ function sync_package($pkg_name, $sync_depends = true, $show_message = false) { global $config, $config_parsed; global $builder_package_install; // If this code is being called by pfspkg_installer // which the builder system uses then return (ignore). if ($builder_package_install) { return; } if (empty($config['installedpackages']['package'])) { return; } if (!is_numeric($pkg_name)) { $pkg_id = get_package_id($pkg_name); if ($pkg_id == -1) { return -1; // This package doesn't really exist - exit the function. } } else { $pkg_id = $pkg_name; } if (!is_array($config['installedpackages']['package'][$pkg_id])) { return; // No package belongs to the pkg_id passed to this function. } $package =& $config['installedpackages']['package'][$pkg_id]; if (!file_exists("/usr/local/pkg/" . $package['configurationfile'])) { log_error(sprintf(gettext("The %s package is missing its configuration file and must be reinstalled."), $package['name'])); delete_package_xml($package['name']); return -1; } $pkg_config = parse_xml_config_pkg("/usr/local/pkg/" . $package['configurationfile'], "packagegui"); if (isset($pkg_config['nosync'])) { return; } /* Bring in package include files */ if (!empty($pkg_config['include_file'])) { $include_file = $pkg_config['include_file']; if (file_exists($include_file)) { require_once($include_file); } else { /* XXX: What the heck is this?! */ log_error("Reinstalling package {$package['name']} because its include file({$include_file}) is missing!"); uninstall_package($package['name']); if (install_package($package['name']) < 0) { log_error("Reinstalling package {$package['name']} failed. Take appropriate measures!!!"); return -1; } } } if (!empty($pkg_config['custom_php_global_functions'])) { eval($pkg_config['custom_php_global_functions']); } if (!empty($pkg_config['custom_php_resync_config_command'])) { eval($pkg_config['custom_php_resync_config_command']); } if ($sync_depends == true) { $depends = get_pkg_depends($pkg_name, ".xml", "files", 1); // Call dependency handler and do a little more error checking. if (is_array($depends)) { foreach ($depends as $item) { if (!file_exists($item)) { require_once("notices.inc"); file_notice($package['name'], sprintf(gettext("The %s package is missing required dependencies and must be reinstalled."), $package['name']), "Packages", "/pkg_mgr_install.php?mode=reinstallpkg&pkg={$package['name']}", 1); log_error("Could not find {$item}. Reinstalling package."); uninstall_package($pkg_name); if (install_package($pkg_name) < 0) { log_error("Reinstalling package {$package['name']} failed. Take appropriate measures!!!"); return -1; } } else { $item_config = parse_xml_config_pkg($item, "packagegui"); if (empty($item_config)) { continue; } if (isset($item_config['nosync'])) { continue; } if (!empty($item_config['include_file'])) { if (file_exists($item_config['include_file'])) { require_once($item_config['include_file']); } else { log_error("Not calling package sync code for dependency {$item_config['name']} of {$package['name']} because some include files are missing."); continue; } } if ($item_config['custom_php_global_functions'] <> "") { eval($item_config['custom_php_global_functions']); } if ($item_config['custom_php_resync_config_command'] <> "") { eval($item_config['custom_php_resync_config_command']); } if ($show_message == true) { print " " . $item_config['name']; } } } } } } /* * pkg_fetch_recursive: Download and install a FreeBSD PBI package. This function provides output to * a progress bar and output window. */ function pkg_fetch_recursive($pkgname, $filename, $dependlevel = 0, $base_url = "") { global $static_output, $g, $config; // Clean up incoming filenames $filename = str_replace(" ", " ", $filename); $filename = str_replace("\n", " ", $filename); $filename = str_replace(" ", " ", $filename); $pkgs = explode(" ", $filename); foreach ($pkgs as $filename) { $filename = trim($filename); if ($g['platform'] == "nanobsd") { $pkgtmpdir = "/usr/bin/env PKG_TMPDIR=/root/ "; $pkgstagingdir = "/root/tmp"; if (!is_dir($pkgstagingdir)) { mkdir($pkgstagingdir); } $pkgstaging = "-o {$pkgstagingdir}/instmp.XXXXXX"; $fetchdir = $pkgstagingdir; } else { $fetchdir = $g['tmp_path']; } /* FreeBSD has no PBI's hosted, so fall back to our own URL for now. (Maybe fail to PC-BSD?) */ $rel = get_freebsd_version(); $priv_url = "https://files.pfsense.org/packages/{$rel}/All/"; if (empty($base_url)) { $base_url = $priv_url; } if (substr($base_url, -1) == "/") { $base_url = substr($base_url, 0, -1); } $fetchto = "{$fetchdir}/apkg_{$filename}"; $static_output .= "\n" . str_repeat(" ", $dependlevel * 2 + 1) . "Downloading {$base_url}/{$filename} ... "; if (download_file_with_progress_bar("{$base_url}/{$filename}", $fetchto) !== true) { if ($base_url != $priv_url && download_file_with_progress_bar("{$priv_url}/{$filename}", $fetchto) !== true) { $static_output .= " could not download from there or {$priv_url}/{$filename}.\n"; update_output_window($static_output); return false; } else if ($base_url == $priv_url) { $static_output .= " failed to download.\n"; update_output_window($static_output); return false; } else { $static_output .= " [{$osname} repository]\n"; update_output_window($static_output); } } $static_output .= " (extracting)\n"; update_output_window($static_output); $pkgaddout = ""; $no_checksig = ""; if (isset($config['system']['pkg_nochecksig'])) { $no_checksig = "--no-checksig"; } $result = exec("/usr/local/sbin/pbi_add " . $pkgstaging . " -f -v {$no_checksig} " . escapeshellarg($fetchto) . " 2>&1", $pkgaddout, $rc); pkg_debug($pkgname . " " . print_r($pkgaddout, true) . "\n"); if ($rc == 0) { $pbi_name = preg_replace('/\.pbi$/','',$filename); $gb = exec("/usr/local/sbin/pbi_info {$pbi_name} | /usr/bin/awk '/Prefix/ {print $2}'", $pbi_prefix); $pbi_prefix = $pbi_prefix[0]; $links = get_pbi_binaries(escapeshellarg($pbi_name)); foreach ($links as $link) { @unlink("/usr/local/{$link['link_name']}"); @symlink("{$link['target']}","/usr/local/{$link['link_name']}"); } $extra_links = array( array("target" => "bin", "link_name" => "sbin"), array("target" => "local/lib", "link_name" => "lib"), array("target" => "local/libexec", "link_name" => "libexec"), array("target" => "local/share", "link_name" => "share"), array("target" => "local/www", "link_name" => "www"), array("target" => "local/etc", "link_name" => "etc") ); foreach ($extra_links as $link) { if (!file_exists($pbi_prefix . "/" . $link['target'])) { continue; } @rmdir("{$pbi_prefix}/{$link['link_name']}"); unlink_if_exists("{$pbi_prefix}/{$link['link_name']}"); @symlink("{$pbi_prefix}/{$link['target']}", "{$pbi_prefix}/{$link['link_name']}"); } pkg_debug("pbi_add successfully completed.\n"); } else { if (is_array($pkgaddout)) { foreach ($pkgaddout as $line) { $static_output .= " " . $line .= "\n"; } } update_output_window($static_output); pkg_debug("pbi_add failed.\n"); return false; } } return true; } function get_pbi_binaries($pbi) { $result = array(); if (empty($pbi)) { return $result; } $gb = exec("/usr/local/sbin/pbi_info {$pbi} | /usr/bin/awk '/Prefix/ {print $2}'", $pbi_prefix); $pbi_prefix = $pbi_prefix[0]; if (empty($pbi_prefix)) { return $result; } foreach (array('bin', 'sbin') as $dir) { if (!is_dir("{$pbi_prefix}/{$dir}")) { continue; } $files = glob("{$pbi_prefix}/{$dir}/*.pbiopt"); foreach ($files as $f) { $pbiopts = file($f, FILE_IGNORE_NEW_LINES | FILE_SKIP_EMPTY_LINES); foreach ($pbiopts as $pbiopt) { if (!preg_match('/^TARGET:\s+(.*)$/', $pbiopt, $matches)) { continue; } $result[] = array( 'target' => preg_replace('/\.pbiopt$/', '', $f), 'link_name' => $matches[1]); } } } return $result; } function install_package($package, $pkg_info = "", $force_install = false) { global $g, $config, $static_output, $pkg_interface; /* safe side. Write config below will send to ro again. */ conf_mount_rw(); if ($pkg_interface == "console") { echo "\n"; } /* fetch package information if needed */ if (empty($pkg_info) or !is_array($pkg_info[$package])) { $pkg_info = get_pkg_info(array($package)); $pkg_info = $pkg_info[$package]; // We're only dealing with one package, so we can strip away the extra array. if (empty($pkg_info)) { conf_mount_ro(); return -1; } } if (!$force_install) { $compatible = true; $version = rtrim(file_get_contents("/etc/version")); if (isset($pkg_info['required_version'])) { $compatible = (pfs_version_compare("", $version, $pkg_info['required_version']) >= 0); } if (isset($pkg_info['maximum_version'])) { $compatible = $compatible && (pfs_version_compare("", $version, $pkg_info['maximum_version']) <= 0); } if (!$compatible) { log_error(sprintf(gettext('Package %s is not supported on this version.'), $pkg_info['name'])); $static_output .= sprintf(gettext("Package %s is not supported on this version."), $pkg_info['name']); update_status($static_output); conf_mount_ro(); return -1; } } pkg_debug(gettext("Beginning package installation.") . "\n"); log_error(sprintf(gettext('Beginning package installation for %s .'), $pkg_info['name'])); $static_output .= sprintf(gettext("Beginning package installation for %s ."), $pkg_info['name']); update_status($static_output); /* fetch the package's configuration file */ pkg_fetch_config_file($package, $pkg_info); /* add package information to config.xml */ $pkgid = get_package_id($pkg_info['name']); $static_output .= gettext("Saving updated package information...") . " "; update_output_window($static_output); if ($pkgid == -1) { $config['installedpackages']['package'][] = $pkg_info; $changedesc = sprintf(gettext("Installed %s package."),$pkg_info['name']); $to_output = gettext("done.") . "\n"; } else { $config['installedpackages']['package'][$pkgid] = $pkg_info; $changedesc = sprintf(gettext("Overwrote previous installation of %s."), $pkg_info['name']); $to_output = gettext("overwrite!") . "\n"; } if (file_exists('/conf/needs_package_sync')) { @unlink('/conf/needs_package_sync'); } conf_mount_ro(); write_config("Intermediate config write during package install for {$pkg_info['name']}."); $static_output .= $to_output; update_output_window($static_output); /* install other package components */ if (!install_package_xml($package)) { uninstall_package($package); write_config($changedesc); log_error(sprintf(gettext("Failed to install package: %s."), $pkg_info['name'])); $static_output .= gettext("Failed to install package.") . "\n"; update_output_window($static_output); return -1; } else { $static_output .= gettext("Writing configuration... "); update_output_window($static_output); write_config($changedesc); log_error(sprintf(gettext("Successfully installed package: %s."), $pkg_info['name'])); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); if ($pkg_info['after_install_info']) { update_output_window($pkg_info['after_install_info']); } } } function get_after_install_info($package) { global $pkg_info; /* fetch package information if needed */ if (!$pkg_info or !is_array($pkg_info[$package])) { $pkg_info = get_pkg_info(array($package)); $pkg_info = $pkg_info[$package]; // We're only dealing with one package, so we can strip away the extra array. } if ($pkg_info['after_install_info']) { return $pkg_info['after_install_info']; } } function eval_once($toeval) { global $evaled; if (!$evaled) { $evaled = array(); } $evalmd5 = md5($toeval); if (!in_array($evalmd5, $evaled)) { @eval($toeval); $evaled[] = $evalmd5; } return; } function install_package_xml($pkg) { global $g, $config, $static_output, $pkg_interface, $config_parsed; if (($pkgid = get_package_id($pkg)) == -1) { $static_output .= sprintf(gettext("The %s package is not installed.%sInstallation aborted."), $pkg, "\n\n"); update_output_window($static_output); if ($pkg_interface <> "console") { echo "\n"; echo "\n"; } sleep(1); return false; } else { $pkg_info = $config['installedpackages']['package'][$pkgid]; } /* pkg_add the package and its dependencies */ if ($pkg_info['depends_on_package_base_url'] != "") { if ($pkg_interface == "console") { echo "\n"; } update_status(gettext("Installing") . " " . $pkg_info['name'] . " " . gettext("and its dependencies.")); $static_output .= gettext("Downloading") . " " . $pkg_info['name'] . " " . gettext("and its dependencies... "); $static_orig = $static_output; $static_output .= "\n"; update_output_window($static_output); foreach ((array) $pkg_info['depends_on_package_pbi'] as $pkgdep) { $pkg_name = substr(reverse_strrchr($pkgdep, "."), 0, -1); $static_output = $static_orig . "\nChecking for package installation... "; update_output_window($static_output); if (!is_pkg_installed($pkg_name)) { if (!pkg_fetch_recursive($pkg_name, $pkgdep, 0, $pkg_info['depends_on_package_base_url'])) { $static_output .= "of {$pkg_name} failed!\n\nInstallation aborted."; update_output_window($static_output); pkg_debug(gettext("Package WAS NOT installed properly.") . "\n"); if ($pkg_interface <> "console") { echo "\n"; echo "\n"; } sleep(1); return false; } } } } $configfile = substr(strrchr($pkg_info['config_file'], '/'), 1); if (file_exists("/usr/local/pkg/" . $configfile)) { $static_output .= gettext("Loading package configuration... "); update_output_window($static_output); $pkg_config = parse_xml_config_pkg("/usr/local/pkg/" . $configfile, "packagegui"); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); $static_output .= gettext("Configuring package components...\n"); if (!empty($pkg_config['filter_rules_needed'])) { $config['installedpackages']['package'][$pkgid]['filter_rule_function'] = $pkg_config['filter_rules_needed']; } update_output_window($static_output); pkg_fetch_additional_files($pkg, $pkg_info); /* if a require exists, include it. this will * show us where an error exists in a package * instead of making us blindly guess */ $missing_include = false; if ($pkg_config['include_file'] <> "") { $static_output .= gettext("Loading package instructions...") . "\n"; update_output_window($static_output); pkg_debug("require_once('{$pkg_config['include_file']}')\n"); if (file_exists($pkg_config['include_file'])) { require_once($pkg_config['include_file']); } else { $missing_include = true; $static_output .= "Include " . basename($pkg_config['include_file']) . " is missing!\n"; update_output_window($static_output); /* XXX: Should undo the steps before this?! */ return false; } } /* custom commands */ $static_output .= gettext("Custom commands...") . "\n"; update_output_window($static_output); if ($missing_include == false) { if ($pkg_config['custom_php_global_functions'] <> "") { $static_output .= gettext("Executing custom_php_global_functions()..."); update_output_window($static_output); eval_once($pkg_config['custom_php_global_functions']); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } if ($pkg_config['custom_php_install_command']) { $static_output .= gettext("Executing custom_php_install_command()..."); update_output_window($static_output); /* XXX: create symlinks for conf files into the PBI directories. * change packages to store configs at /usr/pbi/pkg/etc and remove this */ eval_once($pkg_config['custom_php_install_command']); // Note: pkg may be mixed-case, e.g. "squidGuard" but the PBI names are lowercase. // e.g. "squidguard-1.4_4-i386" so feed lowercase to pbi_info below. // Also add the "-" so that examples like "squid-" do not match "squidguard-". $pkg_name_for_pbi_match = strtolower($pkg) . "-"; exec("/usr/local/sbin/pbi_info | grep '^{$pkg_name_for_pbi_match}' | xargs /usr/local/sbin/pbi_info | awk '/Prefix/ {print $2}'",$pbidirarray); $pbidir0 = $pbidirarray[0]; exec("find /usr/local/etc/ -name *.conf | grep " . escapeshellarg($pkg),$files); foreach ($files as $f) { $pbiconf = str_replace('/usr/local',$pbidir0,$f); if (is_file($pbiconf) || is_link($pbiconf)) { unlink($pbiconf); } if (is_dir(dirname($pbiconf))) { symlink($f,$pbiconf); } else { log_error("The dir for {$pbiconf} does not exist. Cannot add symlink to {$f}."); } } eval_once($pkg_config['custom_php_install_command']); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } if ($pkg_config['custom_php_resync_config_command'] <> "") { $static_output .= gettext("Executing custom_php_resync_config_command()..."); update_output_window($static_output); eval_once($pkg_config['custom_php_resync_config_command']); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } } /* sidebar items */ if (is_array($pkg_config['menu'])) { $static_output .= gettext("Menu items... "); update_output_window($static_output); foreach ($pkg_config['menu'] as $menu) { if (is_array($config['installedpackages']['menu'])) { foreach ($config['installedpackages']['menu'] as $amenu) { if ($amenu['name'] == $menu['name']) { continue 2; } } } else { $config['installedpackages']['menu'] = array(); } $config['installedpackages']['menu'][] = $menu; } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* services */ if (is_array($pkg_config['service'])) { $static_output .= gettext("Services... "); update_output_window($static_output); foreach ($pkg_config['service'] as $service) { if (is_array($config['installedpackages']['service'])) { foreach ($config['installedpackages']['service'] as $aservice) { if ($aservice['name'] == $service['name']) { continue 2; } } } else { $config['installedpackages']['service'] = array(); } $config['installedpackages']['service'][] = $service; } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } } else { $static_output .= gettext("Loading package configuration... failed!") . "\n\n" . gettext("Installation aborted."); update_output_window($static_output); pkg_debug(gettext("Unable to load package configuration. Installation aborted.") ."\n"); if ($pkg_interface <> "console") { echo "\n"; echo "\n"; } sleep(1); return false; } /* set up package logging streams */ if ($pkg_info['logging']) { system_syslogd_start(); } return true; } function delete_package($pkg) { global $config, $g, $static_output, $vardb; if (!$pkg) { return; } // Note: $pkg has the full PBI package name followed by ".pbi". Strip off ".pbi". $pkg = substr(reverse_strrchr($pkg, "."), 0, -1); if ($pkg) { $static_output .= sprintf(gettext("Starting package deletion for %s..."),$pkg); } update_output_window($static_output); pkg_delete($pkg); $static_output .= "done.\n"; update_output_window($static_output); /* Rescan directories for what has been left and avoid fooling other programs. */ mwexec("/sbin/ldconfig"); return; } function delete_package_xml($pkg) { global $g, $config, $static_output, $pkg_interface; conf_mount_rw(); $pkgid = get_package_id($pkg); if ($pkgid == -1) { $static_output .= sprintf(gettext("The %s package is not installed.%sDeletion aborted."), $pkg, "\n\n"); update_output_window($static_output); if ($pkg_interface <> "console") { echo "\n"; echo "\n"; } ob_flush(); sleep(1); conf_mount_ro(); return; } pkg_debug(sprintf(gettext("Removing %s package... "),$pkg)); $static_output .= sprintf(gettext("Removing %s components..."),$pkg) . "\n"; update_output_window($static_output); /* parse package configuration */ $packages = &$config['installedpackages']['package']; $menus =& $config['installedpackages']['menu']; $services = &$config['installedpackages']['service']; $pkg_info =& $packages[$pkgid]; if (file_exists("/usr/local/pkg/" . $pkg_info['configurationfile'])) { $pkg_config = parse_xml_config_pkg("/usr/local/pkg/" . $packages[$pkgid]['configurationfile'], "packagegui"); /* remove menu items */ if (is_array($pkg_config['menu'])) { $static_output .= gettext("Menu items... "); update_output_window($static_output); if (is_array($pkg_config['menu']) && is_array($menus)) { foreach ($pkg_config['menu'] as $menu) { foreach ($menus as $key => $instmenu) { if ($instmenu['name'] == $menu['name']) { unset($menus[$key]); break; } } } } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* remove services */ if (is_array($pkg_config['service'])) { $static_output .= gettext("Services... "); update_output_window($static_output); if (is_array($pkg_config['service']) && is_array($services)) { foreach ($pkg_config['service'] as $service) { foreach ($services as $key => $instservice) { if ($instservice['name'] == $service['name']) { if (platform_booting() != true) { stop_service($service['name']); } if ($service['rcfile']) { $prefix = RCFILEPREFIX; if (!empty($service['prefix'])) { $prefix = $service['prefix']; } if (file_exists("{$prefix}{$service['rcfile']}")) { @unlink("{$prefix}{$service['rcfile']}"); } } unset($services[$key]); } } } } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* * XXX: Otherwise inclusion of config.inc again invalidates actions taken. * Same is done during installation. */ write_config("Intermediate config write during package removal for {$pkg}."); /* * If a require exists, include it. this will * show us where an error exists in a package * instead of making us blindly guess */ $missing_include = false; if ($pkg_config['include_file'] <> "") { $static_output .= gettext("Loading package instructions...") . "\n"; update_output_window($static_output); pkg_debug("require_once(\"{$pkg_config['include_file']}\")\n"); if (file_exists($pkg_config['include_file'])) { require_once($pkg_config['include_file']); } else { $missing_include = true; update_output_window($static_output); $static_output .= "Include file " . basename($pkg_config['include_file']) . " could not be found for inclusion.\n"; } } /* ermal * NOTE: It is not possible to handle parse errors on eval. * So we prevent it from being run at all to not interrupt all the other code. */ if ($missing_include == false) { /* evaluate this package's global functions and pre deinstall commands */ if ($pkg_config['custom_php_global_functions'] <> "") { eval_once($pkg_config['custom_php_global_functions']); } if ($pkg_config['custom_php_pre_deinstall_command'] <> "") { eval_once($pkg_config['custom_php_pre_deinstall_command']); } } /* deinstall commands */ if ($pkg_config['custom_php_deinstall_command'] <> "") { $static_output .= gettext("Deinstall commands... "); update_output_window($static_output); if ($missing_include == false) { eval_once($pkg_config['custom_php_deinstall_command']); $static_output .= gettext("done.") . "\n"; } else { $static_output .= "\nNot executing custom deinstall hook because an include is missing.\n"; } update_output_window($static_output); } if ($pkg_config['include_file'] <> "") { $static_output .= gettext("Removing package instructions..."); update_output_window($static_output); pkg_debug(sprintf(gettext("Remove '%s'"), $pkg_config['include_file']) . "\n"); unlink_if_exists("/usr/local/pkg/" . $pkg_config['include_file']); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* remove all additional files */ if (is_array($pkg_config['additional_files_needed'])) { $static_output .= gettext("Auxiliary files... "); update_output_window($static_output); foreach ($pkg_config['additional_files_needed'] as $afn) { $filename = get_filename_from_url($afn['item'][0]); if ($afn['prefix'] <> "") { $prefix = $afn['prefix']; } else { $prefix = "/usr/local/pkg/"; } unlink_if_exists($prefix . $filename); } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* package XML file */ $static_output .= gettext("Package XML... "); update_output_window($static_output); unlink_if_exists("/usr/local/pkg/" . $packages[$pkgid]['configurationfile']); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } /* syslog */ if (is_array($pkg_info['logging']) && $pkg_info['logging']['logfile_name'] <> "") { $static_output .= "Syslog entries... "; update_output_window($static_output); remove_text_from_file("/etc/syslog.conf", $pkg_info['logging']['facilityname'] . "\t\t\t\t" . $pkg_info['logging']['logfilename']); system_syslogd_start(); @unlink("{$g['varlog_path']}/{$pkg_info['logging']['logfilename']}"); $static_output .= "done.\n"; update_output_window($static_output); } conf_mount_ro(); /* remove config.xml entries */ $static_output .= gettext("Configuration... "); update_output_window($static_output); unset($config['installedpackages']['package'][$pkgid]); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); write_config("Removed {$pkg} package.\n"); } function pkg_reinstall_all() { global $g, $config; @unlink('/conf/needs_package_sync'); if (is_array($config['installedpackages']['package'])) { echo gettext("One moment please, reinstalling packages...\n"); echo gettext(" >>> Trying to fetch package info..."); log_error(gettext("Attempting to reinstall all packages")); $pkg_info = get_pkg_info(); if ($pkg_info) { echo " Done.\n"; } else { $xmlrpc_base_url = get_active_xml_rpc_base_url(); $error = sprintf(gettext(' >>> Unable to communicate with %1$s. Please verify DNS and interface configuration, and that %2$s has functional Internet connectivity.'), $xmlrpc_base_url, $g['product_name']); echo "\n{$error}\n"; log_error(gettext("Cannot reinstall packages: ") . $error); return; } $todo = array(); $all_names = array(); foreach ($config['installedpackages']['package'] as $package) { $todo[] = array('name' => $package['name'], 'version' => $package['version']); $all_names[] = $package['name']; } $package_name_list = gettext("List of packages to reinstall: ") . implode(", ", $all_names); echo " >>> {$package_name_list}\n"; log_error($package_name_list); foreach ($todo as $pkgtodo) { $static_output = ""; if ($pkgtodo['name']) { log_error(gettext("Uninstalling package") . " {$pkgtodo['name']}"); uninstall_package($pkgtodo['name']); log_error(gettext("Finished uninstalling package") . " {$pkgtodo['name']}"); log_error(gettext("Reinstalling package") . " {$pkgtodo['name']}"); install_package($pkgtodo['name'], '', true); log_error(gettext("Finished installing package") . " {$pkgtodo['name']}"); } } log_error(gettext("Finished reinstalling all packages.")); } else { echo "No packages are installed."; } } function stop_packages() { require_once("config.inc"); require_once("functions.inc"); require_once("filter.inc"); require_once("shaper.inc"); require_once("captiveportal.inc"); require_once("pkg-utils.inc"); require_once("pfsense-utils.inc"); require_once("service-utils.inc"); global $config, $g; log_error("Stopping all packages."); $rcfiles = glob(RCFILEPREFIX . "*.sh"); if (!$rcfiles) { $rcfiles = array(); } else { $rcfiles = array_flip($rcfiles); if (!$rcfiles) { $rcfiles = array(); } } if (is_array($config['installedpackages']['package'])) { foreach ($config['installedpackages']['package'] as $package) { echo " Stopping package {$package['name']}..."; $internal_name = get_package_internal_name($package); stop_service($internal_name); unset($rcfiles[RCFILEPREFIX . strtolower($internal_name) . ".sh"]); echo "done.\n"; } } foreach ($rcfiles as $rcfile => $number) { $shell = @popen("/bin/sh", "w"); if ($shell) { echo " Stopping {$rcfile}..."; if (!@fwrite($shell, "{$rcfile} stop >>/tmp/bootup_messages 2>&1")) { if ($shell) { pclose($shell); } $shell = @popen("/bin/sh", "w"); } echo "done.\n"; pclose($shell); } } } function package_skip_tests($index,$requested_version) { global $config, $g; /* Get pfsense version*/ $version = rtrim(file_get_contents("/etc/version")); if ($g['platform'] == "nanobsd") { if ($index['noembedded']) { return true; } } /* If we are on not on HEAD, and the package wants it, skip */ if ($version <> "HEAD" && $index['required_version'] == "HEAD" && $requested_version <> "other") { return true; } /* If there is no required version, and the requested package version is not 'none', then skip */ if (empty($index['required_version']) && $requested_version <> "none") { return true; } /* If the requested version is not 'other', and the required version is newer than what we have, skip. */ if ($requested_version <> "other" && (pfs_version_compare("", $version, $index['required_version']) < 0)) { return true; } /* If the requested version is 'other' and we are on the version requested, skip. */ if ($requested_version == "other" && (pfs_version_compare("", $version, $index['required_version']) == 0)) { return true; } /* Package is only for an older version, lets skip */ if ($index['maximum_version'] && (pfs_version_compare("", $version, $index['maximum_version']) > 0)) { return true; } /* Do not skip package list */ return false; } function verify_all_package_servers() { return verify_package_server(get_active_xml_rpc_base_url()); } /* Check if the active package server is a valid default or if it has been altered. */ function verify_package_server($server) { /* Define the expected default package server domains. Include preceding "." to prevent matching from being too liberal. */ $default_package_domains = array('.pfsense.org', '.pfsense.com', '.netgate.com'); /* For this test we only need to check the hostname. */ $xmlrpcbase = parse_url($server, PHP_URL_HOST); foreach ($default_package_domains as $dom) { if (substr($xmlrpcbase, -(strlen($dom))) == $dom) { return true; } } return false; } /* Test the package server certificate to ensure that it validates properly */ function check_package_server_ssl() { global $g; $xmlrpcurl = get_active_xml_rpc_base_url() . $g['xmlrpcpath']; /* If the package server is using HTTP, we can't verify SSL */ if (substr($xmlrpcurl, 0, 5) == "http:") { return "http"; } /* Setup a basic cURL connection. We do not care about the content of the result, only the SSL verification. */ $ch = curl_init($xmlrpcurl); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, true); curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, '30'); curl_setopt($ch, CURLOPT_TIMEOUT, 60); curl_setopt($ch, CURLOPT_USERAGENT, $g['product_name'] . '/' . rtrim(file_get_contents("/etc/version"))); $result_page = curl_exec($ch); $verifyfail = curl_getinfo($ch, CURLINFO_SSL_VERIFYRESULT); curl_close($ch); /* The result from curl is 1 on failure, 0 on success. */ if ($verifyfail == 0) { return true; } else { return false; } } /* Keep this message centrally since it will be used several times on pages in the GUI. */ function package_server_ssl_failure_message() { $msg = "The package server's SSL certificate could not be verified. " . "The SSL certificate itself may be invalid, its chain of trust may " . "have failed validation, or the server may have been impersonated. " . "Downloaded packages may come from an untrusted source. " . "Proceed with caution."; return sprintf(gettext($msg), htmlspecialchars(get_active_xml_rpc_base_url())); } /* Keep this message centrally since it will be used several times on pages in the GUI. */ function package_server_mismatch_message() { $msg = "The package server currently configured on " . "this firewall (%s) is NOT an official package server. The contents " . "of such servers cannot be verified and may contain malicious files. " . "Return the package server settings to their default values to " . "ensure that verifiable and trusted packages are received."; return sprintf(gettext($msg), htmlspecialchars(get_active_xml_rpc_base_url())) . '

' . '' . gettext("Package Manager Settings") . ''; } function pkg_fetch_config_file($package, $pkg_info = "") { global $g, $config, $static_output, $pkg_interface; conf_mount_rw(); if (empty($pkg_info) or !is_array($pkg_info[$package])) { $pkg_info = get_pkg_info(array($package)); $pkg_info = $pkg_info[$package]; // We're only dealing with one package, so we can strip away the extra array. if (empty($pkg_info)) { conf_mount_ro(); return -1; } } /* fetch the package's configuration file */ if ($pkg_info['config_file'] != "") { $static_output .= "\n" . gettext("Downloading package configuration file... "); update_output_window($static_output); pkg_debug(gettext("Downloading package configuration file...") . "\n"); $fetchto = substr(strrchr($pkg_info['config_file'], '/'), 1); download_file_with_progress_bar($pkg_info['config_file'], '/usr/local/pkg/' . $fetchto); if (!file_exists('/usr/local/pkg/' . $fetchto)) { pkg_debug(gettext("ERROR! Unable to fetch package configuration file. Aborting installation.") . "\n"); if ($pkg_interface == "console") { print "\n" . gettext("ERROR! Unable to fetch package configuration file. Aborting package installation.") . "\n"; } else { $static_output .= gettext("failed!\n\nInstallation aborted.\n"); update_output_window($static_output); echo "
Show install log"; } conf_mount_ro(); return -1; } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } conf_mount_ro(); return true; } function pkg_fetch_additional_files($package, $pkg_info = "") { global $g, $config, $static_output, $pkg_interface; conf_mount_rw(); if (empty($pkg_info) or !is_array($pkg_info[$package])) { $pkg_info = get_pkg_info(array($package)); $pkg_info = $pkg_info[$package]; // We're only dealing with one package, so we can strip away the extra array. if (empty($pkg_info)) { conf_mount_ro(); return -1; } } $configfile = substr(strrchr($pkg_info['config_file'], '/'), 1); if (file_exists("/usr/local/pkg/" . $configfile)) { $static_output .= gettext("Loading package configuration... "); update_output_window($static_output); $pkg_config = parse_xml_config_pkg("/usr/local/pkg/" . $configfile, "packagegui"); $static_output .= gettext("done.") . "\n"; update_output_window($static_output); /* download additional files */ if (is_array($pkg_config['additional_files_needed'])) { $static_output .= gettext("Additional files... "); $static_orig = $static_output; update_output_window($static_output); foreach ($pkg_config['additional_files_needed'] as $afn) { $filename = get_filename_from_url($afn['item'][0]); if ($afn['chmod'] <> "") { $pkg_chmod = $afn['chmod']; } else { $pkg_chmod = ""; } if ($afn['prefix'] <> "") { $prefix = $afn['prefix']; } else { $prefix = "/usr/local/pkg/"; } if (!is_dir($prefix)) { safe_mkdir($prefix); } $static_output .= $filename . " "; update_output_window($static_output); if (download_file_with_progress_bar($afn['item'][0], $prefix . $filename) !== true) { $static_output .= "failed.\n"; @unlink($prefix . $filename); update_output_window($static_output); return false; } if (stristr($filename, ".tgz") <> "") { pkg_debug(gettext("Extracting tarball to -C for ") . $filename . "...\n"); $tarout = ""; exec("/usr/bin/tar xvzf " . escapeshellarg($prefix . $filename) . " -C / 2>&1", $tarout); pkg_debug(print_r($tarout, true) . "\n"); } if ($pkg_chmod <> "") { pkg_debug(sprintf(gettext('Changing file mode to %1$s for %2$s%3$s%4$s'), $pkg_chmod, $prefix, $filename, "\n")); @chmod($prefix . $filename, $pkg_chmod); system("/bin/chmod {$pkg_chmod} {$prefix}{$filename}"); } $static_output = $static_orig; update_output_window($static_output); } $static_output .= gettext("done.") . "\n"; update_output_window($static_output); } conf_mount_ro(); return true; } } ?>