Compare commits

..

No commits in common. "master" and "v3.2.0" have entirely different histories.

5 changed files with 69 additions and 261 deletions

View File

@ -1,29 +0,0 @@
name: Publish PHP Package
on:
push:
tags:
- 'v*'
jobs:
docker:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v3
- name: Get tag name
run: echo "TAG=${GITHUB_REF#refs/*/}" >> $GITHUB_ENV
- name: zip and send
run: |
ls -la
apt-get update -y
apt-get install -y zip
cd ../
zip -r ${{ gitea.event.repository.name }}.zip ${{ gitea.event.repository.name }} -x '*.git*'
curl -v \
--user composer-pbl:${{ secrets.COMPOSER_PSWD }} \
--upload-file ${{ gitea.event.repository.name }}.zip \
https://git.bit5.ru/api/packages/bit/composer?version=${{ env.TAG }}

View File

@ -12,53 +12,23 @@ task('bhl_clean_cache', function()
bhl_clean_cache(); bhl_clean_cache();
}); });
class BhlProj function bhl_proj_file()
{
public static ?BhlProj $active = null;
public string $file_path;
public array $inc_dirs = array();
public array $src_dirs = array();
public array $defines = array();
public array $bindings_sources;
public array $postproc_sources;
public string $postproc_dll;
public string $bindings_dll;
public string $result_file;
public string $error_file;
public string $tmp_dir;
public int $max_threads;
public bool $deterministic;
function getIncPath() : array
{
if($this->inc_dirs)
return $this->inc_dirs;
return $this->src_dirs;
}
}
function bhl_proj_file() : string
{ {
return get('BHL_PROJ_FILE'); return get('BHL_PROJ_FILE');
} }
function bhl_proj_load(string $proj_file) : BhlProj function bhl_proj()
{ {
$arr = json_decode(ensure_read($proj_file), true); global $GAME_ROOT;
if(!$arr)
static $proj;
if(!$proj)
{
$proj_file = bhl_proj_file();
$proj = json_decode(ensure_read($proj_file));
if(!$proj)
throw new Exception("Bad bhl project file: $proj_file"); throw new Exception("Bad bhl project file: $proj_file");
$proj = new BhlProj();
foreach($arr as $k => $v)
$proj->{$k} = $v;
//NOTE: adding path to the file for further convenience
$proj->file_path = $proj_file;
foreach($proj->inc_dirs as $k => $v)
$proj->inc_dirs[$k] = _bhl_make_abs_path($proj_file, $v);
foreach($proj->src_dirs as $k => $v) foreach($proj->src_dirs as $k => $v)
$proj->src_dirs[$k] = _bhl_make_abs_path($proj_file, $v); $proj->src_dirs[$k] = _bhl_make_abs_path($proj_file, $v);
@ -66,40 +36,18 @@ function bhl_proj_load(string $proj_file) : BhlProj
$proj->bindings_sources[$k] = _bhl_make_abs_path($proj_file, $v); $proj->bindings_sources[$k] = _bhl_make_abs_path($proj_file, $v);
$proj->bindings_dll= _bhl_make_abs_path($proj_file, $proj->bindings_dll); $proj->bindings_dll= _bhl_make_abs_path($proj_file, $proj->bindings_dll);
if(isset($proj->postproc_sources))
{
foreach($proj->postproc_sources as $k => $v) foreach($proj->postproc_sources as $k => $v)
$proj->postproc_sources[$k] = _bhl_make_abs_path($proj_file, $v); $proj->postproc_sources[$k] = _bhl_make_abs_path($proj_file, $v);
}
if(isset($proj->postproc_dll))
$proj->postproc_dll= _bhl_make_abs_path($proj_file, $proj->postproc_dll); $proj->postproc_dll= _bhl_make_abs_path($proj_file, $proj->postproc_dll);
$proj->result_file = _bhl_make_abs_path($proj_file, $proj->result_file); $proj->result_file = _bhl_make_abs_path($proj_file, $proj->result_file);
$proj->error_file = _bhl_make_abs_path($proj_file, $proj->error_file); $proj->error_file = _bhl_make_abs_path($proj_file, $proj->error_file);
$proj->tmp_dir = _bhl_make_abs_path($proj_file, $proj->tmp_dir); $proj->tmp_dir = _bhl_make_abs_path($proj_file, $proj->tmp_dir);
}
return $proj; return $proj;
} }
function bhl_proj_set_active(BhlProj $proj) : ?BhlProj function _bhl_make_abs_path($proj_file, $path)
{
$prev = BhlProj::$active;
BhlProj::$active = $proj;
return $prev;
}
function bhl_proj() : BhlProj
{
if(BhlProj::$active == null)
{
$proj = bhl_proj_load(bhl_proj_file());
BhlProj::$active = $proj;
}
return BhlProj::$active;
}
function _bhl_make_abs_path(string $proj_file, string $path) : string
{ {
if($path && $path[0] == '.') if($path && $path[0] == '.')
return dirname($proj_file) . '/' . $path; return dirname($proj_file) . '/' . $path;
@ -107,44 +55,53 @@ function _bhl_make_abs_path(string $proj_file, string $path) : string
return $path; return $path;
} }
function bhl_result_file() : string function bhl_result_file()
{ {
global $GAME_ROOT;
return bhl_proj()->result_file; return bhl_proj()->result_file;
} }
function bhl_dir() : string function bhl_dir()
{ {
global $GAME_ROOT; global $GAME_ROOT;
return "$GAME_ROOT/composer/vendor/bit/bhl"; return "$GAME_ROOT/composer/vendor/bit/bhl";
} }
function bhl_shell(string $cmd, &$ret_var, &$ret_out) function bhl_shell($cmd, &$ret_var, &$ret_out)
{
$prev_path = mono_try_override_path();
try
{ {
shell_try(bhl_dir()."/bhl $cmd", $ret_var, $ret_out); shell_try(bhl_dir()."/bhl $cmd", $ret_var, $ret_out);
} }
finally
{
mono_try_restore_path($prev_path);
}
}
function bhl_shell_ensure(string $cmd) function bhl_shell_ensure($cmd)
{ {
bhl_shell($cmd, $ret_var, $ret_out); bhl_shell($cmd, $ret_var, $ret_out);
if($ret_var !== 0) if($ret_var !== 0)
throw new Exception("Error executing shell cmd: $ret_var"); throw new Exception("Error executing shell cmd: $ret_var");
} }
function bhl_scan_files() : array function bhl_scan_files()
{ {
return scan_files_rec(bhl_proj()->src_dirs, array('.bhl')); return scan_files_rec(bhl_proj()->src_dirs, array('.bhl'));
} }
function bhl_run(bool $debug = true, bool $force = false, bool $exit_on_err = true) function bhl_run($debug = true, $force = false, $exit_on_err = true)
{ {
global $GAME_ROOT; global $GAME_ROOT;
$bhl_proj = bhl_proj(); $result_file = bhl_proj()->result_file;
$result_file = $bhl_proj->result_file;
if($force || need_to_regen($result_file, bhl_scan_files())) if($force || need_to_regen($result_file, bhl_scan_files()))
{ {
bhl_shell("compile -p " . $bhl_proj->file_path . " " . bhl_shell("compile -p " . bhl_proj_file() . " " .
($debug ? " -d" : "") . " " . ($debug ? " -d" : "") . " " .
($force || !getor("BHL_USE_CACHE", true) ? " -C" : ""), ($force || !getor("BHL_USE_CACHE", true) ? " -C" : ""),
$ret_var, $ret_out $ret_var, $ret_out
@ -152,20 +109,22 @@ function bhl_run(bool $debug = true, bool $force = false, bool $exit_on_err = tr
if($ret_var != 0) if($ret_var != 0)
{ {
bhl_handle_error_result($ret_out, $bhl_proj->error_file, $exit_on_err); bhl_handle_error_result($ret_out, bhl_proj()->error_file, $exit_on_err);
if(!$exit_on_err) if(!$exit_on_err)
return false; return false;
} }
else if($debug) else
echo "BHL BUNDLE: total " . kb_len(filesize($result_file)) . ", CRC " . hexdec(hash_file('CRC32', $result_file, false)) . "\n"; echo "BHL BUNDLE: total " . kb_len(filesize($result_file)) . "\n";
}
} }
function bhl_handle_error_result(array $ret_out, string $err_file, bool $exit = true) @touch($result_file);
}
function bhl_handle_error_result(array $ret_out, $err_file, $exit = true)
{ {
if(!is_file($err_file)) if(!is_file($err_file))
{ {
stderr("Something went wrong: " . implode("\n", $ret_out)); stderr("Something went wrong: " . (is_array($ret_out) ? implode("\n", $ret_out) : "????"));
if($exit) if($exit)
exit(1); exit(1);
} }
@ -199,7 +158,7 @@ function bhl_handle_error_result(array $ret_out, string $err_file, bool $exit =
function bhl_clean() function bhl_clean()
{ {
bhl_clean_cache(); bhl_clean_cache();
bhl_shell("clean", $ret, $out); bhl_shell_ensure("clean");
} }
function bhl_clean_cache() function bhl_clean_cache()
@ -207,7 +166,7 @@ function bhl_clean_cache()
ensure_rm(bhl_proj()->tmp_dir); ensure_rm(bhl_proj()->tmp_dir);
} }
function bhl_show_position(int $line, int $row, array $lines) : string function bhl_show_position($line, $row, array $lines)
{ {
if($line > 0 && $line <= count($lines)) if($line > 0 && $line <= count($lines))
{ {
@ -227,7 +186,7 @@ function bhl_show_position(int $line, int $row, array $lines) : string
return "??? @($line:$row)"; return "??? @($line:$row)";
} }
function bhl_line_row_to_pos(string $file, int $line, int $row) : ?int function bhl_line_row_to_pos($file, $line, $row)
{ {
$pos = 0; $pos = 0;
$lines = file($file); $lines = file($file);
@ -240,10 +199,9 @@ function bhl_line_row_to_pos(string $file, int $line, int $row) : ?int
return $pos; return $pos;
} }
function bhl_map_module_to_file(string $module) : ?string function bhl_map_module_to_file($module)
{ {
$bhl_proj = bhl_proj(); foreach(bhl_proj()->src_dirs as $dir)
foreach($bhl_proj->getIncPath() as $dir)
{ {
$tmp = $dir.'/'.$module.'.bhl'; $tmp = $dir.'/'.$module.'.bhl';
if(file_exists($tmp)) if(file_exists($tmp))
@ -252,40 +210,17 @@ function bhl_map_module_to_file(string $module) : ?string
return null; return null;
} }
function bhl_validate_func_ref(string $module_file, string $func_full_name, array $signature) function bhl_validate_func_ref($module, $func, array $signature)
{ {
if(sizeof($signature) == 0) if(sizeof($signature) == 0)
throw new Exception("Signature is invalid"); throw new Exception("Signature is invalid");
$name_items = explode('.', $func_full_name); $module_file = bhl_map_module_to_file($module);
$func = array_pop($name_items);
$namespace = implode('.', $name_items); if(!$module_file)
throw new Exception("Module not found '{$module}'");
$module_src = file_get_contents($module_file); $module_src = file_get_contents($module_file);
$module_src = _bhl_remove_comments($module_src);
if(!$module_src)
throw new Exception("Bad module file '{$module_file}'");
$ns_chunks = _bhl_split_by_namespaces($module_src);
$module_chunks = array();
if($namespace)
{
if(!isset($ns_chunks[$namespace]))
throw new Exception("No namespace '$namespace' found in '$module_file'");
foreach($ns_chunks[$namespace] as $ns_src)
$module_chunks[] = $ns_src;
}
else
{
if(!isset($ns_chunks['']))
throw new Exception("No global namespace found in '$module_file'");
foreach($ns_chunks[''] as $ns_src)
$module_chunks[] = $ns_src;
}
$signature_pattern = ''; $signature_pattern = '';
$signature_pattern .= '~func\s+'; $signature_pattern .= '~func\s+';
@ -308,58 +243,6 @@ function bhl_validate_func_ref(string $module_file, string $func_full_name, arra
$signature_pattern .= '\s*\)'; $signature_pattern .= '\s*\)';
$signature_pattern .= '~'; $signature_pattern .= '~';
foreach($module_chunks as $module_chunk_src) if(!preg_match($signature_pattern, $module_src))
{ throw new Exception("Func '$func' signature '".implode(',', $signature)."' not found in module '$module'");
if(preg_match($signature_pattern, $module_chunk_src))
return;
}
throw new Exception("Func '$func_full_name(".implode(',', $signature).")' not found in '$module_file'");
}
function _bhl_remove_comments(string $txt) : string
{
//block comments
if(strpos($txt, '/*') !== false)
{
$regex = '~/\*.*?\*/~s';
$txt = preg_replace_callback(
$regex,
//preserve the new lines for better error reporting
function($m) { return str_repeat("\n", substr_count($m[0], "\n")); },
$txt);
}
//line comments
$txt = preg_replace("~\s*(?<!:)//.*~", "\n", $txt);
return $txt;
}
function _bhl_split_by_namespaces(string $src) : array
{
$nss = array();
$chunks = preg_split('~^\s*(namespace\s+[^\{]+){~m', $src, -1, PREG_SPLIT_DELIM_CAPTURE|PREG_SPLIT_NO_EMPTY);
for($i=0;$i<count($chunks);)
{
$chunk = $chunks[$i];
if(strpos(ltrim($chunk), 'namespace ') === 0)
{
$ns = trim(substr(ltrim($chunk), 9));
$body = $chunks[$i+1];
if(!isset($nss[$ns]))
$nss[$ns] = array();
$nss[$ns][] = $body;
$i+=2;
}
else
{
if(!isset($nss['']))
$nss[''] = array();
$nss[''][] = $chunk;
++$i;
}
}
return $nss;
} }

View File

@ -2,7 +2,7 @@
namespace metagen_php; namespace metagen_php;
use Exception; use Exception;
function flt_bhl_ref($val, $name, $data, $str_args) function flt_bhl_ref($val, $name, $struct, $args)
{ {
if(!$val) if(!$val)
return $val; return $val;
@ -10,16 +10,14 @@ function flt_bhl_ref($val, $name, $data, $str_args)
if(!isset($val['module'])) if(!isset($val['module']))
throw new Exception("Missing 'module'"); throw new Exception("Missing 'module'");
$module_file = \taskman\bhl_map_module_to_file($val['module']);
if(!$module_file)
throw new Exception("Module not found '{$val['module']}");
if(!isset($val['fn'])) if(!isset($val['fn']))
throw new Exception("Missing 'fn'"); throw new Exception("Missing 'fn'");
$args = explode(",", $str_args); $signature = explode(",", $args);
if(!$signature)
throw new Exception("Invalid signature: $args");
\taskman\bhl_validate_func_ref($module_file, $val['fn'], $args); \taskman\bhl_validate_func_ref($val['module'], $val['fn'], $signature);
return $val; return $val;
} }

View File

@ -1,47 +0,0 @@
<?php
/**
* @global
*/
function macro_BHL_REF($proc, $module, $func, $signature_json = '')
{
//check if it's a relative path
if($module[0] == '.')
{
$abs_module = \taskman\normalize_path(dirname($proc->getRootFile()) . '/' . $module);
$mapped = false;
$bhl_proj = \taskman\bhl_proj();
$inc_path = $bhl_proj->getIncPath();
foreach($inc_path as $dir)
{
$rel_module = str_replace(\taskman\normalize_path($dir), '', $abs_module);
if($rel_module != $abs_module)
{
$mapped = true;
$rel_module = str_replace('\\', '/', $rel_module);
$module = "/" . $rel_module;
break;
}
}
if(!$mapped)
throw new Exception("Could not map relative module: $module");
}
$module_file = \taskman\bhl_map_module_to_file($module);
if(!$module_file)
throw new Exception("Module not found '{$module}'");
//validate signature only if it's explicitely passed
if($signature_json)
{
$signature = json_decode($signature_json, false);
if(!is_array($signature))
throw new Exception("Signature is invalid '$module'");
\taskman\bhl_validate_func_ref($module_file, $func, $signature);
}
return array('module' => ltrim($module, '/'), 'fn' => $func);
}

View File

@ -4,5 +4,8 @@
"homepage": "https://git.bit5.ru/composer/taskman_bhl", "homepage": "https://git.bit5.ru/composer/taskman_bhl",
"require": { "require": {
"php": ">=7.4" "php": ">=7.4"
},
"autoload": {
"classmap": ["bhl.inc.php"]
} }
} }