2012-07-06 14:08:17 +04:00
/*
* Samba Unix / Linux SMB client library
* Registry Editor
* Copyright ( C ) Christopher Davis 2012
*
* 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 ; either version 3 of the License , or
* ( at your option ) any later version .
*
* 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
# include "popt_common.h"
# include "lib/util/data_blob.h"
# include "lib/registry/registry.h"
# include "regedit.h"
2012-07-13 08:11:55 +04:00
# include "regedit_treeview.h"
2012-07-18 13:31:41 +04:00
# include "regedit_valuelist.h"
2012-07-27 14:37:22 +04:00
# include "regedit_dialog.h"
2014-06-10 05:29:56 +04:00
# include "regedit_list.h"
2012-07-10 17:16:35 +04:00
# include <ncurses.h>
# include <menu.h>
2012-07-27 14:37:22 +04:00
# include <panel.h>
2012-08-11 07:05:20 +04:00
# define KEY_START_X 0
2014-05-21 03:17:42 +04:00
# define KEY_START_Y 1
2012-08-11 07:05:20 +04:00
# define KEY_WIDTH (COLS / 4)
2012-08-16 08:05:01 +04:00
# define KEY_HEIGHT (LINES - KEY_START_Y - 2)
2012-08-11 07:05:20 +04:00
# define VAL_START_X KEY_WIDTH
2014-05-21 03:17:42 +04:00
# define VAL_START_Y 1
2012-08-11 07:05:20 +04:00
# define VAL_WIDTH (COLS - KEY_WIDTH)
2012-08-16 08:05:01 +04:00
# define VAL_HEIGHT (LINES - VAL_START_Y - 2)
2014-05-21 03:17:42 +04:00
2012-08-16 08:05:01 +04:00
# define HELP1_START_Y (LINES - 2)
# define HELP1_START_X 0
# define HELP1_WIDTH (LINES)
# define HELP2_START_Y (LINES - 1)
# define HELP2_START_X 0
# define HELP2_WIDTH (LINES)
2012-08-12 09:10:51 +04:00
# define PATH_START_Y 0
# define PATH_START_X 6
2012-08-12 10:22:06 +04:00
# define PATH_MAX_Y (COLS - 1)
# define PATH_WIDTH (COLS - 6)
2012-08-12 09:10:51 +04:00
# define PATH_WIDTH_MAX 1024
2012-08-11 07:05:20 +04:00
2012-07-27 14:37:22 +04:00
struct regedit {
2014-07-01 10:14:20 +04:00
struct registry_context * registry_context ;
2012-07-27 14:37:22 +04:00
WINDOW * main_window ;
WINDOW * path_label ;
2012-08-12 10:22:06 +04:00
size_t path_len ;
2012-07-27 14:37:22 +04:00
struct value_list * vl ;
struct tree_view * keys ;
bool tree_input ;
2014-06-03 08:50:01 +04:00
struct regedit_search_opts active_search ;
2012-07-27 14:37:22 +04:00
} ;
2012-07-06 14:08:17 +04:00
2012-08-11 08:56:58 +04:00
static struct regedit * regedit_main = NULL ;
2012-08-12 09:10:51 +04:00
static void show_path ( struct regedit * regedit )
{
2012-08-12 10:22:06 +04:00
int start_pad = 0 ;
int start_win = PATH_START_X ;
if ( PATH_START_X + regedit - > path_len > COLS ) {
start_pad = 3 + PATH_START_X + regedit - > path_len - COLS ;
mvprintw ( PATH_START_Y , start_win , " ... " ) ;
start_win + = 3 ;
}
2012-08-12 10:29:13 +04:00
copywin ( regedit - > path_label , regedit - > main_window , 0 , start_pad ,
PATH_START_Y , start_win , PATH_START_Y , PATH_MAX_Y , false ) ;
2014-05-23 02:23:52 +04:00
mvchgat ( 0 , 0 , COLS , A_BOLD , PAIR_YELLOW_CYAN , NULL ) ;
2012-08-12 09:10:51 +04:00
}
static void print_path ( struct regedit * regedit , struct tree_node * node )
{
2012-08-12 10:22:06 +04:00
regedit - > path_len = tree_node_print_path ( regedit - > path_label , node ) ;
2012-08-12 10:29:13 +04:00
show_path ( regedit ) ;
2012-08-12 09:10:51 +04:00
}
2012-08-13 09:38:06 +04:00
static void print_help ( struct regedit * regedit )
{
const char * khelp = " [n] New Key [s] New Subkey [d] Del Key "
" [LEFT] Ascend [RIGHT] Descend " ;
2012-08-20 07:02:51 +04:00
const char * vhelp = " [n] New Value [d] Del Value [ENTER] Edit "
" [b] Edit binary " ;
2012-08-13 09:38:06 +04:00
const char * msg = " KEYS " ;
const char * help = khelp ;
2014-06-03 08:50:01 +04:00
const char * genhelp = " [TAB] Switch sections [q] Quit "
" [UP] List up [DOWN] List down "
" [/] Search [x] Next " ;
2012-08-13 09:38:06 +04:00
int i , pad ;
if ( ! regedit - > tree_input ) {
msg = " VALUES " ;
help = vhelp ;
}
2012-08-16 08:05:01 +04:00
move ( HELP1_START_Y , HELP1_START_X ) ;
2012-08-13 09:38:06 +04:00
clrtoeol ( ) ;
2014-05-23 02:23:52 +04:00
attron ( COLOR_PAIR ( PAIR_BLACK_CYAN ) ) ;
2012-08-16 08:05:01 +04:00
mvaddstr ( HELP1_START_Y , HELP1_START_X , help ) ;
2012-08-13 09:38:06 +04:00
pad = COLS - strlen ( msg ) - strlen ( help ) ;
for ( i = 0 ; i < pad ; + + i ) {
addch ( ' ' ) ;
}
2014-05-23 02:23:52 +04:00
attroff ( COLOR_PAIR ( PAIR_BLACK_CYAN ) ) ;
attron ( COLOR_PAIR ( PAIR_YELLOW_CYAN ) | A_BOLD ) ;
2012-08-16 07:50:00 +04:00
addstr ( msg ) ;
2014-05-23 02:23:52 +04:00
attroff ( COLOR_PAIR ( PAIR_YELLOW_CYAN ) | A_BOLD ) ;
2012-08-16 08:05:01 +04:00
move ( HELP2_START_Y , HELP2_START_X ) ;
clrtoeol ( ) ;
mvaddstr ( HELP2_START_Y , HELP2_START_X , genhelp ) ;
2012-08-13 09:38:06 +04:00
}
2012-08-11 07:05:20 +04:00
static void print_heading ( struct regedit * regedit )
2012-07-27 14:37:22 +04:00
{
2012-08-11 07:05:20 +04:00
if ( regedit - > tree_input ) {
2014-05-21 03:17:42 +04:00
tree_view_set_selected ( regedit - > keys , true ) ;
value_list_set_selected ( regedit - > vl , false ) ;
2012-07-27 14:37:22 +04:00
} else {
2014-05-21 03:17:42 +04:00
tree_view_set_selected ( regedit - > keys , false ) ;
value_list_set_selected ( regedit - > vl , true ) ;
2012-07-27 14:37:22 +04:00
}
2012-08-13 09:38:06 +04:00
print_help ( regedit ) ;
2012-07-27 14:37:22 +04:00
}
2012-08-20 05:29:06 +04:00
static void load_values ( struct regedit * regedit )
{
struct tree_node * node ;
2014-06-03 08:50:01 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2012-08-20 05:29:06 +04:00
value_list_load ( regedit - > vl , node - > key ) ;
}
2012-08-11 00:16:20 +04:00
static void add_reg_key ( struct regedit * regedit , struct tree_node * node ,
bool subkey )
{
2014-08-05 08:48:04 +04:00
const char * name ;
2012-08-11 00:16:20 +04:00
const char * msg ;
2014-07-01 10:19:53 +04:00
if ( ! subkey & & tree_node_is_top_level ( node ) ) {
2012-08-11 00:16:20 +04:00
return ;
}
msg = " Enter name of new key " ;
if ( subkey ) {
msg = " Enter name of new subkey " ;
}
2012-08-11 08:56:58 +04:00
dialog_input ( regedit , & name , " New Key " , msg ) ;
2012-08-11 00:16:20 +04:00
if ( name ) {
WERROR rv ;
struct registry_key * new_key ;
struct tree_node * new_node ;
struct tree_node * list ;
struct tree_node * parent ;
if ( subkey ) {
parent = node ;
list = node - > child_head ;
} else {
parent = node - > parent ;
list = tree_node_first ( node ) ;
SMB_ASSERT ( list ! = NULL ) ;
}
rv = reg_key_add_name ( regedit , parent - > key , name ,
NULL , NULL , & new_key ) ;
if ( W_ERROR_IS_OK ( rv ) ) {
/* The list of subkeys may not be present in
cache yet , so if not , don ' t bother allocating
a new node for the key . */
if ( list ) {
new_node = tree_node_new ( parent , parent ,
name , new_key ) ;
SMB_ASSERT ( new_node ) ;
2014-06-03 08:50:01 +04:00
tree_node_insert_sorted ( list , new_node ) ;
2014-06-28 05:01:36 +04:00
} else {
/* Reopen the parent key to make sure the
new subkey will be noticed . */
2014-07-28 04:39:06 +04:00
tree_node_reopen_key ( regedit - > registry_context ,
parent ) ;
2012-08-11 00:16:20 +04:00
}
list = tree_node_first ( node ) ;
tree_view_clear ( regedit - > keys ) ;
tree_view_update ( regedit - > keys , list ) ;
2014-06-26 18:51:22 +04:00
if ( ! subkey ) {
node = new_node ;
}
tree_view_set_current_node ( regedit - > keys , node ) ;
2012-08-11 00:16:20 +04:00
} else {
2014-06-26 08:55:27 +04:00
msg = get_friendly_werror_msg ( rv ) ;
2012-08-11 00:16:20 +04:00
dialog_notice ( regedit , DIA_ALERT , " New Key " ,
2014-06-26 08:55:27 +04:00
" Failed to create key: %s " , msg ) ;
2012-08-11 00:16:20 +04:00
}
2014-08-05 08:48:04 +04:00
talloc_free ( discard_const ( name ) ) ;
2012-08-11 00:16:20 +04:00
}
}
2014-06-03 08:50:01 +04:00
static WERROR next_depth_first ( struct tree_node * * node )
{
WERROR rv = WERR_OK ;
SMB_ASSERT ( node ! = NULL & & * node ! = NULL ) ;
if ( tree_node_has_children ( * node ) ) {
/* 1. If the node has children, go to the first one. */
rv = tree_node_load_children ( * node ) ;
if ( W_ERROR_IS_OK ( rv ) ) {
SMB_ASSERT ( ( * node ) - > child_head ! = NULL ) ;
* node = ( * node ) - > child_head ;
}
} else if ( ( * node ) - > next ) {
/* 2. If there's a node directly after this one, go there */
* node = ( * node ) - > next ;
} else {
/* 3. Otherwise, go up the hierarchy to find the next one */
do {
* node = ( * node ) - > parent ;
if ( * node & & ( * node ) - > next ) {
* node = ( * node ) - > next ;
break ;
}
} while ( * node ) ;
}
return rv ;
}
static WERROR regedit_search_next ( struct regedit * regedit )
{
WERROR rv ;
struct regedit_search_opts * opts = & regedit - > active_search ;
if ( opts - > search_recursive ) {
rv = next_depth_first ( & opts - > node ) ;
if ( ! W_ERROR_IS_OK ( rv ) ) {
return rv ;
}
} else {
opts - > node = opts - > node - > next ;
}
return WERR_OK ;
}
static WERROR regedit_search ( struct regedit * regedit )
{
struct regedit_search_opts * opts ;
struct tree_node * found ;
WERROR rv ;
opts = & regedit - > active_search ;
if ( ! opts - > query | | ! opts - > match ) {
return WERR_OK ;
}
SMB_ASSERT ( opts - > search_key | | opts - > search_value ) ;
for ( found = NULL ; opts - > node & & ! found ; ) {
if ( opts - > search_key & &
opts - > match ( opts - > node - > name , opts - > query ) ) {
found = opts - > node ;
}
if ( opts - > search_value ) {
/* TODO
rv = regedit_search_value ( regedit ) ;
if ( W_ERROR_IS_OK ( rv ) ) {
found = opts - > node ;
} else if ( ! W_ERROR_EQUAL ( rv , WERR_NO_MORE_ITEMS ) ) {
return rv ;
}
*/
}
rv = regedit_search_next ( regedit ) ;
if ( ! W_ERROR_IS_OK ( rv ) ) {
return rv ;
}
}
if ( found ) {
/* Put the cursor on the node that was found */
if ( ! tree_view_is_node_visible ( regedit - > keys , found ) ) {
tree_view_update ( regedit - > keys ,
tree_node_first ( found ) ) ;
print_path ( regedit , found ) ;
}
tree_view_set_current_node ( regedit - > keys , found ) ;
load_values ( regedit ) ;
tree_view_show ( regedit - > keys ) ;
value_list_show ( regedit - > vl ) ;
} else {
beep ( ) ;
}
return WERR_OK ;
}
2012-07-27 14:37:22 +04:00
static void handle_tree_input ( struct regedit * regedit , int c )
2012-07-24 15:12:56 +04:00
{
struct tree_node * node ;
switch ( c ) {
case KEY_DOWN :
2014-06-10 05:29:56 +04:00
tree_view_driver ( regedit - > keys , ML_CURSOR_DOWN ) ;
2012-08-20 05:29:06 +04:00
load_values ( regedit ) ;
2012-07-24 15:12:56 +04:00
break ;
case KEY_UP :
2014-06-10 05:29:56 +04:00
tree_view_driver ( regedit - > keys , ML_CURSOR_UP ) ;
2012-08-20 05:29:06 +04:00
load_values ( regedit ) ;
2012-07-24 15:12:56 +04:00
break ;
2012-07-26 12:56:31 +04:00
case ' \n ' :
2012-07-24 15:12:56 +04:00
case KEY_ENTER :
case KEY_RIGHT :
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2012-07-24 15:12:56 +04:00
if ( node & & tree_node_has_children ( node ) ) {
2014-06-26 08:48:52 +04:00
WERROR rv ;
rv = tree_node_load_children ( node ) ;
if ( W_ERROR_IS_OK ( rv ) ) {
print_path ( regedit , node - > child_head ) ;
tree_view_update ( regedit - > keys , node - > child_head ) ;
value_list_load ( regedit - > vl , node - > child_head - > key ) ;
} else {
2014-06-26 08:55:27 +04:00
const char * msg = get_friendly_werror_msg ( rv ) ;
2014-06-26 08:48:52 +04:00
dialog_notice ( regedit , DIA_ALERT , " Loading Subkeys " ,
2014-06-26 08:55:27 +04:00
" Failed to load subkeys: %s " , msg ) ;
2014-06-26 08:48:52 +04:00
}
2012-07-24 15:12:56 +04:00
}
break ;
case KEY_LEFT :
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2014-07-01 10:19:53 +04:00
if ( node & & ! tree_node_is_top_level ( node ) ) {
2012-08-12 09:10:51 +04:00
print_path ( regedit , node - > parent ) ;
2014-06-26 09:51:27 +04:00
node = node - > parent ;
tree_view_update ( regedit - > keys , tree_node_first ( node ) ) ;
tree_view_set_current_node ( regedit - > keys , node ) ;
2012-07-27 14:37:22 +04:00
value_list_load ( regedit - > vl , node - > key ) ;
2012-07-24 15:12:56 +04:00
}
break ;
2012-08-10 05:14:42 +04:00
case ' n ' :
2012-08-11 00:16:20 +04:00
case ' N ' :
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2012-08-11 00:16:20 +04:00
add_reg_key ( regedit , node , false ) ;
break ;
case ' s ' :
case ' S ' :
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2012-08-11 00:16:20 +04:00
add_reg_key ( regedit , node , true ) ;
2012-08-10 05:14:42 +04:00
break ;
2012-07-27 14:37:22 +04:00
case ' d ' :
2012-08-07 06:11:13 +04:00
case ' D ' : {
int sel ;
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2014-07-01 10:19:53 +04:00
if ( tree_node_is_top_level ( node ) ) {
2012-08-10 05:14:42 +04:00
break ;
}
2012-08-09 02:36:01 +04:00
sel = dialog_notice ( regedit , DIA_CONFIRM ,
" Delete Key " ,
" Really delete key \" %s \" ? " ,
node - > name ) ;
2012-08-10 05:14:42 +04:00
if ( sel = = DIALOG_OK ) {
WERROR rv ;
struct tree_node * pop ;
struct tree_node * parent = node - > parent ;
rv = reg_key_del ( node , parent - > key , node - > name ) ;
if ( W_ERROR_IS_OK ( rv ) ) {
2014-07-28 04:39:06 +04:00
tree_node_reopen_key ( regedit - > registry_context ,
parent ) ;
2012-08-10 05:14:42 +04:00
tree_view_clear ( regedit - > keys ) ;
pop = tree_node_pop ( & node ) ;
2014-07-01 10:14:20 +04:00
talloc_free ( pop ) ;
2012-08-10 05:14:42 +04:00
node = parent - > child_head ;
if ( node = = NULL ) {
node = tree_node_first ( parent ) ;
2012-08-12 09:10:51 +04:00
print_path ( regedit , node ) ;
2012-08-10 05:14:42 +04:00
}
tree_view_update ( regedit - > keys , node ) ;
value_list_load ( regedit - > vl , node - > key ) ;
} else {
2014-06-26 08:55:27 +04:00
const char * msg = get_friendly_werror_msg ( rv ) ;
2012-08-10 05:14:42 +04:00
dialog_notice ( regedit , DIA_ALERT , " Delete Key " ,
2014-06-26 08:55:27 +04:00
" Failed to delete key: %s " , msg ) ;
2012-08-10 05:14:42 +04:00
}
}
2012-07-27 14:37:22 +04:00
break ;
2012-07-24 15:12:56 +04:00
}
2012-08-07 06:11:13 +04:00
}
2012-07-27 14:37:22 +04:00
tree_view_show ( regedit - > keys ) ;
value_list_show ( regedit - > vl ) ;
2012-07-24 15:12:56 +04:00
}
2012-07-27 14:37:22 +04:00
static void handle_value_input ( struct regedit * regedit , int c )
2012-07-24 15:12:56 +04:00
{
2012-07-27 14:37:22 +04:00
struct value_item * vitem ;
2012-08-20 07:02:51 +04:00
bool binmode = false ;
2014-07-27 06:49:33 +04:00
WERROR err ;
int sel ;
2012-07-27 14:37:22 +04:00
2012-07-24 15:12:56 +04:00
switch ( c ) {
case KEY_DOWN :
2014-06-10 21:35:19 +04:00
value_list_driver ( regedit - > vl , ML_CURSOR_DOWN ) ;
2012-07-24 15:12:56 +04:00
break ;
case KEY_UP :
2014-06-10 21:35:19 +04:00
value_list_driver ( regedit - > vl , ML_CURSOR_UP ) ;
2012-07-27 14:37:22 +04:00
break ;
2012-08-20 07:02:51 +04:00
case ' b ' :
case ' B ' :
binmode = true ;
/* Falthrough... */
2012-08-07 06:11:13 +04:00
case ' \n ' :
2012-07-27 14:37:22 +04:00
case KEY_ENTER :
2014-06-10 21:35:19 +04:00
vitem = value_list_get_current_item ( regedit - > vl ) ;
2012-08-07 06:11:13 +04:00
if ( vitem ) {
struct tree_node * node ;
2014-07-27 06:49:33 +04:00
const char * name = NULL ;
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2014-07-27 06:49:33 +04:00
sel = dialog_edit_value ( regedit , node - > key , vitem - > type ,
vitem , binmode , & err , & name ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
const char * msg = get_friendly_werror_msg ( err ) ;
dialog_notice ( regedit , DIA_ALERT , " Error " ,
" Error editing value: \n %s " , msg ) ;
} else if ( sel = = DIALOG_OK ) {
2014-07-28 04:39:06 +04:00
tree_node_reopen_key ( regedit - > registry_context ,
node ) ;
2014-07-27 06:49:33 +04:00
value_list_load ( regedit - > vl , node - > key ) ;
value_list_set_current_item_by_name ( regedit - > vl ,
name ) ;
talloc_free ( discard_const ( name ) ) ;
}
2012-08-07 06:11:13 +04:00
}
2012-07-24 15:12:56 +04:00
break ;
2012-08-08 23:09:03 +04:00
case ' n ' :
case ' N ' : {
int new_type ;
2012-08-11 08:56:58 +04:00
sel = dialog_select_type ( regedit , & new_type ) ;
2012-08-09 01:16:41 +04:00
if ( sel = = DIALOG_OK ) {
struct tree_node * node ;
2014-07-27 06:49:33 +04:00
const char * name = NULL ;
2014-06-10 05:29:56 +04:00
node = tree_view_get_current_node ( regedit - > keys ) ;
2014-07-27 06:49:33 +04:00
sel = dialog_edit_value ( regedit , node - > key , new_type ,
NULL , false , & err , & name ) ;
if ( ! W_ERROR_IS_OK ( err ) ) {
const char * msg = get_friendly_werror_msg ( err ) ;
dialog_notice ( regedit , DIA_ALERT , " Error " ,
" Error creating value: \n %s " , msg ) ;
} else if ( sel = = DIALOG_OK ) {
2014-07-28 04:39:06 +04:00
tree_node_reopen_key ( regedit - > registry_context ,
node ) ;
2014-07-27 06:49:33 +04:00
value_list_load ( regedit - > vl , node - > key ) ;
value_list_set_current_item_by_name ( regedit - > vl ,
name ) ;
talloc_free ( discard_const ( name ) ) ;
}
2012-08-08 23:09:03 +04:00
}
break ;
}
2012-07-27 14:37:22 +04:00
case ' d ' :
case ' D ' :
2014-06-10 21:35:19 +04:00
vitem = value_list_get_current_item ( regedit - > vl ) ;
2012-07-27 14:37:22 +04:00
if ( vitem ) {
2012-08-09 02:36:01 +04:00
sel = dialog_notice ( regedit , DIA_CONFIRM ,
" Delete Value " ,
" Really delete value \" %s \" ? " ,
vitem - > value_name ) ;
2012-08-09 01:16:41 +04:00
if ( sel = = DIALOG_OK ) {
2014-06-10 05:29:56 +04:00
struct tree_node * node ;
node = tree_view_get_current_node ( regedit - > keys ) ;
2012-08-09 01:16:41 +04:00
reg_del_value ( regedit , node - > key ,
vitem - > value_name ) ;
2014-07-28 04:39:06 +04:00
tree_node_reopen_key ( regedit - > registry_context ,
node ) ;
2012-08-09 01:16:41 +04:00
value_list_load ( regedit - > vl , node - > key ) ;
}
2012-07-27 14:37:22 +04:00
}
break ;
}
value_list_show ( regedit - > vl ) ;
}
2014-06-03 08:50:01 +04:00
static bool find_substring ( const char * haystack , const char * needle )
{
return strstr ( haystack , needle ) ! = NULL ;
}
static bool find_substring_nocase ( const char * haystack , const char * needle )
{
return strcasestr ( haystack , needle ) ! = NULL ;
}
2012-07-27 14:37:22 +04:00
static void handle_main_input ( struct regedit * regedit , int c )
2012-07-24 15:12:56 +04:00
{
2012-07-27 14:37:22 +04:00
switch ( c ) {
2014-07-01 10:14:20 +04:00
case 18 : { /* CTRL-R */
struct tree_node * root , * node ;
const char * * path ;
node = tree_view_get_current_node ( regedit - > keys ) ;
path = tree_node_get_path ( regedit , node ) ;
2014-07-01 10:19:53 +04:00
SMB_ASSERT ( path ! = NULL ) ;
root = tree_node_new_root ( regedit , regedit - > registry_context ) ;
SMB_ASSERT ( root ! = NULL ) ;
2014-07-01 10:14:20 +04:00
tree_view_set_root ( regedit - > keys , root ) ;
tree_view_set_path ( regedit - > keys , path ) ;
node = tree_view_get_current_node ( regedit - > keys ) ;
value_list_load ( regedit - > vl , node - > key ) ;
tree_view_show ( regedit - > keys ) ;
value_list_show ( regedit - > vl ) ;
print_path ( regedit , node ) ;
talloc_free ( discard_const ( path ) ) ;
break ;
}
2014-06-03 08:50:01 +04:00
case ' f ' :
case ' F ' :
case ' / ' : {
int rv ;
struct regedit_search_opts * opts ;
opts = & regedit - > active_search ;
if ( opts - > query ) {
talloc_free ( discard_const ( opts - > query ) ) ;
}
rv = dialog_search_input ( regedit , opts ) ;
if ( rv = = DIALOG_OK ) {
SMB_ASSERT ( opts - > query ! = NULL ) ;
opts - > match = find_substring ;
opts - > node = regedit - > keys - > root ;
if ( opts - > search_nocase ) {
opts - > match = find_substring_nocase ;
}
if ( opts - > search_relative ) {
opts - > node =
tree_view_get_current_node ( regedit - > keys ) ;
}
regedit_search ( regedit ) ;
}
break ;
}
case ' x ' :
case ' X ' :
regedit_search ( regedit ) ;
break ;
2012-07-27 14:37:22 +04:00
case ' \t ' :
regedit - > tree_input = ! regedit - > tree_input ;
2012-08-11 07:05:20 +04:00
print_heading ( regedit ) ;
2012-07-27 14:37:22 +04:00
break ;
default :
if ( regedit - > tree_input ) {
handle_tree_input ( regedit , c ) ;
} else {
handle_value_input ( regedit , c ) ;
}
2012-07-24 15:12:56 +04:00
}
}
2012-08-11 08:56:58 +04:00
int regedit_getch ( void )
{
int c ;
SMB_ASSERT ( regedit_main ) ;
c = getch ( ) ;
if ( c = = KEY_RESIZE ) {
tree_view_resize ( regedit_main - > keys , KEY_HEIGHT , KEY_WIDTH ,
KEY_START_Y , KEY_START_X ) ;
value_list_resize ( regedit_main - > vl , VAL_HEIGHT , VAL_WIDTH ,
VAL_START_Y , VAL_START_X ) ;
print_heading ( regedit_main ) ;
2012-08-12 10:29:13 +04:00
show_path ( regedit_main ) ;
2012-08-11 08:56:58 +04:00
}
return c ;
}
2014-07-02 03:00:16 +04:00
static void regedit_panic_handler ( const char * msg )
{
endwin ( ) ;
smb_panic_s3 ( msg ) ;
}
2012-08-11 08:56:58 +04:00
static void display_window ( TALLOC_CTX * mem_ctx , struct registry_context * ctx )
2012-07-13 08:11:55 +04:00
{
2012-07-27 14:37:22 +04:00
struct regedit * regedit ;
2012-07-24 15:12:56 +04:00
struct tree_node * root ;
2012-10-26 21:52:08 +04:00
bool colors ;
2012-10-26 21:21:50 +04:00
int key ;
2012-07-13 08:11:55 +04:00
initscr ( ) ;
2012-10-26 21:52:08 +04:00
2012-07-13 08:11:55 +04:00
cbreak ( ) ;
noecho ( ) ;
2014-07-02 03:00:16 +04:00
fault_configure ( regedit_panic_handler ) ;
2012-10-26 21:52:08 +04:00
colors = has_colors ( ) ;
if ( colors ) {
start_color ( ) ;
use_default_colors ( ) ;
2014-05-19 22:08:09 +04:00
assume_default_colors ( COLOR_WHITE , COLOR_BLUE ) ;
2014-05-23 02:23:52 +04:00
init_pair ( PAIR_YELLOW_CYAN , COLOR_YELLOW , COLOR_CYAN ) ;
init_pair ( PAIR_BLACK_CYAN , COLOR_BLACK , COLOR_CYAN ) ;
2014-06-10 05:29:56 +04:00
init_pair ( PAIR_YELLOW_BLUE , COLOR_YELLOW , COLOR_BLUE ) ;
2012-10-26 21:52:08 +04:00
}
2012-07-27 14:37:22 +04:00
regedit = talloc_zero ( mem_ctx , struct regedit ) ;
SMB_ASSERT ( regedit ! = NULL ) ;
2012-08-11 08:56:58 +04:00
regedit_main = regedit ;
2012-07-27 14:37:22 +04:00
2014-07-01 10:14:20 +04:00
regedit - > registry_context = ctx ;
2012-08-11 07:05:20 +04:00
regedit - > main_window = stdscr ;
2012-07-27 14:37:22 +04:00
keypad ( regedit - > main_window , TRUE ) ;
2012-07-13 08:11:55 +04:00
2012-07-27 14:37:22 +04:00
mvwprintw ( regedit - > main_window , 0 , 0 , " Path: " ) ;
2012-08-12 09:10:51 +04:00
regedit - > path_label = newpad ( 1 , PATH_WIDTH_MAX ) ;
SMB_ASSERT ( regedit - > path_label ) ;
2012-07-27 14:37:22 +04:00
wprintw ( regedit - > path_label , " / " ) ;
2012-08-12 10:29:13 +04:00
show_path ( regedit_main ) ;
2012-07-14 04:16:51 +04:00
2014-07-01 10:19:53 +04:00
root = tree_node_new_root ( regedit , ctx ) ;
2012-07-13 08:11:55 +04:00
SMB_ASSERT ( root ! = NULL ) ;
2012-08-11 07:05:20 +04:00
regedit - > keys = tree_view_new ( regedit , root , KEY_HEIGHT , KEY_WIDTH ,
KEY_START_Y , KEY_START_X ) ;
2012-07-27 14:37:22 +04:00
SMB_ASSERT ( regedit - > keys ! = NULL ) ;
2012-08-11 07:05:20 +04:00
regedit - > vl = value_list_new ( regedit , VAL_HEIGHT , VAL_WIDTH ,
VAL_START_Y , VAL_START_X ) ;
2012-07-27 14:37:22 +04:00
SMB_ASSERT ( regedit - > vl ! = NULL ) ;
regedit - > tree_input = true ;
2012-08-11 07:05:20 +04:00
print_heading ( regedit ) ;
2012-07-27 14:37:22 +04:00
tree_view_show ( regedit - > keys ) ;
2012-08-20 05:29:06 +04:00
load_values ( regedit ) ;
2012-07-27 14:37:22 +04:00
value_list_show ( regedit - > vl ) ;
update_panels ( ) ;
doupdate ( ) ;
2012-10-26 21:21:50 +04:00
do {
key = regedit_getch ( ) ;
handle_main_input ( regedit , key ) ;
2012-07-27 14:37:22 +04:00
update_panels ( ) ;
doupdate ( ) ;
2012-10-26 21:21:50 +04:00
} while ( key ! = ' q ' | | key = = ' Q ' ) ;
2012-07-13 08:11:55 +04:00
endwin ( ) ;
}
2014-02-26 23:16:26 +04:00
int main ( int argc , const char * * argv )
2012-07-06 14:08:17 +04:00
{
struct poptOption long_options [ ] = {
POPT_AUTOHELP
/* ... */
POPT_COMMON_SAMBA
POPT_COMMON_CONNECTION
POPT_COMMON_CREDENTIALS
POPT_TABLEEND
} ;
int opt ;
poptContext pc ;
struct user_auth_info * auth_info ;
TALLOC_CTX * frame ;
struct registry_context * ctx ;
WERROR rv ;
frame = talloc_stackframe ( ) ;
setup_logging ( " regedit " , DEBUG_DEFAULT_STDERR ) ;
lp_set_cmdline ( " log level " , " 0 " ) ;
/* process options */
auth_info = user_auth_info_init ( frame ) ;
if ( auth_info = = NULL ) {
exit ( 1 ) ;
}
popt_common_set_auth_info ( auth_info ) ;
2014-02-26 23:16:26 +04:00
pc = poptGetContext ( " regedit " , argc , argv , long_options , 0 ) ;
2012-07-06 14:08:17 +04:00
while ( ( opt = poptGetNextOpt ( pc ) ) ! = - 1 ) {
/* TODO */
}
2012-07-13 12:57:02 +04:00
if ( ! lp_load_global ( get_dyn_CONFIGFILE ( ) ) ) {
DEBUG ( 0 , ( " ERROR loading config file... \n " ) ) ;
exit ( 1 ) ;
}
2012-07-06 14:08:17 +04:00
/* some simple tests */
rv = reg_open_samba3 ( frame , & ctx ) ;
2012-08-02 14:41:49 +04:00
if ( ! W_ERROR_IS_OK ( rv ) ) {
TALLOC_FREE ( frame ) ;
return 1 ;
}
2012-07-06 14:08:17 +04:00
2012-08-11 08:56:58 +04:00
display_window ( frame , ctx ) ;
2012-07-06 14:08:17 +04:00
TALLOC_FREE ( frame ) ;
return 0 ;
}