util-vserver-0.30.208
[util-vserver.git] / src / chroot-sh.c
diff --git a/src/chroot-sh.c b/src/chroot-sh.c
new file mode 100644 (file)
index 0000000..a1d4bb9
--- /dev/null
@@ -0,0 +1,219 @@
+// $Id: chroot-sh.c,v 1.2 2005/07/03 17:40:15 ensc Exp $    --*- c -*--
+
+// Copyright (C) 2005 Enrico Scholz <enrico.scholz@informatik.tu-chemnitz.de>
+//  
+// 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; version 2 of the License.
+//  
+// 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.
+
+
+#ifdef HAVE_CONFIG_H
+#  include <config.h>
+#endif
+
+#include <lib_internal/util.h>
+
+#include <unistd.h>
+#include <stdlib.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <sys/stat.h>
+
+#define ENSC_WRAPPERS_PREFIX   "chroot-sh"
+#define ENSC_WRAPPERS_UNISTD   1
+#define ENSC_WRAPPERS_IO       1
+#define ENSC_WRAPPERS_FCNTL    1
+#include <ensc_wrappers/wrappers.h>
+
+int    wrapper_exit_code = EXIT_FAILURE;
+
+static void
+showFD(int fd_in, int fd_out)
+{
+  for (;;) {
+    char               buf[4096];
+    char const *       ptr=buf;
+    ssize_t            len;
+
+    len = Eread(fd_in, buf, sizeof(buf));
+    if (len<=0) break;
+
+    EwriteAll(fd_out, ptr, len);
+  }
+}
+
+static int
+redirectFileInternal(int argc, char *argv[],
+                    int mode, bool is_input,
+                    char const *operation)
+{
+  int          fd;
+
+  if (argc<2) {
+    WRITE_MSG(2, "Not enough parameters for '");
+    WRITE_STR(2, operation);
+    WRITE_MSG(2, "' operation; use '--help' for more information\n");
+    return wrapper_exit_code;
+  }
+
+  fd = EopenD(argv[1], mode, 0644);
+  if (is_input) showFD(fd,  1);
+  else          showFD( 0, fd);
+  Eclose(fd);
+
+  return EXIT_SUCCESS;
+}
+
+static mode_t
+testInternal(int argc, char *argv[], char const *operation)
+{
+  struct stat          st;
+    
+  if (argc<2) {
+    WRITE_MSG(2, "Not enough parameters for '");
+    WRITE_STR(2, operation);
+    WRITE_MSG(2, "' operation; use '--help' for more information\n");
+    return wrapper_exit_code;
+  }
+
+  if (stat(argv[1], &st)==-1) return 0;
+  else                        return st.st_mode;
+}
+
+static int
+execCat(int argc, char *argv[])
+{
+  return redirectFileInternal(argc, argv,
+                             O_RDONLY|O_NOCTTY, true,
+                             "cat");
+}
+
+static int
+execAppend(int argc, char *argv[])
+{
+  return redirectFileInternal(argc, argv,
+                             O_WRONLY|O_CREAT|O_APPEND, false,
+                             "append");
+}
+
+static int
+execTruncate(int argc, char *argv[])
+{
+  return redirectFileInternal(argc, argv,
+                             O_WRONLY|O_CREAT|O_TRUNC, false,
+                             "truncate");
+}
+
+static int
+execRm(int argc, char *argv[])
+{
+  int          i   = 1;
+  int          res = EXIT_SUCCESS;
+  
+  if (argc<2) {
+    WRITE_MSG(2, "No files specified for 'rm' operation; try '--help' for more information\n");
+    return wrapper_exit_code;
+  }
+
+  for (;i<argc; ++i) {
+    if (unlink(argv[i])==-1) {
+      PERROR_Q(ENSC_WRAPPERS_PREFIX "unlink", argv[i]);
+      res = EXIT_FAILURE;
+    }
+  }
+
+  return res;
+}
+
+static int
+execTestFile(int argc, char *argv[])
+{
+  int          res = testInternal(argc, argv, "testfile");
+  
+  return res!=-1 && S_ISREG(res) ? EXIT_SUCCESS : EXIT_FAILURE;
+}
+
+
+static struct Command {
+    char const         *cmd;
+    int                        (*handler)(int argc, char *argv[]);
+} const                COMMANDS[] = {
+  { "cat",      execCat },
+  { "append",   execAppend },
+  { "truncate", execTruncate },
+  { "testfile", execTestFile },
+  { "rm",       execRm },
+  { 0,0 }
+};
+
+static void
+showHelp()
+{
+  WRITE_MSG(1,
+           "Usage: chroot-sh "
+           " [--] <cmd> <args>*\n\n"
+           "This program chroots into the current directory and executes the specified\n"
+           "commands there. This means that all used paths are relative to the current\n"
+           "directory, and symlinks can point to files under the current path only.\n"
+           "\n"
+           "The supported commands are:\n"
+           "  cat <file>      ...  gives out <file> to stdout\n"
+           "  append <file>   ...  appends stdin to <file> which is created when needed\n"
+           "  truncate <file> ...  clear <file> and fill it with stdin; the <file> is\n"
+           "                       created when needed\n"
+           "  rm <file>+      ...  unlink the given files\n\n"
+           "Please report bugs to " PACKAGE_BUGREPORT "\n");
+  exit(0);
+}
+
+static void
+showVersion()
+{
+  WRITE_MSG(1,
+           "chroot-sh " VERSION " -- execute commands within a chroot\n"
+           "This program is part of " PACKAGE_STRING "\n\n"
+           "Copyright (C) 2005 Enrico Scholz\n"
+           VERSION_COPYRIGHT_DISCLAIMER);
+  exit(0);
+}
+
+
+int main(int argc, char *argv[])
+{
+  struct Command const *cmd;
+  int                  idx = 1;
+  
+  if (argc>=2) {
+    if (strcmp(argv[idx], "--help")   ==0) showHelp();
+    if (strcmp(argv[idx], "--version")==0) showVersion();
+    if (strcmp(argv[idx], "--")==0)        ++idx;
+  }
+
+  if (argc<idx+1) {
+    WRITE_MSG(2, "No command specified; try '--help' for more information\n");
+    return wrapper_exit_code;
+  }
+
+  Echroot(".");
+  Echdir("/");
+
+  for (cmd=COMMANDS+0; cmd->cmd!=0; ++cmd) {
+    if (strcmp(cmd->cmd, argv[idx])==0)
+      return cmd->handler(argc-idx, argv+idx);
+  }
+
+  WRITE_MSG(2, "Invalid command '");
+  WRITE_STR(2, argv[idx]);
+  WRITE_MSG(2, "'; try '--help' for more information\n");
+
+  return wrapper_exit_code;
+}