/*************************************************************************** gbx_project.c (c) 2000-2017 BenoƮt Minisini 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 2, 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, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ***************************************************************************/ #define __GBX_PROJECT_C #include "config.h" #include "gb_common.h" #include "gb_common_case.h" #include "gb_alloc.h" #include "gb_error.h" #include #include "gb_limit.h" #include "gb_buffer.h" #include "gb_file.h" #include "gbx_stream.h" #include "gbx_archive.h" #include "gbx_exec.h" #include "gbx_stack.h" #include "gb_component.h" #include "gbx_component.h" #include "gbx_project.h" char *PROJECT_path = NULL; char *PROJECT_exec_path = NULL; char *PROJECT_name = NULL; char *PROJECT_source = NULL; char *PROJECT_title = NULL; const char *PROJECT_startup = NULL; char *PROJECT_version = NULL; CLASS *PROJECT_class = NULL; int PROJECT_argc = 0; char **PROJECT_argv = NULL; //char *PROJECT_argname = NULL; char *PROJECT_oldcwd = NULL; bool PROJECT_run_httpd = FALSE; bool PROJECT_run_tests = FALSE; const char *PROJECT_override = NULL; static char *project_buffer; static char *environment_buffer; //static char *project_ptr; static int project_line; static const char *_last_component = NULL; static void raise_error(const char *msg) { char line[16]; snprintf(line, sizeof(line), "%d", project_line); THROW(E_PROJECT, line, msg); } static void project_title(char *name, int len) { name[len] = 0; PROJECT_title = name; } static void project_source(char *name, int len) { if (name[0] == '#') { name[len] = 0; PROJECT_source = &name[1]; } } static void project_version(char *name, int len) { name[len] = 0; PROJECT_version = name; } static void project_component(char *name, int len) { _last_component = name; name[len] = 0; COMPONENT_create(name); // If 'gb.httpd' is set explicitely, then always run through it. if (strcmp(name, "gb.httpd") == 0) PROJECT_run_httpd = TRUE; _last_component = NULL; } static void project_startup(char *name, int len) { if (PROJECT_startup) return; if (len == 0) raise_error("Project startup class name is void"); name[len] = 0; PROJECT_startup = name; } static void project_library_path(char *name, int len) { if (!EXEC_debug) { ARCHIVE_path = STRING_new_zero(STRING_conv_file_name(name, len)); if (*name != '/') { name = STRING_new_zero(FILE_cat(PROJECT_path, ARCHIVE_path, NULL)); STRING_free(&ARCHIVE_path); ARCHIVE_path = name; } } } static void check_after_analyze() { if (!PROJECT_name || PROJECT_name[0] == 0) raise_error("No project name"); if (!PROJECT_startup || PROJECT_startup[0] == 0) raise_error("No startup class"); if (!PROJECT_title || PROJECT_title[0] == 0) PROJECT_title = PROJECT_name; } static bool get_line(char **addr, const char *end, char **start, int *len) { char *p = *addr; if (p >= end) return FALSE; while (p < end && *p && *p != '\n') p++; *start = *addr; *len = p - *start; *addr = p + 1; return (*len > 0); } void PROJECT_analyze_startup(char *addr, int len, PROJECT_COMPONENT_CALLBACK cb) { char *end = &addr[len]; char *p; int l, i; if (!cb) { if (get_line(&addr, end, &p, &l)) project_startup(p, l); if (get_line(&addr, end, &p, &l)) project_title(p, l); if (get_line(&addr, end, &p, &l)) project_source(p, l); get_line(&addr, end, &p, &l); // Deprecated "StackTrace" if (get_line(&addr, end, &p, &l)) project_version(p, l); } else { for (i = 1; i <= 5; i++) get_line(&addr, end, &p, &l); } if (get_line(&addr, end, &p, &l)) { project_library_path(p, l); while (get_line(&addr, end, &p, &l)); } if (!cb) { while (get_line(&addr, end, &p, &l)) project_component(p, l); check_after_analyze(); } else { while (get_line(&addr, end, &p, &l)) { p[l] = 0; (*cb)(p); } } } static void init_environment(char *addr, int len) { char *end = &addr[len]; char *p; int l; while (get_line(&addr, end, &p, &l)) { p[l] = 0; putenv(p); } } void PROJECT_init(const char *file) { int len; const char *path; /* Save the working directory */ PROJECT_oldcwd = STRING_new_zero(FILE_getcwd(NULL)); /* Gambas installation path */ path = FILE_find_gambas(); PROJECT_exec_path = STRING_new_zero(FILE_get_dir(FILE_get_dir(path))); /* Component paths */ #ifdef OS_64BITS COMPONENT_path = STRING_new_zero(FILE_cat(PROJECT_exec_path, GAMBAS_LIB64_PATH, NULL)); if (access(COMPONENT_path, F_OK)) { STRING_free(&COMPONENT_path); COMPONENT_path = STRING_new_zero(FILE_cat(PROJECT_exec_path, GAMBAS_LIB_PATH, NULL)); } #else COMPONENT_path = STRING_new_zero(FILE_cat(PROJECT_exec_path, GAMBAS_LIB_PATH, NULL)); #endif //STRING_new(&COMPONENT_user_path, FILE_cat(PROJECT_get_home(), ".local", GAMBAS_LIB_PATH, NULL), 0); /* Project path & name*/ if (!file) { // "gbx3 -e" case PROJECT_path = STRING_new("", 0); PROJECT_name = STRING_new("", 0); return; } if (*file == '.' && file[1] == '/') file += 2; if (EXEC_arch) { if (FILE_is_relative(file)) { path = FILE_getcwd(file); if (path == NULL) goto _PANIC; } else path = file; path = FILE_get_dir(path); FILE_chdir(path); } else { if (FILE_is_absolute(file)) { path = file; } else { path = FILE_getcwd(file); if (path == NULL) goto _PANIC; if (!chdir(path)) { path = FILE_getcwd(NULL); if (path == NULL) goto _PANIC; } } } len = strlen(path); while (len > 1) { if (path[len - 1] != '/') break; len--; /*path[len] = 0;*/ } PROJECT_path = STRING_new(path, len); FILE_chdir(PROJECT_path); /* Project name */ if (EXEC_arch) PROJECT_name = STRING_new_zero(FILE_get_basename(file)); else PROJECT_name = STRING_new_zero(FILE_get_name(PROJECT_path)); /* Main archive creation */ ARCHIVE_create_main(EXEC_arch ? FILE_get_name(file) : NULL); return; _PANIC: ERROR_panic("Cannot initialize project: %s", strerror(errno)); } void PROJECT_load() { const char *file; int len; /* Project file analyze */ STACK_init(); if (EXEC_arch) file = ".startup"; else file = FILE_cat(PROJECT_path, ".startup", NULL); TRY { STREAM_load(file, &project_buffer, &len); } CATCH { ERROR_fatal("unable to find startup file"); } END_TRY TRY { PROJECT_analyze_startup(project_buffer, len, NULL); } CATCH { if (_last_component) ERROR_fatal("unable to load component: %s", _last_component); else ERROR_fatal("unable to analyze startup file"); } END_TRY if (EXEC_arch) file = ".environment"; else file = FILE_cat(PROJECT_path, ".environment", NULL); if (FILE_exist(file)) { TRY { STREAM_load(file, &environment_buffer, &len); } CATCH { ERROR_fatal("unable to load environment file"); } END_TRY init_environment(environment_buffer, len); } // Loads all component COMPONENT_load_all(); } void PROJECT_load_finish(void) { // Load exported class of components written in Gambas COMPONENT_load_all_finish(); // Loads main archive ARCHIVE_load_main(); // Startup class PROJECT_class = CLASS_find(PROJECT_run_tests ? "Test" : PROJECT_startup); } void PROJECT_exit(void) { if (project_buffer) FREE(&project_buffer); if (environment_buffer) FREE(&environment_buffer); //STRING_free(&PROJECT_argname); STRING_free(&PROJECT_name); STRING_free(&PROJECT_path); STRING_free(&PROJECT_oldcwd); STRING_free(&PROJECT_exec_path); }