summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAchim Gottinger <achim@gentoo.org>2001-04-26 03:01:20 +0000
committerAchim Gottinger <achim@gentoo.org>2001-04-26 03:01:20 +0000
commit74f663fded9f9797d467ddb1ddecdb766dc94d9b (patch)
tree89624c2fe04ef8aac33fb6675c9946446752c457 /app-admin
parentFixed glibc-2.2.2 header problem (diff)
downloadgentoo-2-74f663fded9f9797d467ddb1ddecdb766dc94d9b.tar.gz
gentoo-2-74f663fded9f9797d467ddb1ddecdb766dc94d9b.tar.bz2
gentoo-2-74f663fded9f9797d467ddb1ddecdb766dc94d9b.zip
Warrens gentool webadmin interface
Diffstat (limited to 'app-admin')
-rwxr-xr-xapp-admin/gentool/files/0.0.1/gentool.py609
-rw-r--r--app-admin/gentool/files/gentoo.gifbin0 -> 6265 bytes
-rwxr-xr-xapp-admin/gentool/files/gentool-run2
-rwxr-xr-xapp-admin/gentool/files/svc-gentool27
-rw-r--r--app-admin/gentool/gentool-0.0.1.ebuild22
5 files changed, 660 insertions, 0 deletions
diff --git a/app-admin/gentool/files/0.0.1/gentool.py b/app-admin/gentool/files/0.0.1/gentool.py
new file mode 100755
index 000000000000..052abd799479
--- /dev/null
+++ b/app-admin/gentool/files/0.0.1/gentool.py
@@ -0,0 +1,609 @@
+#!/usr/bin/python
+#
+# Gentool. Written by warren. Released under GNU.
+#
+#
+# NOTE: this should be ran as root... but then again.. that's probably a security hazard... :)
+#
+#
+#
+# Change Log
+# ----------
+#
+# 0.0.1
+# fixed a problem with viewing details of a package that isn't already installed
+# added run-command for rsync
+# 0.0.0
+# initial release
+#
+#
+# To-Do
+# -----
+#
+# -set up some sort of buffering system so popen (as well as everything else) doesn't tie up
+# up other connections. (have it fork perhaps? not sure...). (but then again... there
+# should never really be more than one person on it... oh well still a bug)
+# -try to hack it (buffer overflows? etc...) and fix it if anything is vulnerable.. :)
+# -add an idiot-proof section so you can modify the USE variables easily.
+#
+
+server_version = "Gentool 0.0.1 (Gentoo)" # don't change this
+
+port = 8088 # port this runs on (i.e. http://localhost:port/)
+username = "gentoo" # username
+password = "gentoo" # password
+install_command = "/usr/sbin/emerge --pretend" # change to "/usr/sbin/emerge" if you want it to actually merge packages
+uninstall_command = "/bin/echo removing:" # change to "/usr/sbin/ebuild" if you want it to actually unmerge packages
+rsync_command = "/usr/sbin/emerge rsync" # command to rsync
+path_to_graphics = "/usr/share/gentool/images/" # change to the path of gentoo.gif
+
+
+
+#
+# nothing below here should be changed...
+#
+
+import os
+import sys
+import socket
+import select
+import string
+import base64
+import portage
+import urllib
+
+
+class session:
+ raw_data = "" # raw data before it gets parsed
+ all_headers = [] # each line of the header
+ form_data = "" # data from FORM post and/or in the URL
+ method = "" # usually GET or POST
+ version = "" # usually HTTP/1.1
+ uri = "" # the document the user wants minus any ?blah=blah&blah=blah stuff
+
+ authorized = 0
+ def CheckSocketForData(self):
+ # returns 0 if everything is okay, and 1 if the socket has been dropped
+
+ # get all headers and possible post data (can't get headers/body bigger than a meg...)
+ # hrmm... if the browser doesn't send all the data at once... this won't read all of the
+ # data... TODO: make it loop until it gets <ENTER><ENTER> and then continue with sending
+ # to different functions
+ self.raw_data = self.socket.recv(1048576)
+ if len(self.raw_data) != 0:
+ self.ParseClientData()
+ self.SendOutput()
+ self.socket.close()
+ return 1
+
+ def ParseClientData(self):
+ # NOTE: assumes linefeeds == "\r\n"... might not be true for all browsers?
+ print "Parsing headers..."
+
+ splitting_point = string.find(self.raw_data,"\r\n\r\n")
+ header_part = self.raw_data[0:splitting_point]
+ body_part = self.raw_data[splitting_point+4:]
+
+ request = string.split(header_part,"\r\n")[0]
+ if len(string.split(request)) != 3:
+ print "Invalid request"
+ return 1
+
+ self.method = string.split(request)[0] # for instance, GET or POST
+ print " Method: ", self.method
+ self.version = string.split(request)[2] # for instance, HTTP/1.1
+ print " Version: ", self.version
+
+ temp_uri = string.split(string.split(request)[1],"?")
+ if len(temp_uri) == 2:
+ self.uri = temp_uri[0]
+ self.form_data = temp_uri[1]
+ else:
+ self.uri = temp_uri[0]
+
+ print " URL: ", self.uri
+ print " Headers:"
+
+ num_lines = 1
+ for line in string.split(header_part,"\r\n")[1:]:
+ self.all_headers[num_lines:num_lines] = [line]
+ num_lines = num_lines + 1
+ print " ", line
+ if len(line) == 0:
+ break
+
+ print "Storing data..."
+ if body_part != "":
+ if self.form_data!= "":
+ # ?something=something in the URI (hrmm... posting and a ?something=something in the url? odd...)
+ self.form_data = self.form_data + "&" + body_part
+ else:
+ self.form_data = body_part
+
+ print "Checking authorization... ",
+ for x in self.all_headers:
+ line = string.split(x)
+ if len(line) == 3:
+ if string.lower(line[0]) == "authorization:":
+ base64_string = line[2]
+ decoded_string = base64.decodestring(base64_string)
+ entered_username = string.split(decoded_string,":")[0]
+ entered_password = string.split(decoded_string,":")[1]
+ if entered_username == username:
+ if entered_password == password:
+ self.authorized = 1
+ break #does this break out of the for or one of the if's?
+ else:
+ break #does this break out of the for or one of the if's?
+ else:
+ break #does this break out of the for or one of the if's?
+ if self.authorized == 0:
+ print "failed"
+ else:
+ print "passed"
+
+
+ def SendOutput(self):
+
+ if self.authorized:
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Index page
+ ##
+ if self.uri == "/":
+ self.PrintHeader()
+ self.PrintHTMLHeader("Pick which packages you want to upgrade...")
+ self.socket.send("""<CENTER>Welcome to gentoo linux!</CENTER><BR><BR>(This page will have more in the future).""")
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Print out packages that can be updated
+ ##
+ elif self.uri == "/view-upgrades":
+
+ self.PrintHeader()
+ self.PrintHTMLHeader("Pick which packages you want to upgrade...")
+
+ myvirtuals=portage.getvirtuals(portage.root)
+
+ allpackages = portage.portagetree("/",myvirtuals)
+ allpackages.populate()
+
+ mypackages = portage.vartree("/",myvirtuals)
+ mypackages.populate()
+
+ self.socket.send("""
+ The following packages should be upgraded:<BR><BR>
+
+ <FORM METHOD="post" ACTION="/upgrade-packages">
+ <TABLE WIDTH="70%" CELLSPACING=1 CELLPADDING=0 BORDER=0 ALIGN="center" BGCOLOR="#000000"><TR><TD>
+ <TABLE WIDTH="100%" CELLSPACING=0 CELLPADDING=3 BORDER=0>
+ <TR>
+ <TD WIDTH="45%" BGCOLOR="#202020"><FONT COLOR="#ffffff"><B>Package Name</FONT></TD>
+ <TD WIDTH="25%" ALIGN="center" BGCOLOR="#202020"><FONT COLOR="#ffffff"><B>Current Version</FONT></TD>
+ <TD WIDTH="25%" ALIGN="center" BGCOLOR="#202020"><FONT COLOR="#ffffff"><B>Latest Version</B></FONT></TD>
+ <TD WIDTH="5%" ALIGN="center" BGCOLOR="#202020"><FONT COLOR="#ffffff"><B>Upgrade</B></FONT></TD>
+ </TR>""")
+ switch = 0
+ for x in mypackages.tree.keys():
+ should_we_upgrade = 0 # 0 == don't know, 1 == yes, 2 == no
+ temp_package = mypackages.tree[x][0][1][0] + "/" + mypackages.tree[x][0][1][1]
+ temp_best_match = allpackages.dep_bestmatch(temp_package)
+ latest_package = portage.catpkgsplit(temp_best_match)[1:]
+ for y in mypackages.tree[x]:
+ current_package = portage.catpkgsplit(y[0])[1:]
+ compare_value = portage.pkgcmp(current_package,latest_package)
+ if compare_value == -1:
+ if should_we_upgrade != 2:
+ should_we_upgrade = 1
+ else:
+ should_we_upgrade = 2
+ if should_we_upgrade == 1:
+ if switch == 0:
+ switch = 1
+ color = "#cccccc"
+ else:
+ switch = 0
+ color = "#bbbbbb"
+
+ latest_ebuild_name = allpackages.getname(mypackages.tree[x][0][1][0] + "/" + latest_package[0] + "-" + latest_package[1] + "-" + latest_package[2])
+
+ ebuild_names = latest_ebuild_name + "|"
+ first_one = 1
+ for y in mypackages.tree[x]:
+ if first_one == 1:
+ ebuild_names = ebuild_names + allpackages.getname(y[0])
+ first_one = 0
+ else:
+ ebuild_names = "|" + ebuild_names + allpackages.getname(y[0])
+
+ self.socket.send("""
+ <TR>
+ <TD BGCOLOR=""" + "\"" + color + "\">" + latest_package[0] + """</TD>
+ <TD ALIGN="center" BGCOLOR=""" + "\"" + color + "\">" + current_package[1] + "-" + current_package[2] + """</TD>
+ <TD ALIGN="center" BGCOLOR=""" + "\"" + color + "\">" + latest_package[1] + "-" + latest_package[2] + """</TD>
+ <TD ALIGN="center" BGCOLOR=""" + "\"" + color + "\"" + """>
+ <INPUT TYPE="checkbox" NAME="package" VALUE=""" + "\"" + ebuild_names + "\"" + """>
+ </TD>
+ </TR>""")
+ if switch == 0:
+ switch = 1
+ color = "#cccccc"
+ else:
+ switch = 0
+ color = "#bbbbbb"
+ self.socket.send("""
+ <TR>
+ <TD ALIGN="center" COLSPAN=4 BGCOLOR=""" + "\"" + color + "\"" + """>
+ <INPUT TYPE="submit" VALUE="Upgrade Packages Now">
+ </TD>
+ </TR>
+ </TABLE>
+ </TD></TR></TABLE>
+ </FORM>
+ """)
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Print out all packages that have already been installed
+ ##
+ elif self.uri == "/view-installed":
+
+ self.PrintHeader()
+ self.PrintHTMLHeader("Packages that are already installed...")
+ myvirtuals=portage.getvirtuals(portage.root)
+
+ mypackages = portage.vartree("/",myvirtuals)
+ mypackages.populate()
+
+ sorted_keys = mypackages.tree.keys()
+ sorted_keys.sort()
+
+ for c in portage.categories:
+ # each category
+ printed_category = 0
+ for x in sorted_keys:
+ # each package
+ if mypackages.tree[x][0][1][0] == c:
+ if printed_category == 0:
+ self.socket.send("<B>" + c + "</B><BR>")
+ printed_category = 1
+
+ self.socket.send("&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<A HREF=\"/package-details?package=" + mypackages.tree[x][0][1][0] + "/" + mypackages.tree[x][0][1][1] + "\">" + mypackages.tree[x][0][1][1] + "</A> (")
+ printed_first = 0
+ for y in mypackages.tree[x]:
+ # each version of it
+ if printed_first == 0:
+ printed_first = 1
+ else:
+ self.socket.send(", ")
+ self.socket.send(y[1][2] + "-" + y[1][3])
+ self.socket.send(")<BR>")
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## package details
+ ##
+ elif self.uri == "/package-details":
+ self.PrintHeader()
+ self.PrintHTMLHeader("Package details...")
+
+ package_name = string.split(self.form_data,"=")[1]
+
+ myvirtuals = portage.getvirtuals(portage.root)
+
+ mypackages = portage.vartree("/",myvirtuals)
+ mypackages.populate()
+
+ allpackages = portage.portagetree("/",myvirtuals)
+ allpackages.populate()
+
+
+ self.socket.send("<CENTER><H1>" + package_name + "</H1></CENTER><BR>")
+ self.socket.send("<BR><B>You have the following version(s) installed:</B><BR>")
+ if package_name in mypackages.tree.keys():
+ for x in mypackages.tree[package_name]:
+ self.socket.send("&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;" + x[0] + " <SMALL>[ <A HREF=\"remove-package?package=" + allpackages.getname(x[0]) + "\">remove</A> ]</SMALL><BR>")
+
+
+ self.socket.send("<BR><B>You can install the following version(s):</B><BR>")
+ for x in allpackages.tree[package_name]:
+ if package_name in mypackages.tree.keys() and x in mypackages.tree[package_name]:
+ #already installed
+ pass
+ else:
+ self.socket.send("&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;" + x[0] + " <SMALL>[ <A HREF=\"install-package?package=" + allpackages.getname(x[0]) + "\">install</A> ]</SMALL><BR>")
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## run emerge rsync
+ ##
+ elif self.uri == "/rsync":
+
+ self.PrintHeader()
+ self.PrintHTMLHeader("Synchronizing... Please wait...")
+ rsync_log = os.popen4(rsync_command)[1]
+
+ while 1:
+ c = rsync_log.readline()
+ if c == "":
+ break
+ self.socket.send(c + "<BR>")
+
+
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Print out all packages that can be installed
+ ##
+ elif self.uri == "/view-available":
+
+ self.PrintHeader()
+ self.PrintHTMLHeader("Packages Available to Install...")
+
+ myvirtuals=portage.getvirtuals(portage.root)
+
+ allpackages = portage.portagetree("/",myvirtuals)
+ allpackages.populate()
+
+ sorted_keys = allpackages.tree.keys()
+ sorted_keys.sort()
+
+ for c in portage.categories:
+ # each category
+ self.socket.send("<B>" + c + "</B><BR>")
+ for x in sorted_keys:
+ # each package
+ if allpackages.tree[x][0][1][0] == c:
+ self.socket.send("&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<A HREF=\"/package-details?package=" + allpackages.tree[x][0][1][0] + "/" + allpackages.tree[x][0][1][1] + "\">" + allpackages.tree[x][0][1][1] + "</A> (")
+ printed_first = 0
+ for y in allpackages.tree[x]:
+ # each version of it
+ if printed_first == 0:
+ printed_first = 1
+ else:
+ self.socket.send(", ")
+ self.socket.send(y[1][2] + "-" + y[1][3])
+ self.socket.send(")<BR>")
+
+
+ self.PrintHTMLFooter()
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Do the actual upgrading of packages. Data is passed in in the form of:
+ ## package=/path/to/latest.ebuild|/path/to/prior.ebuild|/path/to/other/prior.ebuild&package=...
+ ##
+ elif self.uri == "/upgrade-packages":
+
+ self.PrintHeader()
+ self.PrintHTMLHeader("Upgrading Packages... Please wait...")
+ for x in string.split(self.form_data,"&"):
+ key_and_value = string.split(urllib.unquote(x),"=")
+ if len(key_and_value) == 2:
+ value = key_and_value[1]
+ ebuild_path = string.split(value,"|")[0]
+
+ self.socket.send("<B>" + install_command + " " + ebuild_path + "</B><BR>")
+ install_log = os.popen4(install_command + " " + ebuild_path)[1]
+ while 1:
+ c = install_log.readline()
+ if c == "":
+ break
+ self.socket.send(c + "<BR>")
+ self.socket.send("<BR><BR>")
+
+ for y in string.split(value,"|")[1:]:
+ self.socket.send("<B>" + uninstall_command + " " + ebuild_path + " unmerge</B><BR>")
+ install_log = os.popen4(uninstall_command + " " + y + " unmerge")[1]
+ while 1:
+ c = install_log.readline()
+ if c == "":
+ break
+ self.socket.send(c + "<BR>")
+ self.socket.send("<BR><BR>")
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Install a package
+ ##
+ elif self.uri == "/install-package":
+ self.PrintHeader()
+ self.PrintHTMLHeader("Installing package... Please wait...")
+
+ key_and_value = string.split(urllib.unquote(self.form_data),"=")
+ if len(key_and_value) == 2:
+ self.socket.send("<B>" + install_command + " " + key_and_value[1] + "</B><BR>")
+ install_log = os.popen4(install_command + " " + key_and_value[1])[1]
+ while 1:
+ c = install_log.readline()
+ if c == "":
+ break
+ self.socket.send(c + "<BR>")
+ self.socket.send("<BR><BR>")
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Remove a package
+ ##
+ elif self.uri == "/remove-package":
+ self.PrintHeader()
+ self.PrintHTMLHeader("Removing package... Please wait...")
+
+ key_and_value = string.split(urllib.unquote(self.form_data),"=")
+ if len(key_and_value) == 2:
+ self.socket.send("<B>" + uninstall_command + " " + key_and_value[1] + " unmerge</B><BR>")
+ install_log = os.popen4(uninstall_command + " " + key_and_value[1] + " unmerge")[1]
+ while 1:
+ c = install_log.readline()
+ if c == "":
+ break
+ self.socket.send(c + "<BR>")
+ self.socket.send("<BR><BR>")
+
+ self.PrintHTMLFooter()
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Print Image
+ ##
+ ## NOTE: only works if the image is in /home/warren/python/images/<imagename> and is of type gif
+ ## NOTE: if you request /images/asdf/fdsa/../../../../../test.asdf, this will try to open
+ ## the file /home/warren/python/images/test.asdf (i.e. it only takes stuff after last /)
+ ##
+ elif self.uri[0:8] == "/images/":
+ self.PrintHeader(200,"image/" + self.uri[-3:])
+
+ filename = path_to_graphics + string.split(self.uri,'/')[-1]
+ inp = open(filename,"rb")
+
+ temp = inp.readlines()
+ for x in temp:
+ self.socket.send(x)
+
+
+ #####################################################################################################
+ #####################################################################################################
+ ##
+ ## Error 404: File Not Found
+ ##
+ else:
+ self.PrintHeader(404)
+ self.socket.send("<HTML><HEAD>ERROR 404: File Not Found!</BODY></HTML>")
+
+ else:
+ self.PrintHeader(401,"text/html;",'WWW-Authenticate: Basic realm="Gentool"')
+ self.socket.send("""<HTML><HEAD>
+ <TITLE>401 Authorization Required</TITLE>
+ </HEAD><BODY>
+ <H1>Authorization Required</H1>
+ This server could not verify that you
+ are authorized to access the document
+ requested. Either you supplied the wrong
+ credentials (e.g., bad password), or your
+ browser doesn't understand how to supply
+ the credentials required.<P>
+ <HR>
+ <ADDRESS>""" + server_version + """ Server at localhost Port 80</ADDRESS>
+ </BODY></HTML>""")
+
+ def PrintHeader(self,status=200,contenttype="text/html;",extra=""):
+ print "Printing server header (status: " + repr(status) + ") \n"
+ if status == 200:
+ self.socket.send("HTTP/1.1 200 OK\r\n")
+ elif status == 401:
+ self.socket.send("HTTP/1.1 401 Authorization Required\r\n")
+ elif status == 404:
+ self.socket.send("HTTP/1.1 404 File Not Found\r\n")
+
+ self.socket.send("Server: " + server_version + "\r\n")
+ self.socket.send("Content-Type: " + contenttype + "\r\n")
+ if extra != "":
+ self.socket.send(extra + "\r\n")
+ self.socket.send("\r\n")
+
+ def PrintHTMLHeader(self,title=""):
+ self.socket.send("""<HTML>
+ <HEAD>
+ <TITLE>Gentool</TITLE>
+ </HEAD>
+ <BODY LEFTMARGIN=0 TOPMARGIN=0 MARGINHEIGHT=0 MARGINWIDTH=0>
+ <TABLE WIDTH="100%" BORDER=0 CELLSPACING=0 CELLPADDING=0 HEIGHT=100>
+ <TR>
+ <TD BGCOLOR="#000000">
+ <IMG SRC="/images/gentoo.gif" BORDER=0>
+ </TD>
+ <TD BGCOLOR="#000000" ALIGN="right" VALIGN="top">
+ <BR>
+ <A HREF="/" STYLE="text-decoration:none"><FONT COLOR="#ffffff">Home</FONT></A><BR>
+ <A HREF="/rsync" STYLE="text-decoration:none"><FONT COLOR="#ffffff">rsync</FONT></A><BR>
+ <A HREF="/view-upgrades" STYLE="text-decoration:none"><FONT COLOR="#ffffff">Upgrade Packages</FONT></A><BR>
+ <A HREF="/view-installed" STYLE="text-decoration:none"><FONT COLOR="#ffffff">Already Installed Packages</FONT></A><BR>
+ <A HREF="/view-available" STYLE="text-decoration:none"><FONT COLOR="#ffffff">Install New Packages</FONT></A><BR>
+ </TD>
+ </TR>
+ <TR>
+ <TD BGCOLOR="#000000" COLSPAN=2>
+ <TABLE WIDTH="100%" CELLSPACING=0 CELLPADDING=3><TR><TD>
+ <FONT COLOR="#00FF00"><B>""" + server_version + " " + title + """</B></FONT>
+ </TD></TR></TABLE>
+ </TD>
+ </TR>
+ </TABLE>
+ <BR>""")
+ def PrintHTMLFooter(self):
+ self.socket.send("""<BR><BR></BODY>
+ </HTML>""")
+
+
+
+
+
+class HTTPD:
+ def __init__ (self):
+ self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
+ self.socket.bind(('', port))
+ self.socket.setblocking (0)
+ self.socket.listen(5)
+ self.connections = []
+ def MainLoop(self):
+ # this loop will wait for connections, as well as data on connections...
+ # if it gets data on connections, it'll call another function...
+ r = []; w = []; e = []
+
+ while 1:
+ r,w,e = select.select ([self.socket.fileno()],[self.socket.fileno()],[self.socket.fileno()],0.1)
+ if len(r) != 0:
+ # new connection
+ conn, addr = self.socket.accept()
+ self.connections.insert(0,session())
+ self.connections[0].socket = conn
+ self.connections[0].addr = addr
+
+ print "Connection from", addr
+ # if you want to add some ip-checking (localhost only perhaps?), do it here against "addr"
+
+ for x in self.connections:
+ # go through existing connections and check for input
+ r,w,e = select.select ([x.socket.fileno()],[x.socket.fileno()],[x.socket.fileno()],0.1)
+ if len(r) != 0:
+ if x.CheckSocketForData() == 1:
+ #client dropped the connection... remove it from the list
+ self.connections.remove(x)
+ print "Client disconnected"
+
+
+
+temp = HTTPD()
+temp.MainLoop() \ No newline at end of file
diff --git a/app-admin/gentool/files/gentoo.gif b/app-admin/gentool/files/gentoo.gif
new file mode 100644
index 000000000000..5358540919bf
--- /dev/null
+++ b/app-admin/gentool/files/gentoo.gif
Binary files differ
diff --git a/app-admin/gentool/files/gentool-run b/app-admin/gentool/files/gentool-run
new file mode 100755
index 000000000000..c0e61e8dc35e
--- /dev/null
+++ b/app-admin/gentool/files/gentool-run
@@ -0,0 +1,2 @@
+#!/bin/sh
+exec /usr/sbin/gentool /usr/bin/multilog t /var/log/gentool.d
diff --git a/app-admin/gentool/files/svc-gentool b/app-admin/gentool/files/svc-gentool
new file mode 100755
index 000000000000..833580c11d76
--- /dev/null
+++ b/app-admin/gentool/files/svc-gentool
@@ -0,0 +1,27 @@
+#!/bin/sh
+#RCUPDATE:3 4:72:This line is required for script management
+
+. /etc/rc.d/config/functions
+
+SERVICE=gentool
+opts="start stop"
+
+
+start() {
+ ebegin "Starting supervised ${SERVICE}"
+ ln -sf ../services/${SERVICE} ${SVCDIR}/control/${SERVICE}
+ eend $?
+}
+
+stop() {
+ ebegin "Stopping supervised ${SERVICE}"
+ if [ -e ${SVCDIR}/control/${SERVICE} ]
+ then
+ /usr/bin/svc -dx ${SVCDIR}/control/${SERVICE}
+ rm ${SVCDIR}/control/${SERVICE}
+ fi
+ eend $?
+}
+
+doservice ${@}
+
diff --git a/app-admin/gentool/gentool-0.0.1.ebuild b/app-admin/gentool/gentool-0.0.1.ebuild
new file mode 100644
index 000000000000..20b7c62f90b2
--- /dev/null
+++ b/app-admin/gentool/gentool-0.0.1.ebuild
@@ -0,0 +1,22 @@
+# Copyright 1999-2000 Gentoo Technologies, Inc.
+# Distributed under the terms of the GNU General Public License, v2 or later
+# Author Achim Gottinger <achim@gentoo.org>
+# $Header: /var/cvsroot/gentoo-x86/app-admin/gentool/gentool-0.0.1.ebuild,v 1.1 2001/04/26 03:01:20 achim Exp $
+
+S=${WORKDIR}/${P}
+DESCRIPTION="A stand alone webserver written in python actiong as a webinterface to portage"
+
+
+src_install () {
+
+ exeinto /usr/sbin
+ doexe ${FILESDIR}/${PV}/gentool.py
+ insinto /usr/share/gentool/images
+ doins ${FILESDIR}/gentoo.gif
+ exeinto /etc/rc.d/init.d
+ doexe ${FILESDIR}/svc-gentool
+ exeinto /var/lib/supervise/services/gentool
+ newexe ${FILESDIR}/gentool-run run
+
+}
+