qfile-dom0-unpacker.c 1.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374
  1. #define _GNU_SOURCE
  2. #include <ioall.h>
  3. #include <grp.h>
  4. #include <unistd.h>
  5. #include <stdio.h>
  6. #include <stdlib.h>
  7. #include <pwd.h>
  8. #include <sys/stat.h>
  9. #include <string.h>
  10. #include <unistd.h>
  11. #include <sys/fsuid.h>
  12. #include <errno.h>
  13. #include "filecopy.h"
  14. #define DEFAULT_MAX_UPDATES_BYTES (2L<<30)
  15. #define DEFAULT_MAX_UPDATES_FILES 2048
  16. int prepare_creds_return_uid(char *username)
  17. {
  18. struct passwd *pwd;
  19. // First try name
  20. pwd = getpwnam(username);
  21. if (!pwd) {
  22. // Then try UID
  23. pwd = getpwuid(atoi(username));
  24. if (!pwd) {
  25. perror("getpwuid");
  26. exit(1);
  27. }
  28. }
  29. setenv("HOME", pwd->pw_dir, 1);
  30. setenv("USER", pwd->pw_name, 1);
  31. setgid(pwd->pw_gid);
  32. initgroups(pwd->pw_name, pwd->pw_gid);
  33. setfsuid(pwd->pw_uid);
  34. return pwd->pw_uid;
  35. }
  36. extern int do_unpack(void);
  37. int main(int argc, char ** argv)
  38. {
  39. char *incoming_dir;
  40. int uid;
  41. char *var;
  42. long long files_limit = DEFAULT_MAX_UPDATES_FILES;
  43. long long bytes_limit = DEFAULT_MAX_UPDATES_BYTES;
  44. if (argc < 3) {
  45. fprintf(stderr, "Invalid parameters, usage: %s user dir\n", argv[0]);
  46. exit(1);
  47. }
  48. if ((var=getenv("UPDATES_MAX_BYTES")))
  49. bytes_limit = atoll(var);
  50. if ((var=getenv("UPDATES_MAX_FILES")))
  51. files_limit = atoll(var);
  52. uid = prepare_creds_return_uid(argv[1]);
  53. incoming_dir = argv[2];
  54. mkdir(incoming_dir, 0700);
  55. if (chdir(incoming_dir)) {
  56. fprintf(stderr, "Error chdir to %s", incoming_dir);
  57. exit(1);
  58. }
  59. if (chroot(incoming_dir)) {//impossible
  60. fprintf(stderr, "Error chroot to %s", incoming_dir);
  61. exit(1);
  62. }
  63. setuid(uid);
  64. set_size_limit(bytes_limit, files_limit);
  65. return do_unpack();
  66. }