main.cc 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559
  1. /*
  2. * dselect - Debian package maintenance user interface
  3. * main.cc - main program
  4. *
  5. * Copyright © 1994-1996 Ian Jackson <ijackson@chiark.greenend.org.uk>
  6. * Copyright © 2000,2001 Wichert Akkerman <wakkerma@debian.org>
  7. * Copyright © 2006-2015 Guillem Jover <guillem@debian.org>
  8. *
  9. * This is free software; you can redistribute it and/or modify
  10. * it under the terms of the GNU General Public License as published by
  11. * the Free Software Foundation; either version 2 of the License, or
  12. * (at your option) any later version.
  13. *
  14. * This is distributed in the hope that it will be useful,
  15. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. * GNU General Public License for more details.
  18. *
  19. * You should have received a copy of the GNU General Public License
  20. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  21. */
  22. #include <config.h>
  23. #include <compat.h>
  24. #include <sys/types.h>
  25. #include <sys/wait.h>
  26. #include <assert.h>
  27. #include <errno.h>
  28. #include <limits.h>
  29. #if HAVE_LOCALE_H
  30. #include <locale.h>
  31. #endif
  32. #include <ctype.h>
  33. #include <string.h>
  34. #include <fcntl.h>
  35. #include <dirent.h>
  36. #include <unistd.h>
  37. #include <stdlib.h>
  38. #include <stdio.h>
  39. // Solaris requires curses.h to be included before term.h
  40. #include "dselect-curses.h"
  41. #if defined(HAVE_NCURSESW_TERM_H)
  42. #include <ncursesw/term.h>
  43. #elif defined(HAVE_NCURSES_TERM_H)
  44. #include <ncurses/term.h>
  45. #else
  46. #include <term.h>
  47. #endif
  48. #include <dpkg/i18n.h>
  49. #include <dpkg/dpkg.h>
  50. #include <dpkg/dpkg-db.h>
  51. #include <dpkg/options.h>
  52. #include "dselect.h"
  53. #include "bindings.h"
  54. #include "pkglist.h"
  55. static const char printforhelp[] = N_("Type dselect --help for help.");
  56. bool expertmode = false;
  57. static const char *admindir = ADMINDIR;
  58. static keybindings packagelistbindings(packagelist_kinterps,packagelist_korgbindings);
  59. struct table_t {
  60. const char *name;
  61. const int num;
  62. };
  63. static const struct table_t colourtable[]= {
  64. {"black", COLOR_BLACK },
  65. {"red", COLOR_RED },
  66. {"green", COLOR_GREEN },
  67. {"yellow", COLOR_YELLOW },
  68. {"blue", COLOR_BLUE },
  69. {"magenta", COLOR_MAGENTA },
  70. {"cyan", COLOR_CYAN },
  71. {"white", COLOR_WHITE },
  72. {nullptr, 0},
  73. };
  74. static const struct table_t attrtable[]= {
  75. {"normal", A_NORMAL },
  76. {"standout", A_STANDOUT },
  77. {"underline", A_UNDERLINE },
  78. {"reverse", A_REVERSE },
  79. {"blink", A_BLINK },
  80. {"bright", A_BLINK }, // on some terminals
  81. {"dim", A_DIM },
  82. {"bold", A_BOLD },
  83. {nullptr, 0},
  84. };
  85. /* A slightly confusing mapping from dselect's internal names to
  86. * the user-visible names.*/
  87. static const struct table_t screenparttable[]= {
  88. {"list", list },
  89. {"listsel", listsel },
  90. {"title", title },
  91. {"infohead", thisstate },
  92. {"pkgstate", selstate },
  93. {"pkgstatesel", selstatesel },
  94. {"listhead", colheads },
  95. {"query", query },
  96. {"info", info_body },
  97. {"infodesc", info_head },
  98. {"infofoot", whatinfo },
  99. {"helpscreen", helpscreen },
  100. {nullptr, 0},
  101. };
  102. /* Historical (patriotic?) colours. */
  103. struct colordata color[]= {
  104. /* fore back attr */
  105. {COLOR_WHITE, COLOR_BLACK, 0 }, // default, not used
  106. {COLOR_WHITE, COLOR_BLACK, 0 }, // list
  107. {COLOR_WHITE, COLOR_BLACK, A_REVERSE }, // listsel
  108. {COLOR_WHITE, COLOR_RED, 0 }, // title
  109. {COLOR_WHITE, COLOR_BLUE, 0 }, // thisstate
  110. {COLOR_WHITE, COLOR_BLACK, A_BOLD }, // selstate
  111. {COLOR_WHITE, COLOR_BLACK, A_REVERSE | A_BOLD }, // selstatesel
  112. {COLOR_WHITE, COLOR_BLUE, 0 }, // colheads
  113. {COLOR_WHITE, COLOR_RED, 0 }, // query
  114. {COLOR_WHITE, COLOR_BLACK, 0 }, // info_body
  115. {COLOR_WHITE, COLOR_BLACK, A_BOLD }, // info_head
  116. {COLOR_WHITE, COLOR_BLUE, 0 }, // whatinfo
  117. {COLOR_WHITE, COLOR_BLACK, 0 }, // help
  118. };
  119. struct menuentry {
  120. const char *command;
  121. const char *key;
  122. const char *option;
  123. const char *menuent;
  124. urqfunction *fn;
  125. };
  126. static const menuentry menuentries[]= {
  127. { "access", N_("a"), N_("[A]ccess"), N_("Choose the access method to use."), &urq_setup },
  128. { "update", N_("u"), N_("[U]pdate"), N_("Update list of available packages, if possible."), &urq_update },
  129. { "select", N_("s"), N_("[S]elect"), N_("Request which packages you want on your system."), &urq_list },
  130. { "install", N_("i"), N_("[I]nstall"),N_("Install and upgrade wanted packages."), &urq_install },
  131. { "config", N_("c"), N_("[C]onfig"), N_("Configure any packages that are unconfigured."), &urq_config },
  132. { "remove", N_("r"), N_("[R]emove"), N_("Remove unwanted software."), &urq_remove },
  133. { "quit", N_("q"), N_("[Q]uit"), N_("Quit dselect."), &urq_quit },
  134. { nullptr, nullptr, N_("menu"), nullptr, &urq_menu },
  135. { nullptr }
  136. };
  137. static const char programdesc[]=
  138. N_("Debian '%s' package handling frontend version %s.\n");
  139. static const char licensestring[]= N_(
  140. "This is free software; see the GNU General Public License version 2 or\n"
  141. "later for copying conditions. There is NO warranty.\n");
  142. static void DPKG_ATTR_NORET
  143. printversion(const struct cmdinfo *ci, const char *value)
  144. {
  145. printf(gettext(programdesc), DSELECT, PACKAGE_RELEASE);
  146. printf("%s", gettext(licensestring));
  147. m_output(stdout, _("<standard output>"));
  148. exit(0);
  149. }
  150. static void DPKG_ATTR_NORET
  151. usage(const struct cmdinfo *ci, const char *value)
  152. {
  153. int i;
  154. printf(_(
  155. "Usage: %s [<option>...] [<command>...]\n"
  156. "\n"), DSELECT);
  157. printf(_("Commands:\n"));
  158. for (i = 0; menuentries[i].command; i++)
  159. printf(" %-10s %s\n", menuentries[i].command, menuentries[i].menuent);
  160. fputs("\n", stdout);
  161. printf(_(
  162. "Options:\n"
  163. " --admindir <directory> Use <directory> instead of %s.\n"
  164. " --expert Turn on expert mode.\n"
  165. " -D, --debug <file> Turn on debugging, send output to <file>.\n"
  166. " --color <color-spec> Configure screen colors.\n"
  167. " --colour <color-spec> Ditto.\n"
  168. ), ADMINDIR);
  169. printf(_(
  170. " -?, --help Show this help message.\n"
  171. " --version Show the version.\n"
  172. "\n"));
  173. printf(_("<color-spec> is <screen-part>:[<foreground>],[<background>][:<attr>[+<attr>]...]\n"));
  174. printf(_("<screen-part> is:"));
  175. for (i=0; screenparttable[i].name; i++)
  176. printf(" %s", screenparttable[i].name);
  177. fputs("\n", stdout);
  178. printf(_("<color> is:"));
  179. for (i=0; colourtable[i].name; i++)
  180. printf(" %s", colourtable[i].name);
  181. fputs("\n", stdout);
  182. printf(_("<attr> is:"));
  183. for (i=0; attrtable[i].name; i++)
  184. printf(" %s", attrtable[i].name);
  185. fputs("\n", stdout);
  186. m_output(stdout, _("<standard output>"));
  187. exit(0);
  188. }
  189. /* These are called by C code, so need to have C calling convention */
  190. extern "C" {
  191. static void
  192. set_debug(const struct cmdinfo*, const char *v)
  193. {
  194. FILE *fp;
  195. fp = fopen(v, "a");
  196. if (!fp)
  197. ohshite(_("couldn't open debug file '%.255s'\n"), v);
  198. debug_set_output(fp, v);
  199. debug_set_mask(dbg_general | dbg_depcon);
  200. }
  201. static void
  202. set_expert(const struct cmdinfo*, const char *v)
  203. {
  204. expertmode = true;
  205. }
  206. static int
  207. findintable(const struct table_t *table, const char *item, const char *tablename)
  208. {
  209. int i;
  210. for (i = 0; item && (table[i].name != nullptr); i++)
  211. if (strcasecmp(item, table[i].name) == 0)
  212. return table[i].num;
  213. ohshit(_("invalid %s '%s'"), tablename, item);
  214. }
  215. /*
  216. * The string's format is:
  217. * screenpart:[forecolor][,backcolor][:[<attr>, ...]
  218. * Examples: --color title:black,cyan:bright+underline
  219. * --color list:red,yellow
  220. * --color colheads:,green:bright
  221. * --color selstate::reverse // doesn't work FIXME
  222. */
  223. static void
  224. set_color(const struct cmdinfo*, const char *string)
  225. {
  226. char *s;
  227. char *colours, *attributes, *attrib, *colourname;
  228. int screenpart, aval;
  229. s = m_strdup(string); // strtok modifies strings, keep string const
  230. screenpart= findintable(screenparttable, strtok(s, ":"), _("screen part"));
  231. colours = strtok(nullptr, ":");
  232. attributes = strtok(nullptr, ":");
  233. if ((colours == nullptr || ! strlen(colours)) &&
  234. (attributes == nullptr || ! strlen(attributes))) {
  235. ohshit(_("null colour specification"));
  236. }
  237. if (colours != nullptr && strlen(colours)) {
  238. colourname= strtok(colours, ",");
  239. if (colourname != nullptr && strlen(colourname)) {
  240. // normalize attributes to prevent confusion
  241. color[screenpart].attr= A_NORMAL;
  242. color[screenpart].fore=findintable(colourtable, colourname, _("colour"));
  243. }
  244. colourname = strtok(nullptr, ",");
  245. if (colourname != nullptr && strlen(colourname)) {
  246. color[screenpart].attr= A_NORMAL;
  247. color[screenpart].back=findintable(colourtable, colourname, _("colour"));
  248. }
  249. }
  250. if (attributes != nullptr && strlen(attributes)) {
  251. for (attrib= strtok(attributes, "+");
  252. attrib != nullptr && strlen(attrib);
  253. attrib = strtok(nullptr, "+")) {
  254. aval=findintable(attrtable, attrib, _("colour attribute"));
  255. if (aval == A_NORMAL) // set to normal
  256. color[screenpart].attr= aval;
  257. else // add to existing attribs
  258. color[screenpart].attr= color[screenpart].attr | aval;
  259. }
  260. }
  261. free(s);
  262. }
  263. } /* End of extern "C" */
  264. static const struct cmdinfo cmdinfos[]= {
  265. { "admindir", 0, 1, nullptr, &admindir, nullptr },
  266. { "debug", 'D', 1, nullptr, nullptr, set_debug },
  267. { "expert", 'E', 0, nullptr, nullptr, set_expert },
  268. { "help", '?', 0, nullptr, nullptr, usage },
  269. { "version", 0, 0, nullptr, nullptr, printversion },
  270. { "color", 0, 1, nullptr, nullptr, set_color }, /* US spelling */
  271. { "colour", 0, 1, nullptr, nullptr, set_color }, /* UK spelling */
  272. { nullptr, 0, 0, nullptr, nullptr, nullptr }
  273. };
  274. static bool cursesareon = false;
  275. void curseson() {
  276. if (!cursesareon) {
  277. const char *cup, *smso;
  278. initscr();
  279. cup= tigetstr("cup");
  280. smso= tigetstr("smso");
  281. if (!cup || !smso) {
  282. endwin();
  283. if (!cup)
  284. fputs(_("Terminal does not appear to support cursor addressing.\n"),stderr);
  285. if (!smso)
  286. fputs(_("Terminal does not appear to support highlighting.\n"),stderr);
  287. fprintf(stderr,
  288. _("Set your TERM variable correctly, use a better terminal,\n"
  289. "or make do with the per-package management tool %s.\n"),
  290. DPKG);
  291. ohshit(_("terminal lacks necessary features, giving up"));
  292. }
  293. }
  294. cursesareon = true;
  295. }
  296. void cursesoff() {
  297. if (cursesareon) {
  298. clear();
  299. refresh();
  300. endwin();
  301. }
  302. cursesareon = false;
  303. }
  304. extern void *
  305. operator new(size_t size) DPKG_ATTR_THROW(std::bad_alloc)
  306. {
  307. void *p;
  308. p= m_malloc(size);
  309. assert(p);
  310. return p;
  311. }
  312. extern void
  313. operator delete(void *p) DPKG_ATTR_NOEXCEPT
  314. {
  315. free(p);
  316. }
  317. extern void
  318. operator delete(void *p, size_t size) DPKG_ATTR_NOEXCEPT
  319. {
  320. free(p);
  321. }
  322. urqresult urq_list(void) {
  323. modstatdb_open((modstatdb_rw)(msdbrw_writeifposs |
  324. msdbrw_available_readonly));
  325. curseson();
  326. packagelist *l= new packagelist(&packagelistbindings);
  327. l->resolvesuggest();
  328. l->display();
  329. delete l;
  330. modstatdb_shutdown();
  331. pkg_db_reset();
  332. return urqr_normal;
  333. }
  334. static void
  335. dme(int i, int so)
  336. {
  337. char buf[120];
  338. const menuentry *me= &menuentries[i];
  339. sprintf(buf," %c %d. %-11.11s %-80.80s ",
  340. so ? '*' : ' ', i,
  341. gettext(me->option),
  342. gettext(me->menuent));
  343. int x, y DPKG_ATTR_UNUSED;
  344. getmaxyx(stdscr,y,x);
  345. attrset(so ? A_REVERSE : A_NORMAL);
  346. mvaddnstr(i+2,0, buf,x-1);
  347. attrset(A_NORMAL);
  348. }
  349. static int
  350. refreshmenu(void)
  351. {
  352. char buf[2048];
  353. curseson(); cbreak(); noecho(); nonl(); keypad(stdscr,TRUE);
  354. int x, y DPKG_ATTR_UNUSED;
  355. getmaxyx(stdscr,y,x);
  356. clear();
  357. attrset(A_BOLD);
  358. sprintf(buf, gettext(programdesc), DSELECT, PACKAGE_RELEASE);
  359. mvaddnstr(0,0,buf,x-1);
  360. attrset(A_NORMAL);
  361. const struct menuentry *mep; int i;
  362. for (mep=menuentries, i=0; mep->option && mep->menuent; mep++, i++)
  363. dme(i,0);
  364. attrset(A_BOLD);
  365. addstr(_("\n\n"
  366. "Move around with ^P and ^N, cursor keys, initial letters, or digits;\n"
  367. "Press <enter> to confirm selection. ^L redraws screen.\n\n"));
  368. attrset(A_NORMAL);
  369. addstr(_("Copyright (C) 1994-1996 Ian Jackson.\n"
  370. "Copyright (C) 2000,2001 Wichert Akkerman.\n"));
  371. addstr(gettext(licensestring));
  372. modstatdb_init();
  373. if (!modstatdb_can_lock())
  374. addstr(_("\n\n"
  375. "Read-only access: only preview of selections is available!"));
  376. modstatdb_done();
  377. return i;
  378. }
  379. urqresult urq_menu(void) {
  380. int entries, c;
  381. entries= refreshmenu();
  382. int cursor=0;
  383. dme(0,1);
  384. for (;;) {
  385. refresh();
  386. do
  387. c= getch();
  388. while (c == ERR && errno == EINTR);
  389. if (c==ERR) {
  390. if(errno != 0)
  391. ohshite(_("failed to getch in main menu"));
  392. else {
  393. clearok(stdscr,TRUE); clear(); refreshmenu(); dme(cursor,1);
  394. }
  395. }
  396. if (c == CTRL('n') || c == KEY_DOWN || c == ' ' || c == 'j') {
  397. dme(cursor,0); cursor++; cursor %= entries; dme(cursor,1);
  398. } else if (c == CTRL('p') || c == KEY_UP || c == CTRL('h') ||
  399. c==KEY_BACKSPACE || c==KEY_DC || c=='k') {
  400. dme(cursor,0); cursor+= entries-1; cursor %= entries; dme(cursor,1);
  401. } else if (c=='\n' || c=='\r' || c==KEY_ENTER) {
  402. clear(); refresh();
  403. /* FIXME: trap errors in urq_... */
  404. urqresult res = menuentries[cursor].fn();
  405. switch (res) {
  406. case urqr_quitmenu:
  407. return urqr_quitmenu;
  408. case urqr_normal:
  409. cursor++; cursor %= entries;
  410. case urqr_fail:
  411. break;
  412. default:
  413. internerr("unknown menufn %d", res);
  414. }
  415. refreshmenu(); dme(cursor,1);
  416. } else if (c == CTRL('l')) {
  417. clearok(stdscr,TRUE); clear(); refreshmenu(); dme(cursor,1);
  418. } else if (isdigit(c)) {
  419. char buf[2]; buf[0]=c; buf[1]=0; c=atoi(buf);
  420. if (c < entries) {
  421. dme(cursor,0); cursor=c; dme(cursor,1);
  422. } else {
  423. beep();
  424. }
  425. } else if (isalpha(c)) {
  426. c= tolower(c);
  427. int i = 0;
  428. while (i < entries && gettext(menuentries[i].key)[0] != c)
  429. i++;
  430. if (i < entries) {
  431. dme(cursor,0); cursor=i; dme(cursor,1);
  432. } else {
  433. beep();
  434. }
  435. } else {
  436. beep();
  437. }
  438. }
  439. }
  440. urqresult urq_quit(void) {
  441. /* FIXME: check packages OK. */
  442. return urqr_quitmenu;
  443. }
  444. static void
  445. dselect_catch_fatal_error()
  446. {
  447. cursesoff();
  448. catch_fatal_error();
  449. }
  450. int
  451. main(int, const char *const *argv)
  452. {
  453. dpkg_locales_init(DSELECT);
  454. dpkg_set_progname(DSELECT);
  455. push_error_context_func(dselect_catch_fatal_error, print_fatal_error, nullptr);
  456. dpkg_options_load(DSELECT, cmdinfos);
  457. dpkg_options_parse(&argv, cmdinfos, printforhelp);
  458. admindir = dpkg_db_set_dir(admindir);
  459. if (*argv) {
  460. const char *a;
  461. while ((a = *argv++) != nullptr) {
  462. const menuentry *me = menuentries;
  463. while (me->command && strcmp(me->command, a))
  464. me++;
  465. if (!me->command)
  466. badusage(_("unknown action string '%.50s'"), a);
  467. me->fn();
  468. }
  469. } else {
  470. urq_menu();
  471. }
  472. cursesoff();
  473. dpkg_program_done();
  474. return(0);
  475. }