Change read_package_field interface, and rewrite using low level functions
Fixes for a few bugs that have crept into dpkg in the last few days
This commit is contained in:
parent
8d3b0497a4
commit
c3fbec73fb
127
archival/dpkg.c
127
archival/dpkg.c
@ -499,34 +499,17 @@ unsigned int fill_package_struct(char *control_buffer)
|
|||||||
{
|
{
|
||||||
common_node_t *new_node = (common_node_t *) xcalloc(1, sizeof(common_node_t));
|
common_node_t *new_node = (common_node_t *) xcalloc(1, sizeof(common_node_t));
|
||||||
|
|
||||||
char *field;
|
char *field_name = xmalloc(sizeof(char *));
|
||||||
char *field_name;
|
char *field_value = xmalloc(sizeof(char *));
|
||||||
char *field_value;
|
|
||||||
int field_start = 0;
|
int field_start = 0;
|
||||||
int field_length;
|
|
||||||
int seperator_offset;
|
|
||||||
int num = -1;
|
int num = -1;
|
||||||
int buffer_length = strlen(control_buffer);
|
int buffer_length = strlen(control_buffer);
|
||||||
|
|
||||||
new_node->version = search_name_hashtable("unknown");
|
new_node->version = search_name_hashtable("unknown");
|
||||||
while (field_start < buffer_length) {
|
while (field_start < buffer_length) {
|
||||||
field = read_package_field(&control_buffer[field_start]);
|
field_start += read_package_field(&control_buffer[field_start], &field_name, &field_value);
|
||||||
|
|
||||||
/* Setup start point for next field */
|
if (field_name == NULL) {
|
||||||
field_length = strlen(field);
|
|
||||||
field_start += (field_length + 1);
|
|
||||||
|
|
||||||
seperator_offset = strcspn(field, ":");
|
|
||||||
if (seperator_offset == 0) {
|
|
||||||
free(field);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
field_name = xstrndup(field, seperator_offset);
|
|
||||||
field_value = field + seperator_offset + 1;
|
|
||||||
field_value += strspn(field_value, " \n\t");
|
|
||||||
|
|
||||||
/* Should be able to replace this strlen with pointer arithmatic */
|
|
||||||
if (strlen(field_value) == 0) {
|
|
||||||
goto fill_package_struct_cleanup; // Oh no, the dreaded goto statement !!
|
goto fill_package_struct_cleanup; // Oh no, the dreaded goto statement !!
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -562,7 +545,7 @@ unsigned int fill_package_struct(char *control_buffer)
|
|||||||
}
|
}
|
||||||
fill_package_struct_cleanup:
|
fill_package_struct_cleanup:
|
||||||
free(field_name);
|
free(field_name);
|
||||||
free(field);
|
free(field_value);
|
||||||
}
|
}
|
||||||
if (new_node->version == search_name_hashtable("unknown")) {
|
if (new_node->version == search_name_hashtable("unknown")) {
|
||||||
free_package(new_node);
|
free_package(new_node);
|
||||||
@ -739,6 +722,23 @@ char *get_depends_field(common_node_t *package, const int depends_type)
|
|||||||
return(depends);
|
return(depends);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void write_buffer_no_status(FILE *new_status_file, const char *control_buffer)
|
||||||
|
{
|
||||||
|
char *name;
|
||||||
|
char *value;
|
||||||
|
int start = 0;
|
||||||
|
while (1) {
|
||||||
|
start += read_package_field(&control_buffer[start], &name, &value);
|
||||||
|
if (name == NULL) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (strcmp(name, "Status") != 0) {
|
||||||
|
fprintf(new_status_file, "%s: %s\n", name, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
/* This could do with a cleanup */
|
/* This could do with a cleanup */
|
||||||
void write_status_file(deb_file_t **deb_file)
|
void write_status_file(deb_file_t **deb_file)
|
||||||
{
|
{
|
||||||
@ -748,11 +748,8 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
char *status_from_file;
|
char *status_from_file;
|
||||||
char *control_buffer = NULL;
|
char *control_buffer = NULL;
|
||||||
char *tmp_string;
|
char *tmp_string;
|
||||||
char *field;
|
|
||||||
int status_num;
|
int status_num;
|
||||||
int field_length;
|
|
||||||
int field_start = 0;
|
int field_start = 0;
|
||||||
int buffer_length;
|
|
||||||
int write_flag;
|
int write_flag;
|
||||||
int i = 0;
|
int i = 0;
|
||||||
|
|
||||||
@ -762,7 +759,6 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
tmp_string += strspn(tmp_string, " \n\t");
|
tmp_string += strspn(tmp_string, " \n\t");
|
||||||
package_name = xstrndup(tmp_string, strcspn(tmp_string, "\n\0"));
|
package_name = xstrndup(tmp_string, strcspn(tmp_string, "\n\0"));
|
||||||
write_flag = FALSE;
|
write_flag = FALSE;
|
||||||
|
|
||||||
tmp_string = strstr(control_buffer, "Status:");
|
tmp_string = strstr(control_buffer, "Status:");
|
||||||
if (tmp_string != NULL) {
|
if (tmp_string != NULL) {
|
||||||
/* Seperate the status value from the control buffer */
|
/* Seperate the status value from the control buffer */
|
||||||
@ -786,20 +782,11 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
i = 0;
|
i = 0;
|
||||||
while(deb_file[i] != NULL) {
|
while(deb_file[i] != NULL) {
|
||||||
if (strcmp(package_name, name_hashtable[package_hashtable[deb_file[i]->package]->name]) == 0) {
|
if (strcmp(package_name, name_hashtable[package_hashtable[deb_file[i]->package]->name]) == 0) {
|
||||||
char *last_char;
|
|
||||||
/* Write a status file entry with a modified status */
|
/* Write a status file entry with a modified status */
|
||||||
/* remove trailing \n's */
|
/* remove trailing \n's */
|
||||||
while(1) {
|
write_buffer_no_status(new_status_file, deb_file[i]->control_file);
|
||||||
last_char = last_char_is(deb_file[i]->control_file, '\n');
|
|
||||||
if (last_char) {
|
|
||||||
*last_char = '\0';
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fputs(deb_file[i]->control_file, new_status_file);
|
|
||||||
set_status(status_num, "ok", 2);
|
set_status(status_num, "ok", 2);
|
||||||
fprintf(new_status_file, "\nStatus: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
fprintf(new_status_file, "Status: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
||||||
write_flag = TRUE;
|
write_flag = TRUE;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -814,16 +801,17 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
/* Only write the Package, Status, Priority and Section lines */
|
/* Only write the Package, Status, Priority and Section lines */
|
||||||
fprintf(new_status_file, "Package: %s\n", package_name);
|
fprintf(new_status_file, "Package: %s\n", package_name);
|
||||||
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
||||||
buffer_length = strlen(control_buffer);
|
|
||||||
while (field_start < buffer_length) {
|
while (1) {
|
||||||
field = read_package_field(&control_buffer[field_start]);
|
char *field_name;
|
||||||
field_length = strlen(field);
|
char *field_value;
|
||||||
field_start += (field_length + 1);
|
field_start += read_package_field(&control_buffer[field_start], &field_name, &field_value);
|
||||||
if (strncmp(field, "Priority:", 9) == 0) {
|
if (field_name == NULL) {
|
||||||
fprintf(new_status_file, "Priority:%s\n", field + 9);
|
break;
|
||||||
}
|
}
|
||||||
if (strncmp(field, "Section:", 8) == 0) {
|
if ((strcmp(field_name, "Priority") == 0) ||
|
||||||
fprintf(new_status_file, "Section:%s\n", field + 8);
|
(strcmp(field_name, "Section") == 0)) {
|
||||||
|
fprintf(new_status_file, "%s: %s\n", field_name, field_value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
write_flag = TRUE;
|
write_flag = TRUE;
|
||||||
@ -831,25 +819,26 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
}
|
}
|
||||||
else if (strcmp("config-files", name_hashtable[state_status]) == 0) {
|
else if (strcmp("config-files", name_hashtable[state_status]) == 0) {
|
||||||
/* only change the status line */
|
/* only change the status line */
|
||||||
buffer_length = strlen(control_buffer);
|
// buffer_length = strlen(control_buffer);
|
||||||
while (field_start < buffer_length) {
|
while (1) {
|
||||||
field = read_package_field(&control_buffer[field_start]);
|
char *field_name;
|
||||||
|
char *field_value;
|
||||||
|
field_start += read_package_field(&control_buffer[field_start], &field_name, &field_value);
|
||||||
|
if (field_name == NULL) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
/* Setup start point for next field */
|
/* Setup start point for next field */
|
||||||
field_length = strlen(field);
|
if (strcmp(field_name, "Status") == 0) {
|
||||||
field_start += (field_length + 1);
|
|
||||||
if (strncmp(field, "Status:", 7) == 0) {
|
|
||||||
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
||||||
} else {
|
} else {
|
||||||
fprintf(new_status_file, "%s\n", field);
|
fprintf(new_status_file, "%s: %s\n", field_name, field_value);
|
||||||
}
|
}
|
||||||
free(field);
|
|
||||||
}
|
}
|
||||||
write_flag = TRUE;
|
write_flag = TRUE;
|
||||||
fputs("\n", new_status_file);
|
fputs("\n", new_status_file);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If the package from the status file wasnt handle above, do it now*/
|
/* If the package from the status file wasnt handle above, do it now*/
|
||||||
if (write_flag == FALSE) {
|
if (write_flag == FALSE) {
|
||||||
fprintf(new_status_file, "%s\n\n", control_buffer);
|
fprintf(new_status_file, "%s\n\n", control_buffer);
|
||||||
@ -866,20 +855,9 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
for(i = 0; deb_file[i] != NULL; i++) {
|
for(i = 0; deb_file[i] != NULL; i++) {
|
||||||
status_num = search_status_hashtable(name_hashtable[package_hashtable[deb_file[i]->package]->name]);
|
status_num = search_status_hashtable(name_hashtable[package_hashtable[deb_file[i]->package]->name]);
|
||||||
if (strcmp("reinstreq", name_hashtable[get_status(status_num, 2)]) == 0) {
|
if (strcmp("reinstreq", name_hashtable[get_status(status_num, 2)]) == 0) {
|
||||||
char *last_char;
|
write_buffer_no_status(new_status_file, deb_file[i]->control_file);
|
||||||
/* remove trailing \n's */
|
|
||||||
while(1) {
|
|
||||||
last_char = last_char_is(deb_file[i]->control_file, '\n');
|
|
||||||
if (last_char) {
|
|
||||||
*last_char = '\0';
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fputs(deb_file[i]->control_file, new_status_file);
|
|
||||||
set_status(status_num, "ok", 2);
|
set_status(status_num, "ok", 2);
|
||||||
fprintf(new_status_file, "\nStatus: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
fprintf(new_status_file, "Status: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fclose(old_status_file);
|
fclose(old_status_file);
|
||||||
@ -1037,9 +1015,11 @@ char **create_list(const char *filename)
|
|||||||
int length = 0;
|
int length = 0;
|
||||||
int count = 0;
|
int count = 0;
|
||||||
|
|
||||||
list_stream = xfopen(filename, "r");
|
/* dont use [xw]fopen here, handle error ourself */
|
||||||
|
list_stream = fopen(filename, "r");
|
||||||
if (list_stream == NULL) {
|
if (list_stream == NULL) {
|
||||||
return(NULL);
|
*file_list = NULL;
|
||||||
|
return(file_list);
|
||||||
}
|
}
|
||||||
while (getline(&line, &length, list_stream) != -1) {
|
while (getline(&line, &length, list_stream) != -1) {
|
||||||
file_list = xrealloc(file_list, sizeof(char *) * (length + 1));
|
file_list = xrealloc(file_list, sizeof(char *) * (length + 1));
|
||||||
@ -1053,6 +1033,7 @@ char **create_list(const char *filename)
|
|||||||
length = 0;
|
length = 0;
|
||||||
}
|
}
|
||||||
fclose(list_stream);
|
fclose(list_stream);
|
||||||
|
|
||||||
if (count == 0) {
|
if (count == 0) {
|
||||||
return(NULL);
|
return(NULL);
|
||||||
} else {
|
} else {
|
||||||
@ -1212,6 +1193,7 @@ void purge_package(const unsigned int package_num)
|
|||||||
if (run_package_script(package_name, "postrm") == -1) {
|
if (run_package_script(package_name, "postrm") == -1) {
|
||||||
error_msg_and_die("postrm fialure.. set status to what?");
|
error_msg_and_die("postrm fialure.. set status to what?");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change package status */
|
/* Change package status */
|
||||||
set_status(status_num, "purge", 1);
|
set_status(status_num, "purge", 1);
|
||||||
set_status(status_num, "not-installed", 3);
|
set_status(status_num, "not-installed", 3);
|
||||||
@ -1219,7 +1201,6 @@ void purge_package(const unsigned int package_num)
|
|||||||
|
|
||||||
void unpack_package(deb_file_t *deb_file)
|
void unpack_package(deb_file_t *deb_file)
|
||||||
{
|
{
|
||||||
// const unsigned int package_name_num = package_hashtable[deb_file->package]->name;
|
|
||||||
const char *package_name = name_hashtable[package_hashtable[deb_file->package]->name];
|
const char *package_name = name_hashtable[package_hashtable[deb_file->package]->name];
|
||||||
const unsigned int status_num = search_status_hashtable(package_name);
|
const unsigned int status_num = search_status_hashtable(package_name);
|
||||||
const unsigned int status_package_num = status_hashtable[status_num]->status;
|
const unsigned int status_package_num = status_hashtable[status_num]->status;
|
||||||
@ -1248,7 +1229,6 @@ void unpack_package(deb_file_t *deb_file)
|
|||||||
|
|
||||||
/* Create the list file */
|
/* Create the list file */
|
||||||
strcat(info_prefix, "list");
|
strcat(info_prefix, "list");
|
||||||
|
|
||||||
out_stream = xfopen(info_prefix, "w");
|
out_stream = xfopen(info_prefix, "w");
|
||||||
deb_extract(deb_file->filename, out_stream, (extract_quiet | extract_data_tar_gz | extract_list), NULL, NULL);
|
deb_extract(deb_file->filename, out_stream, (extract_quiet | extract_data_tar_gz | extract_list), NULL, NULL);
|
||||||
fclose(out_stream);
|
fclose(out_stream);
|
||||||
@ -1393,7 +1373,7 @@ extern int dpkg_main(int argc, char **argv)
|
|||||||
/* TODO: check dependencies before removing */
|
/* TODO: check dependencies before removing */
|
||||||
if ((dpkg_opt & dpkg_opt_force_ignore_depends) != dpkg_opt_force_ignore_depends) {
|
if ((dpkg_opt & dpkg_opt_force_ignore_depends) != dpkg_opt_force_ignore_depends) {
|
||||||
if (!check_deps(deb_file, 0, deb_count)) {
|
if (!check_deps(deb_file, 0, deb_count)) {
|
||||||
error_msg_and_die("Dependency check fialed");
|
error_msg_and_die("Dependency check failed");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1416,6 +1396,7 @@ extern int dpkg_main(int argc, char **argv)
|
|||||||
configure_package(deb_file[i]);
|
configure_package(deb_file[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
write_status_file(deb_file);
|
write_status_file(deb_file);
|
||||||
|
|
||||||
for (i = 0; i < NAME_HASH_PRIME; i++) {
|
for (i = 0; i < NAME_HASH_PRIME; i++) {
|
||||||
|
@ -110,12 +110,17 @@ extern int dpkg_deb_main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
else if (arg_type == arg_type_field) {
|
else if (arg_type == arg_type_field) {
|
||||||
char *field = NULL;
|
char *field = NULL;
|
||||||
|
char *name;
|
||||||
|
char *value;
|
||||||
int field_start = 0;
|
int field_start = 0;
|
||||||
|
|
||||||
while ((field = read_package_field(&output_buffer[field_start])) != NULL) {
|
while (1) {
|
||||||
field_start += (strlen(field) + 1);
|
field_start += read_package_field(&output_buffer[field_start], &name, &value);
|
||||||
if (strstr(field, argv[optind + 1]) == field) {
|
if (name == NULL) {
|
||||||
puts(field + strlen(argv[optind + 1]) + 2);
|
break;
|
||||||
|
}
|
||||||
|
if (strcmp(name, argv[optind + 1]) == 0) {
|
||||||
|
puts(value);
|
||||||
}
|
}
|
||||||
free(field);
|
free(field);
|
||||||
}
|
}
|
||||||
|
127
dpkg.c
127
dpkg.c
@ -499,34 +499,17 @@ unsigned int fill_package_struct(char *control_buffer)
|
|||||||
{
|
{
|
||||||
common_node_t *new_node = (common_node_t *) xcalloc(1, sizeof(common_node_t));
|
common_node_t *new_node = (common_node_t *) xcalloc(1, sizeof(common_node_t));
|
||||||
|
|
||||||
char *field;
|
char *field_name = xmalloc(sizeof(char *));
|
||||||
char *field_name;
|
char *field_value = xmalloc(sizeof(char *));
|
||||||
char *field_value;
|
|
||||||
int field_start = 0;
|
int field_start = 0;
|
||||||
int field_length;
|
|
||||||
int seperator_offset;
|
|
||||||
int num = -1;
|
int num = -1;
|
||||||
int buffer_length = strlen(control_buffer);
|
int buffer_length = strlen(control_buffer);
|
||||||
|
|
||||||
new_node->version = search_name_hashtable("unknown");
|
new_node->version = search_name_hashtable("unknown");
|
||||||
while (field_start < buffer_length) {
|
while (field_start < buffer_length) {
|
||||||
field = read_package_field(&control_buffer[field_start]);
|
field_start += read_package_field(&control_buffer[field_start], &field_name, &field_value);
|
||||||
|
|
||||||
/* Setup start point for next field */
|
if (field_name == NULL) {
|
||||||
field_length = strlen(field);
|
|
||||||
field_start += (field_length + 1);
|
|
||||||
|
|
||||||
seperator_offset = strcspn(field, ":");
|
|
||||||
if (seperator_offset == 0) {
|
|
||||||
free(field);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
field_name = xstrndup(field, seperator_offset);
|
|
||||||
field_value = field + seperator_offset + 1;
|
|
||||||
field_value += strspn(field_value, " \n\t");
|
|
||||||
|
|
||||||
/* Should be able to replace this strlen with pointer arithmatic */
|
|
||||||
if (strlen(field_value) == 0) {
|
|
||||||
goto fill_package_struct_cleanup; // Oh no, the dreaded goto statement !!
|
goto fill_package_struct_cleanup; // Oh no, the dreaded goto statement !!
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -562,7 +545,7 @@ unsigned int fill_package_struct(char *control_buffer)
|
|||||||
}
|
}
|
||||||
fill_package_struct_cleanup:
|
fill_package_struct_cleanup:
|
||||||
free(field_name);
|
free(field_name);
|
||||||
free(field);
|
free(field_value);
|
||||||
}
|
}
|
||||||
if (new_node->version == search_name_hashtable("unknown")) {
|
if (new_node->version == search_name_hashtable("unknown")) {
|
||||||
free_package(new_node);
|
free_package(new_node);
|
||||||
@ -739,6 +722,23 @@ char *get_depends_field(common_node_t *package, const int depends_type)
|
|||||||
return(depends);
|
return(depends);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void write_buffer_no_status(FILE *new_status_file, const char *control_buffer)
|
||||||
|
{
|
||||||
|
char *name;
|
||||||
|
char *value;
|
||||||
|
int start = 0;
|
||||||
|
while (1) {
|
||||||
|
start += read_package_field(&control_buffer[start], &name, &value);
|
||||||
|
if (name == NULL) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (strcmp(name, "Status") != 0) {
|
||||||
|
fprintf(new_status_file, "%s: %s\n", name, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
/* This could do with a cleanup */
|
/* This could do with a cleanup */
|
||||||
void write_status_file(deb_file_t **deb_file)
|
void write_status_file(deb_file_t **deb_file)
|
||||||
{
|
{
|
||||||
@ -748,11 +748,8 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
char *status_from_file;
|
char *status_from_file;
|
||||||
char *control_buffer = NULL;
|
char *control_buffer = NULL;
|
||||||
char *tmp_string;
|
char *tmp_string;
|
||||||
char *field;
|
|
||||||
int status_num;
|
int status_num;
|
||||||
int field_length;
|
|
||||||
int field_start = 0;
|
int field_start = 0;
|
||||||
int buffer_length;
|
|
||||||
int write_flag;
|
int write_flag;
|
||||||
int i = 0;
|
int i = 0;
|
||||||
|
|
||||||
@ -762,7 +759,6 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
tmp_string += strspn(tmp_string, " \n\t");
|
tmp_string += strspn(tmp_string, " \n\t");
|
||||||
package_name = xstrndup(tmp_string, strcspn(tmp_string, "\n\0"));
|
package_name = xstrndup(tmp_string, strcspn(tmp_string, "\n\0"));
|
||||||
write_flag = FALSE;
|
write_flag = FALSE;
|
||||||
|
|
||||||
tmp_string = strstr(control_buffer, "Status:");
|
tmp_string = strstr(control_buffer, "Status:");
|
||||||
if (tmp_string != NULL) {
|
if (tmp_string != NULL) {
|
||||||
/* Seperate the status value from the control buffer */
|
/* Seperate the status value from the control buffer */
|
||||||
@ -786,20 +782,11 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
i = 0;
|
i = 0;
|
||||||
while(deb_file[i] != NULL) {
|
while(deb_file[i] != NULL) {
|
||||||
if (strcmp(package_name, name_hashtable[package_hashtable[deb_file[i]->package]->name]) == 0) {
|
if (strcmp(package_name, name_hashtable[package_hashtable[deb_file[i]->package]->name]) == 0) {
|
||||||
char *last_char;
|
|
||||||
/* Write a status file entry with a modified status */
|
/* Write a status file entry with a modified status */
|
||||||
/* remove trailing \n's */
|
/* remove trailing \n's */
|
||||||
while(1) {
|
write_buffer_no_status(new_status_file, deb_file[i]->control_file);
|
||||||
last_char = last_char_is(deb_file[i]->control_file, '\n');
|
|
||||||
if (last_char) {
|
|
||||||
*last_char = '\0';
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fputs(deb_file[i]->control_file, new_status_file);
|
|
||||||
set_status(status_num, "ok", 2);
|
set_status(status_num, "ok", 2);
|
||||||
fprintf(new_status_file, "\nStatus: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
fprintf(new_status_file, "Status: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
||||||
write_flag = TRUE;
|
write_flag = TRUE;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@ -814,16 +801,17 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
/* Only write the Package, Status, Priority and Section lines */
|
/* Only write the Package, Status, Priority and Section lines */
|
||||||
fprintf(new_status_file, "Package: %s\n", package_name);
|
fprintf(new_status_file, "Package: %s\n", package_name);
|
||||||
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
||||||
buffer_length = strlen(control_buffer);
|
|
||||||
while (field_start < buffer_length) {
|
while (1) {
|
||||||
field = read_package_field(&control_buffer[field_start]);
|
char *field_name;
|
||||||
field_length = strlen(field);
|
char *field_value;
|
||||||
field_start += (field_length + 1);
|
field_start += read_package_field(&control_buffer[field_start], &field_name, &field_value);
|
||||||
if (strncmp(field, "Priority:", 9) == 0) {
|
if (field_name == NULL) {
|
||||||
fprintf(new_status_file, "Priority:%s\n", field + 9);
|
break;
|
||||||
}
|
}
|
||||||
if (strncmp(field, "Section:", 8) == 0) {
|
if ((strcmp(field_name, "Priority") == 0) ||
|
||||||
fprintf(new_status_file, "Section:%s\n", field + 8);
|
(strcmp(field_name, "Section") == 0)) {
|
||||||
|
fprintf(new_status_file, "%s: %s\n", field_name, field_value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
write_flag = TRUE;
|
write_flag = TRUE;
|
||||||
@ -831,25 +819,26 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
}
|
}
|
||||||
else if (strcmp("config-files", name_hashtable[state_status]) == 0) {
|
else if (strcmp("config-files", name_hashtable[state_status]) == 0) {
|
||||||
/* only change the status line */
|
/* only change the status line */
|
||||||
buffer_length = strlen(control_buffer);
|
// buffer_length = strlen(control_buffer);
|
||||||
while (field_start < buffer_length) {
|
while (1) {
|
||||||
field = read_package_field(&control_buffer[field_start]);
|
char *field_name;
|
||||||
|
char *field_value;
|
||||||
|
field_start += read_package_field(&control_buffer[field_start], &field_name, &field_value);
|
||||||
|
if (field_name == NULL) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
/* Setup start point for next field */
|
/* Setup start point for next field */
|
||||||
field_length = strlen(field);
|
if (strcmp(field_name, "Status") == 0) {
|
||||||
field_start += (field_length + 1);
|
|
||||||
if (strncmp(field, "Status:", 7) == 0) {
|
|
||||||
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
fprintf(new_status_file, "Status: %s\n", status_from_hashtable);
|
||||||
} else {
|
} else {
|
||||||
fprintf(new_status_file, "%s\n", field);
|
fprintf(new_status_file, "%s: %s\n", field_name, field_value);
|
||||||
}
|
}
|
||||||
free(field);
|
|
||||||
}
|
}
|
||||||
write_flag = TRUE;
|
write_flag = TRUE;
|
||||||
fputs("\n", new_status_file);
|
fputs("\n", new_status_file);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* If the package from the status file wasnt handle above, do it now*/
|
/* If the package from the status file wasnt handle above, do it now*/
|
||||||
if (write_flag == FALSE) {
|
if (write_flag == FALSE) {
|
||||||
fprintf(new_status_file, "%s\n\n", control_buffer);
|
fprintf(new_status_file, "%s\n\n", control_buffer);
|
||||||
@ -866,20 +855,9 @@ void write_status_file(deb_file_t **deb_file)
|
|||||||
for(i = 0; deb_file[i] != NULL; i++) {
|
for(i = 0; deb_file[i] != NULL; i++) {
|
||||||
status_num = search_status_hashtable(name_hashtable[package_hashtable[deb_file[i]->package]->name]);
|
status_num = search_status_hashtable(name_hashtable[package_hashtable[deb_file[i]->package]->name]);
|
||||||
if (strcmp("reinstreq", name_hashtable[get_status(status_num, 2)]) == 0) {
|
if (strcmp("reinstreq", name_hashtable[get_status(status_num, 2)]) == 0) {
|
||||||
char *last_char;
|
write_buffer_no_status(new_status_file, deb_file[i]->control_file);
|
||||||
/* remove trailing \n's */
|
|
||||||
while(1) {
|
|
||||||
last_char = last_char_is(deb_file[i]->control_file, '\n');
|
|
||||||
if (last_char) {
|
|
||||||
*last_char = '\0';
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fputs(deb_file[i]->control_file, new_status_file);
|
|
||||||
set_status(status_num, "ok", 2);
|
set_status(status_num, "ok", 2);
|
||||||
fprintf(new_status_file, "\nStatus: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
fprintf(new_status_file, "Status: %s\n\n", name_hashtable[status_hashtable[status_num]->status]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fclose(old_status_file);
|
fclose(old_status_file);
|
||||||
@ -1037,9 +1015,11 @@ char **create_list(const char *filename)
|
|||||||
int length = 0;
|
int length = 0;
|
||||||
int count = 0;
|
int count = 0;
|
||||||
|
|
||||||
list_stream = xfopen(filename, "r");
|
/* dont use [xw]fopen here, handle error ourself */
|
||||||
|
list_stream = fopen(filename, "r");
|
||||||
if (list_stream == NULL) {
|
if (list_stream == NULL) {
|
||||||
return(NULL);
|
*file_list = NULL;
|
||||||
|
return(file_list);
|
||||||
}
|
}
|
||||||
while (getline(&line, &length, list_stream) != -1) {
|
while (getline(&line, &length, list_stream) != -1) {
|
||||||
file_list = xrealloc(file_list, sizeof(char *) * (length + 1));
|
file_list = xrealloc(file_list, sizeof(char *) * (length + 1));
|
||||||
@ -1053,6 +1033,7 @@ char **create_list(const char *filename)
|
|||||||
length = 0;
|
length = 0;
|
||||||
}
|
}
|
||||||
fclose(list_stream);
|
fclose(list_stream);
|
||||||
|
|
||||||
if (count == 0) {
|
if (count == 0) {
|
||||||
return(NULL);
|
return(NULL);
|
||||||
} else {
|
} else {
|
||||||
@ -1212,6 +1193,7 @@ void purge_package(const unsigned int package_num)
|
|||||||
if (run_package_script(package_name, "postrm") == -1) {
|
if (run_package_script(package_name, "postrm") == -1) {
|
||||||
error_msg_and_die("postrm fialure.. set status to what?");
|
error_msg_and_die("postrm fialure.. set status to what?");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change package status */
|
/* Change package status */
|
||||||
set_status(status_num, "purge", 1);
|
set_status(status_num, "purge", 1);
|
||||||
set_status(status_num, "not-installed", 3);
|
set_status(status_num, "not-installed", 3);
|
||||||
@ -1219,7 +1201,6 @@ void purge_package(const unsigned int package_num)
|
|||||||
|
|
||||||
void unpack_package(deb_file_t *deb_file)
|
void unpack_package(deb_file_t *deb_file)
|
||||||
{
|
{
|
||||||
// const unsigned int package_name_num = package_hashtable[deb_file->package]->name;
|
|
||||||
const char *package_name = name_hashtable[package_hashtable[deb_file->package]->name];
|
const char *package_name = name_hashtable[package_hashtable[deb_file->package]->name];
|
||||||
const unsigned int status_num = search_status_hashtable(package_name);
|
const unsigned int status_num = search_status_hashtable(package_name);
|
||||||
const unsigned int status_package_num = status_hashtable[status_num]->status;
|
const unsigned int status_package_num = status_hashtable[status_num]->status;
|
||||||
@ -1248,7 +1229,6 @@ void unpack_package(deb_file_t *deb_file)
|
|||||||
|
|
||||||
/* Create the list file */
|
/* Create the list file */
|
||||||
strcat(info_prefix, "list");
|
strcat(info_prefix, "list");
|
||||||
|
|
||||||
out_stream = xfopen(info_prefix, "w");
|
out_stream = xfopen(info_prefix, "w");
|
||||||
deb_extract(deb_file->filename, out_stream, (extract_quiet | extract_data_tar_gz | extract_list), NULL, NULL);
|
deb_extract(deb_file->filename, out_stream, (extract_quiet | extract_data_tar_gz | extract_list), NULL, NULL);
|
||||||
fclose(out_stream);
|
fclose(out_stream);
|
||||||
@ -1393,7 +1373,7 @@ extern int dpkg_main(int argc, char **argv)
|
|||||||
/* TODO: check dependencies before removing */
|
/* TODO: check dependencies before removing */
|
||||||
if ((dpkg_opt & dpkg_opt_force_ignore_depends) != dpkg_opt_force_ignore_depends) {
|
if ((dpkg_opt & dpkg_opt_force_ignore_depends) != dpkg_opt_force_ignore_depends) {
|
||||||
if (!check_deps(deb_file, 0, deb_count)) {
|
if (!check_deps(deb_file, 0, deb_count)) {
|
||||||
error_msg_and_die("Dependency check fialed");
|
error_msg_and_die("Dependency check failed");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1416,6 +1396,7 @@ extern int dpkg_main(int argc, char **argv)
|
|||||||
configure_package(deb_file[i]);
|
configure_package(deb_file[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
write_status_file(deb_file);
|
write_status_file(deb_file);
|
||||||
|
|
||||||
for (i = 0; i < NAME_HASH_PRIME; i++) {
|
for (i = 0; i < NAME_HASH_PRIME; i++) {
|
||||||
|
13
dpkg_deb.c
13
dpkg_deb.c
@ -110,12 +110,17 @@ extern int dpkg_deb_main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
else if (arg_type == arg_type_field) {
|
else if (arg_type == arg_type_field) {
|
||||||
char *field = NULL;
|
char *field = NULL;
|
||||||
|
char *name;
|
||||||
|
char *value;
|
||||||
int field_start = 0;
|
int field_start = 0;
|
||||||
|
|
||||||
while ((field = read_package_field(&output_buffer[field_start])) != NULL) {
|
while (1) {
|
||||||
field_start += (strlen(field) + 1);
|
field_start += read_package_field(&output_buffer[field_start], &name, &value);
|
||||||
if (strstr(field, argv[optind + 1]) == field) {
|
if (name == NULL) {
|
||||||
puts(field + strlen(argv[optind + 1]) + 2);
|
break;
|
||||||
|
}
|
||||||
|
if (strcmp(name, argv[optind + 1]) == 0) {
|
||||||
|
puts(value);
|
||||||
}
|
}
|
||||||
free(field);
|
free(field);
|
||||||
}
|
}
|
||||||
|
@ -245,7 +245,7 @@ char *unarchive(FILE *src_stream, FILE *out_stream, file_header_t *(*get_header)
|
|||||||
const int extract_function, const char *prefix, char **extract_names);
|
const int extract_function, const char *prefix, char **extract_names);
|
||||||
char *deb_extract(const char *package_filename, FILE *out_stream, const int extract_function,
|
char *deb_extract(const char *package_filename, FILE *out_stream, const int extract_function,
|
||||||
const char *prefix, const char *filename);
|
const char *prefix, const char *filename);
|
||||||
char *read_package_field(const char *package_buffer);
|
int read_package_field(const char *package_buffer, char **field_name, char **field_value);
|
||||||
char *fgets_str(FILE *file, const char *terminating_string);
|
char *fgets_str(FILE *file, const char *terminating_string);
|
||||||
|
|
||||||
extern int unzip(FILE *l_in_file, FILE *l_out_file);
|
extern int unzip(FILE *l_in_file, FILE *l_out_file);
|
||||||
|
@ -245,7 +245,7 @@ char *unarchive(FILE *src_stream, FILE *out_stream, file_header_t *(*get_header)
|
|||||||
const int extract_function, const char *prefix, char **extract_names);
|
const int extract_function, const char *prefix, char **extract_names);
|
||||||
char *deb_extract(const char *package_filename, FILE *out_stream, const int extract_function,
|
char *deb_extract(const char *package_filename, FILE *out_stream, const int extract_function,
|
||||||
const char *prefix, const char *filename);
|
const char *prefix, const char *filename);
|
||||||
char *read_package_field(const char *package_buffer);
|
int read_package_field(const char *package_buffer, char **field_name, char **field_value);
|
||||||
char *fgets_str(FILE *file, const char *terminating_string);
|
char *fgets_str(FILE *file, const char *terminating_string);
|
||||||
|
|
||||||
extern int unzip(FILE *l_in_file, FILE *l_out_file);
|
extern int unzip(FILE *l_in_file, FILE *l_out_file);
|
||||||
|
@ -3,25 +3,89 @@
|
|||||||
#include "libbb.h"
|
#include "libbb.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Returns a [multi-line] package field
|
* Gets the next package field from package_buffer, seperated into the field name
|
||||||
|
* and field value, it returns the int offset to the first character of the next field
|
||||||
*/
|
*/
|
||||||
extern char *read_package_field(const char *package_buffer)
|
int read_package_field(const char *package_buffer, char **field_name, char **field_value)
|
||||||
{
|
{
|
||||||
int field_length = 0;
|
int offset_name_start = 0;
|
||||||
int buffer_length = 0;
|
int offset_name_end = 0;
|
||||||
|
int offset_value_start = 0;
|
||||||
|
int offset_value_end = 0;
|
||||||
|
int offset = 0;
|
||||||
|
int next_offset;
|
||||||
|
int name_length;
|
||||||
|
int value_length;
|
||||||
|
int exit_flag = FALSE;
|
||||||
|
|
||||||
if (package_buffer == NULL) {
|
if (package_buffer == NULL) {
|
||||||
return(NULL);
|
*field_name = NULL;
|
||||||
|
*field_value = NULL;
|
||||||
|
return(-1);
|
||||||
}
|
}
|
||||||
buffer_length = strlen(package_buffer);
|
while (1) {
|
||||||
field_length = strcspn(package_buffer, "\n");
|
next_offset = offset + 1;
|
||||||
while (field_length < buffer_length) {
|
switch (package_buffer[offset]) {
|
||||||
if (package_buffer[field_length + 1] != ' ') {
|
case('\0'):
|
||||||
return(xstrndup(package_buffer, field_length));
|
exit_flag = TRUE;
|
||||||
|
break;
|
||||||
|
case(':'):
|
||||||
|
if (offset_name_end == 0) {
|
||||||
|
offset_name_end = offset;
|
||||||
|
offset_value_start = next_offset;
|
||||||
}
|
}
|
||||||
field_length++;
|
/* TODO: Name might still have trailing spaces if ':' isnt
|
||||||
field_length += strcspn(&package_buffer[field_length], "\n");
|
* immediately after name */
|
||||||
|
break;
|
||||||
|
case('\n'):
|
||||||
|
/* TODO: The char next_offset may be out of bounds */
|
||||||
|
if (package_buffer[next_offset] != ' ') {
|
||||||
|
exit_flag = TRUE;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
return(xstrdup(package_buffer));
|
case('\t'):
|
||||||
|
case(' '):
|
||||||
|
/* increment the value start point if its a just filler */
|
||||||
|
if (offset_name_start == offset) {
|
||||||
|
offset_name_start++;
|
||||||
|
}
|
||||||
|
if (offset_value_start == offset) {
|
||||||
|
offset_value_start++;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (exit_flag == TRUE) {
|
||||||
|
/* Check that the names are valid */
|
||||||
|
offset_value_end = offset;
|
||||||
|
name_length = offset_name_end - offset_name_start;
|
||||||
|
value_length = offset_value_end - offset_value_start;
|
||||||
|
if (name_length == 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if ((name_length > 0) && (value_length > 0)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If not valid, start fresh with next field */
|
||||||
|
exit_flag = FALSE;
|
||||||
|
offset_name_start = offset + 1;
|
||||||
|
offset_name_end = 0;
|
||||||
|
offset_value_start = offset + 1;
|
||||||
|
offset_value_end = offset + 1;
|
||||||
|
offset++;
|
||||||
|
}
|
||||||
|
offset++;
|
||||||
|
}
|
||||||
|
if (name_length == 0) {
|
||||||
|
*field_name = NULL;
|
||||||
|
} else {
|
||||||
|
*field_name = xstrndup(&package_buffer[offset_name_start], name_length);
|
||||||
|
}
|
||||||
|
if (value_length > 0) {
|
||||||
|
*field_value = xstrndup(&package_buffer[offset_value_start], value_length);
|
||||||
|
} else {
|
||||||
|
*field_value = NULL;
|
||||||
|
}
|
||||||
|
return(next_offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user