mirror of
https://github.com/danog/ext-php-rs.git
synced 2024-12-12 09:09:46 +01:00
Merge remote-tracking branch 'origin/master' into process-globals
This commit is contained in:
commit
21134fb618
15
.github/actions/embed/Dockerfile
vendored
Normal file
15
.github/actions/embed/Dockerfile
vendored
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
FROM php:8.2-bullseye
|
||||||
|
|
||||||
|
WORKDIR /tmp
|
||||||
|
|
||||||
|
RUN apt update -y && apt upgrade -y
|
||||||
|
RUN apt install lsb-release wget gnupg software-properties-common -y
|
||||||
|
RUN bash -c "$(wget -O - https://apt.llvm.org/llvm.sh)"
|
||||||
|
|
||||||
|
ENV RUSTUP_HOME=/rust
|
||||||
|
ENV CARGO_HOME=/cargo
|
||||||
|
ENV PATH=/cargo/bin:/rust/bin:$PATH
|
||||||
|
|
||||||
|
RUN curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y --no-modify-path
|
||||||
|
|
||||||
|
ENTRYPOINT [ "/cargo/bin/cargo", "test", "--lib", "--release", "--all-features" ]
|
5
.github/actions/embed/action.yml
vendored
Normal file
5
.github/actions/embed/action.yml
vendored
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
name: 'PHP Embed and Rust'
|
||||||
|
description: 'Builds the crate after installing the latest PHP with php embed and stable Rust.'
|
||||||
|
runs:
|
||||||
|
using: 'docker'
|
||||||
|
image: 'Dockerfile'
|
22
.github/workflows/build.yml
vendored
22
.github/workflows/build.yml
vendored
@ -15,9 +15,9 @@ jobs:
|
|||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
os: [ubuntu-latest, macos-latest, windows-latest]
|
os: [ubuntu-latest, macos-latest, windows-latest]
|
||||||
php: ["8.0", "8.1", "8.2"]
|
php: ["8.0", "8.1", "8.2", "8.3"]
|
||||||
rust: [stable, nightly]
|
rust: [stable, nightly]
|
||||||
clang: ["14"]
|
clang: ["15", "17"]
|
||||||
phpts: [ts, nts]
|
phpts: [ts, nts]
|
||||||
exclude:
|
exclude:
|
||||||
# ext-php-rs requires nightly Rust when on Windows.
|
# ext-php-rs requires nightly Rust when on Windows.
|
||||||
@ -28,6 +28,12 @@ jobs:
|
|||||||
phpts: ts
|
phpts: ts
|
||||||
- os: ubuntu-latest
|
- os: ubuntu-latest
|
||||||
phpts: ts
|
phpts: ts
|
||||||
|
- os: macos-latest
|
||||||
|
clang: "17"
|
||||||
|
- os: ubuntu-latest
|
||||||
|
clang: "15"
|
||||||
|
- os: windows-latest
|
||||||
|
clang: "15"
|
||||||
env:
|
env:
|
||||||
CARGO_TERM_COLOR: always
|
CARGO_TERM_COLOR: always
|
||||||
steps:
|
steps:
|
||||||
@ -83,10 +89,10 @@ jobs:
|
|||||||
- name: Build
|
- name: Build
|
||||||
env:
|
env:
|
||||||
EXT_PHP_RS_TEST: ""
|
EXT_PHP_RS_TEST: ""
|
||||||
run: cargo build --release --all-features --all
|
run: cargo build --release --features closure,anyhow --all
|
||||||
# Test & lint
|
# Test & lint
|
||||||
- name: Test inline examples
|
- name: Test inline examples
|
||||||
run: cargo test --release --all --all-features
|
run: cargo test --release --all --features closure,anyhow
|
||||||
- name: Run rustfmt
|
- name: Run rustfmt
|
||||||
if: matrix.rust == 'stable' && matrix.os == 'ubuntu-latest' && matrix.php == '8.2'
|
if: matrix.rust == 'stable' && matrix.os == 'ubuntu-latest' && matrix.php == '8.2'
|
||||||
run: cargo fmt --all -- --check
|
run: cargo fmt --all -- --check
|
||||||
@ -110,3 +116,11 @@ jobs:
|
|||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v3
|
||||||
- name: Build
|
- name: Build
|
||||||
uses: ./.github/actions/zts
|
uses: ./.github/actions/zts
|
||||||
|
test-embed:
|
||||||
|
name: Test with embed
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
- name: Test
|
||||||
|
uses: ./.github/actions/embed
|
||||||
|
2
.github/workflows/docs.yml
vendored
2
.github/workflows/docs.yml
vendored
@ -16,7 +16,7 @@ jobs:
|
|||||||
matrix:
|
matrix:
|
||||||
os: ["ubuntu-latest"]
|
os: ["ubuntu-latest"]
|
||||||
php: ["8.2"]
|
php: ["8.2"]
|
||||||
clang: ["14"]
|
clang: ["17"]
|
||||||
mdbook: ["latest"]
|
mdbook: ["latest"]
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout code
|
- name: Checkout code
|
||||||
|
17
Cargo.toml
17
Cargo.toml
@ -5,7 +5,7 @@ repository = "https://github.com/davidcole1340/ext-php-rs"
|
|||||||
homepage = "https://github.com/davidcole1340/ext-php-rs"
|
homepage = "https://github.com/davidcole1340/ext-php-rs"
|
||||||
license = "MIT OR Apache-2.0"
|
license = "MIT OR Apache-2.0"
|
||||||
keywords = ["php", "ffi", "zend"]
|
keywords = ["php", "ffi", "zend"]
|
||||||
version = "0.10.1"
|
version = "0.10.5"
|
||||||
authors = ["David Cole <david.cole1340@gmail.com>"]
|
authors = ["David Cole <david.cole1340@gmail.com>"]
|
||||||
edition = "2018"
|
edition = "2018"
|
||||||
categories = ["api-bindings"]
|
categories = ["api-bindings"]
|
||||||
@ -17,30 +17,31 @@ parking_lot = "0.12"
|
|||||||
cfg-if = "1.0"
|
cfg-if = "1.0"
|
||||||
once_cell = "1.17"
|
once_cell = "1.17"
|
||||||
anyhow = { version = "1", optional = true }
|
anyhow = { version = "1", optional = true }
|
||||||
ext-php-rs-derive = { version = "=0.10.0", path = "./crates/macros" }
|
ext-php-rs-derive = { version = "=0.10.1", path = "./crates/macros" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
skeptic = "0.13"
|
skeptic = "0.13"
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
anyhow = "1"
|
anyhow = "1"
|
||||||
bindgen = "0.65.1"
|
bindgen = "0.68.1"
|
||||||
cc = "1.0"
|
cc = "1.0"
|
||||||
skeptic = "0.13"
|
skeptic = "0.13"
|
||||||
|
|
||||||
[target.'cfg(windows)'.build-dependencies]
|
[target.'cfg(windows)'.build-dependencies]
|
||||||
ureq = { version = "2.4", features = ["native-tls", "gzip"], default-features = false }
|
ureq = { version = "2.4", features = [
|
||||||
|
"native-tls",
|
||||||
|
"gzip",
|
||||||
|
], default-features = false }
|
||||||
native-tls = "0.2"
|
native-tls = "0.2"
|
||||||
zip = "0.6"
|
zip = "0.6"
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
closure = []
|
closure = []
|
||||||
|
embed = []
|
||||||
|
|
||||||
[workspace]
|
[workspace]
|
||||||
members = [
|
members = ["crates/macros", "crates/cli"]
|
||||||
"crates/macros",
|
|
||||||
"crates/cli"
|
|
||||||
]
|
|
||||||
|
|
||||||
[package.metadata.docs.rs]
|
[package.metadata.docs.rs]
|
||||||
rustdoc-args = ["--cfg", "docs"]
|
rustdoc-args = ["--cfg", "docs"]
|
||||||
|
@ -26,6 +26,8 @@ bind! {
|
|||||||
_efree,
|
_efree,
|
||||||
_emalloc,
|
_emalloc,
|
||||||
_zend_executor_globals,
|
_zend_executor_globals,
|
||||||
|
_sapi_globals_struct,
|
||||||
|
_sapi_module_struct,
|
||||||
_zend_expected_type,
|
_zend_expected_type,
|
||||||
_zend_expected_type_Z_EXPECTED_ARRAY,
|
_zend_expected_type_Z_EXPECTED_ARRAY,
|
||||||
_zend_expected_type_Z_EXPECTED_BOOL,
|
_zend_expected_type_Z_EXPECTED_BOOL,
|
||||||
@ -55,6 +57,8 @@ bind! {
|
|||||||
zend_array_destroy,
|
zend_array_destroy,
|
||||||
zend_array_dup,
|
zend_array_dup,
|
||||||
zend_call_known_function,
|
zend_call_known_function,
|
||||||
|
zend_fetch_function_str,
|
||||||
|
zend_hash_str_find_ptr_lc,
|
||||||
zend_ce_argument_count_error,
|
zend_ce_argument_count_error,
|
||||||
zend_ce_arithmetic_error,
|
zend_ce_arithmetic_error,
|
||||||
zend_ce_compile_error,
|
zend_ce_compile_error,
|
||||||
@ -99,6 +103,8 @@ bind! {
|
|||||||
zend_objects_clone_members,
|
zend_objects_clone_members,
|
||||||
zend_register_bool_constant,
|
zend_register_bool_constant,
|
||||||
zend_register_double_constant,
|
zend_register_double_constant,
|
||||||
|
zend_register_ini_entries,
|
||||||
|
zend_ini_entry_def,
|
||||||
zend_register_internal_class_ex,
|
zend_register_internal_class_ex,
|
||||||
zend_register_long_constant,
|
zend_register_long_constant,
|
||||||
zend_register_string_constant,
|
zend_register_string_constant,
|
||||||
@ -106,6 +112,7 @@ bind! {
|
|||||||
zend_string,
|
zend_string,
|
||||||
zend_string_init_interned,
|
zend_string_init_interned,
|
||||||
zend_throw_exception_ex,
|
zend_throw_exception_ex,
|
||||||
|
zend_throw_exception_object,
|
||||||
zend_type,
|
zend_type,
|
||||||
zend_value,
|
zend_value,
|
||||||
zend_wrong_parameters_count_error,
|
zend_wrong_parameters_count_error,
|
||||||
@ -137,6 +144,7 @@ bind! {
|
|||||||
IS_CONSTANT_AST_EX,
|
IS_CONSTANT_AST_EX,
|
||||||
IS_DOUBLE,
|
IS_DOUBLE,
|
||||||
IS_FALSE,
|
IS_FALSE,
|
||||||
|
IS_INDIRECT,
|
||||||
IS_INTERNED_STRING_EX,
|
IS_INTERNED_STRING_EX,
|
||||||
IS_LONG,
|
IS_LONG,
|
||||||
IS_MIXED,
|
IS_MIXED,
|
||||||
@ -157,6 +165,10 @@ bind! {
|
|||||||
IS_PTR,
|
IS_PTR,
|
||||||
MAY_BE_ANY,
|
MAY_BE_ANY,
|
||||||
MAY_BE_BOOL,
|
MAY_BE_BOOL,
|
||||||
|
PHP_INI_USER,
|
||||||
|
PHP_INI_PERDIR,
|
||||||
|
PHP_INI_SYSTEM,
|
||||||
|
PHP_INI_ALL,
|
||||||
USING_ZTS,
|
USING_ZTS,
|
||||||
ZEND_ACC_ABSTRACT,
|
ZEND_ACC_ABSTRACT,
|
||||||
ZEND_ACC_ANON_CLASS,
|
ZEND_ACC_ANON_CLASS,
|
||||||
@ -233,6 +245,8 @@ bind! {
|
|||||||
zend_class_serialize_deny,
|
zend_class_serialize_deny,
|
||||||
zend_class_unserialize_deny,
|
zend_class_unserialize_deny,
|
||||||
zend_executor_globals,
|
zend_executor_globals,
|
||||||
|
sapi_globals_struct,
|
||||||
|
sapi_module_struct,
|
||||||
zend_objects_store_del,
|
zend_objects_store_del,
|
||||||
zend_hash_move_forward_ex,
|
zend_hash_move_forward_ex,
|
||||||
zend_hash_get_current_key_type_ex,
|
zend_hash_get_current_key_type_ex,
|
||||||
@ -247,6 +261,8 @@ bind! {
|
|||||||
core_globals,
|
core_globals,
|
||||||
sapi_globals_struct,
|
sapi_globals_struct,
|
||||||
sapi_globals,
|
sapi_globals,
|
||||||
|
sapi_globals,
|
||||||
|
sapi_module,
|
||||||
php_printf,
|
php_printf,
|
||||||
__zend_malloc,
|
__zend_malloc,
|
||||||
tsrm_get_ls_cache,
|
tsrm_get_ls_cache,
|
||||||
@ -268,6 +284,13 @@ bind! {
|
|||||||
zend_is_auto_global,
|
zend_is_auto_global,
|
||||||
zend_llist_get_next_ex,
|
zend_llist_get_next_ex,
|
||||||
zend_llist_get_prev_ex,
|
zend_llist_get_prev_ex,
|
||||||
|
sapi_globals_offset,
|
||||||
zend_atomic_bool_store,
|
zend_atomic_bool_store,
|
||||||
zend_interrupt_function
|
zend_interrupt_function,
|
||||||
|
zend_eval_string,
|
||||||
|
zend_file_handle,
|
||||||
|
zend_stream_init_filename,
|
||||||
|
php_execute_script,
|
||||||
|
zend_register_module_ex,
|
||||||
|
_zend_bailout
|
||||||
}
|
}
|
||||||
|
38
build.rs
38
build.rs
@ -16,7 +16,7 @@ use bindgen::RustTarget;
|
|||||||
use impl_::Provider;
|
use impl_::Provider;
|
||||||
|
|
||||||
const MIN_PHP_API_VER: u32 = 20200930;
|
const MIN_PHP_API_VER: u32 = 20200930;
|
||||||
const MAX_PHP_API_VER: u32 = 20220829;
|
const MAX_PHP_API_VER: u32 = 20230831;
|
||||||
|
|
||||||
pub trait PHPProvider<'a>: Sized {
|
pub trait PHPProvider<'a>: Sized {
|
||||||
/// Create a new PHP provider.
|
/// Create a new PHP provider.
|
||||||
@ -151,9 +151,31 @@ fn build_wrapper(defines: &[(&str, &str)], includes: &[PathBuf]) -> Result<()> {
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
/// Builds the embed library.
|
||||||
|
fn build_embed(defines: &[(&str, &str)], includes: &[PathBuf]) -> Result<()> {
|
||||||
|
let mut build = cc::Build::new();
|
||||||
|
for (var, val) in defines {
|
||||||
|
build.define(var, *val);
|
||||||
|
}
|
||||||
|
build
|
||||||
|
.file("src/embed/embed.c")
|
||||||
|
.includes(includes)
|
||||||
|
.try_compile("embed")
|
||||||
|
.context("Failed to compile ext-php-rs C embed interface")?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// Generates bindings to the Zend API.
|
/// Generates bindings to the Zend API.
|
||||||
fn generate_bindings(defines: &[(&str, &str)], includes: &[PathBuf]) -> Result<String> {
|
fn generate_bindings(defines: &[(&str, &str)], includes: &[PathBuf]) -> Result<String> {
|
||||||
let mut bindgen = bindgen::Builder::default()
|
let mut bindgen = bindgen::Builder::default();
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
{
|
||||||
|
bindgen = bindgen.header("src/embed/embed.h");
|
||||||
|
}
|
||||||
|
|
||||||
|
bindgen = bindgen
|
||||||
.header("src/wrapper.h")
|
.header("src/wrapper.h")
|
||||||
.clang_args(
|
.clang_args(
|
||||||
includes
|
includes
|
||||||
@ -206,6 +228,8 @@ fn check_php_version(info: &PHPInfo) -> Result<()> {
|
|||||||
|
|
||||||
const PHP_82_API_VER: u32 = 20220829;
|
const PHP_82_API_VER: u32 = 20220829;
|
||||||
|
|
||||||
|
const PHP_83_API_VER: u32 = 20230831;
|
||||||
|
|
||||||
println!("cargo:rustc-cfg=php80");
|
println!("cargo:rustc-cfg=php80");
|
||||||
|
|
||||||
if (PHP_81_API_VER..PHP_82_API_VER).contains(&version) {
|
if (PHP_81_API_VER..PHP_82_API_VER).contains(&version) {
|
||||||
@ -216,6 +240,10 @@ fn check_php_version(info: &PHPInfo) -> Result<()> {
|
|||||||
println!("cargo:rustc-cfg=php82");
|
println!("cargo:rustc-cfg=php82");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if version >= PHP_83_API_VER {
|
||||||
|
println!("cargo:rustc-cfg=php83");
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -226,6 +254,8 @@ fn main() -> Result<()> {
|
|||||||
for path in [
|
for path in [
|
||||||
manifest.join("src").join("wrapper.h"),
|
manifest.join("src").join("wrapper.h"),
|
||||||
manifest.join("src").join("wrapper.c"),
|
manifest.join("src").join("wrapper.c"),
|
||||||
|
manifest.join("src").join("embed").join("embed.h"),
|
||||||
|
manifest.join("src").join("embed").join("embed.c"),
|
||||||
manifest.join("allowed_bindings.rs"),
|
manifest.join("allowed_bindings.rs"),
|
||||||
manifest.join("windows_build.rs"),
|
manifest.join("windows_build.rs"),
|
||||||
manifest.join("unix_build.rs"),
|
manifest.join("unix_build.rs"),
|
||||||
@ -257,6 +287,10 @@ fn main() -> Result<()> {
|
|||||||
|
|
||||||
check_php_version(&info)?;
|
check_php_version(&info)?;
|
||||||
build_wrapper(&defines, &includes)?;
|
build_wrapper(&defines, &includes)?;
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
build_embed(&defines, &includes)?;
|
||||||
|
|
||||||
let bindings = generate_bindings(&defines, &includes)?;
|
let bindings = generate_bindings(&defines, &includes)?;
|
||||||
|
|
||||||
let out_file =
|
let out_file =
|
||||||
|
@ -5,7 +5,7 @@ repository = "https://github.com/davidcole1340/ext-php-rs"
|
|||||||
homepage = "https://github.com/davidcole1340/ext-php-rs"
|
homepage = "https://github.com/davidcole1340/ext-php-rs"
|
||||||
license = "MIT OR Apache-2.0"
|
license = "MIT OR Apache-2.0"
|
||||||
keywords = ["php", "ffi", "zend"]
|
keywords = ["php", "ffi", "zend"]
|
||||||
version = "0.1.8"
|
version = "0.1.9"
|
||||||
authors = ["David Cole <david.cole1340@gmail.com>"]
|
authors = ["David Cole <david.cole1340@gmail.com>"]
|
||||||
edition = "2018"
|
edition = "2018"
|
||||||
categories = ["api-bindings", "command-line-interface"]
|
categories = ["api-bindings", "command-line-interface"]
|
||||||
|
@ -4,6 +4,7 @@ use anyhow::{Context, Result};
|
|||||||
use ext_php_rs::describe::Description;
|
use ext_php_rs::describe::Description;
|
||||||
use libloading::os::unix::{Library, Symbol};
|
use libloading::os::unix::{Library, Symbol};
|
||||||
|
|
||||||
|
#[allow(improper_ctypes_definitions)]
|
||||||
pub struct Ext {
|
pub struct Ext {
|
||||||
// These need to be here to keep the libraries alive. The extension library needs to be alive
|
// These need to be here to keep the libraries alive. The extension library needs to be alive
|
||||||
// to access the describe function. Missing here is the lifetime on `Symbol<'a, fn() ->
|
// to access the describe function. Missing here is the lifetime on `Symbol<'a, fn() ->
|
||||||
|
@ -4,7 +4,7 @@ description = "Derive macros for ext-php-rs."
|
|||||||
repository = "https://github.com/davidcole1340/ext-php-rs"
|
repository = "https://github.com/davidcole1340/ext-php-rs"
|
||||||
homepage = "https://github.com/davidcole1340/ext-php-rs"
|
homepage = "https://github.com/davidcole1340/ext-php-rs"
|
||||||
license = "MIT OR Apache-2.0"
|
license = "MIT OR Apache-2.0"
|
||||||
version = "0.10.0"
|
version = "0.10.1"
|
||||||
authors = ["David Cole <david.cole1340@gmail.com>"]
|
authors = ["David Cole <david.cole1340@gmail.com>"]
|
||||||
edition = "2018"
|
edition = "2018"
|
||||||
|
|
||||||
|
@ -26,6 +26,7 @@ pub struct Arg {
|
|||||||
pub ty: String,
|
pub ty: String,
|
||||||
pub nullable: bool,
|
pub nullable: bool,
|
||||||
pub default: Option<String>,
|
pub default: Option<String>,
|
||||||
|
pub as_ref: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
@ -249,12 +250,19 @@ pub fn get_return_type(output_type: &ReturnType) -> Result<Option<(String, bool)
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Arg {
|
impl Arg {
|
||||||
pub fn new(name: String, ty: String, nullable: bool, default: Option<String>) -> Self {
|
pub fn new(
|
||||||
|
name: String,
|
||||||
|
ty: String,
|
||||||
|
nullable: bool,
|
||||||
|
default: Option<String>,
|
||||||
|
as_ref: bool,
|
||||||
|
) -> Self {
|
||||||
Self {
|
Self {
|
||||||
name,
|
name,
|
||||||
ty,
|
ty,
|
||||||
nullable,
|
nullable,
|
||||||
default,
|
default,
|
||||||
|
as_ref,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -268,6 +276,7 @@ impl Arg {
|
|||||||
match ty {
|
match ty {
|
||||||
Type::Path(TypePath { path, .. }) => {
|
Type::Path(TypePath { path, .. }) => {
|
||||||
let mut path = path.clone();
|
let mut path = path.clone();
|
||||||
|
let mut pass_by_ref = false;
|
||||||
path.drop_lifetimes();
|
path.drop_lifetimes();
|
||||||
|
|
||||||
let seg = path.segments.last()?;
|
let seg = path.segments.last()?;
|
||||||
@ -283,9 +292,45 @@ impl Arg {
|
|||||||
None
|
None
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// For for types that are `Option<&mut T>` to turn them into `Option<&T>`,
|
||||||
|
// marking the Arg as as "passed by reference".
|
||||||
|
let option = Some(seg)
|
||||||
|
.filter(|seg| seg.ident == "Option")
|
||||||
|
.and_then(|seg| {
|
||||||
|
if let PathArguments::AngleBracketed(args) = &seg.arguments {
|
||||||
|
args.args
|
||||||
|
.iter()
|
||||||
|
.find(|arg| matches!(arg, GenericArgument::Type(_)))
|
||||||
|
.map(|ga| {
|
||||||
|
let new_ga = match ga {
|
||||||
|
GenericArgument::Type(ty) => {
|
||||||
|
let _rtype = match ty {
|
||||||
|
Type::Reference(r) => {
|
||||||
|
let mut new_ref = r.clone();
|
||||||
|
new_ref.mutability = None;
|
||||||
|
pass_by_ref = true;
|
||||||
|
Type::Reference(new_ref)
|
||||||
|
}
|
||||||
|
_ => ty.clone(),
|
||||||
|
};
|
||||||
|
GenericArgument::Type(_rtype)
|
||||||
|
}
|
||||||
|
_ => ga.clone(),
|
||||||
|
};
|
||||||
|
new_ga.to_token_stream().to_string()
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
let stringified = match result {
|
let stringified = match result {
|
||||||
Some(result) if is_return => result,
|
Some(result) if is_return => result,
|
||||||
_ => path.to_token_stream().to_string(),
|
_ => match option {
|
||||||
|
Some(result) => result,
|
||||||
|
None => path.to_token_stream().to_string(),
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
Some(Arg::new(
|
Some(Arg::new(
|
||||||
@ -293,6 +338,7 @@ impl Arg {
|
|||||||
stringified,
|
stringified,
|
||||||
seg.ident == "Option" || default.is_some(),
|
seg.ident == "Option" || default.is_some(),
|
||||||
default,
|
default,
|
||||||
|
pass_by_ref,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
Type::Reference(ref_) => {
|
Type::Reference(ref_) => {
|
||||||
@ -302,6 +348,7 @@ impl Arg {
|
|||||||
ref_.to_token_stream().to_string(),
|
ref_.to_token_stream().to_string(),
|
||||||
false,
|
false,
|
||||||
default,
|
default,
|
||||||
|
ref_.mutability.is_some(),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
_ => None,
|
_ => None,
|
||||||
@ -361,6 +408,7 @@ impl Arg {
|
|||||||
let ty = self.get_type_ident();
|
let ty = self.get_type_ident();
|
||||||
|
|
||||||
let null = self.nullable.then(|| quote! { .allow_null() });
|
let null = self.nullable.then(|| quote! { .allow_null() });
|
||||||
|
let passed_by_ref = self.as_ref.then(|| quote! { .as_ref() });
|
||||||
let default = self.default.as_ref().map(|val| {
|
let default = self.default.as_ref().map(|val| {
|
||||||
quote! {
|
quote! {
|
||||||
.default(#val)
|
.default(#val)
|
||||||
@ -368,7 +416,7 @@ impl Arg {
|
|||||||
});
|
});
|
||||||
|
|
||||||
quote! {
|
quote! {
|
||||||
::ext_php_rs::args::Arg::new(#name, #ty) #null #default
|
::ext_php_rs::args::Arg::new(#name, #ty) #null #passed_by_ref #default
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -23,12 +23,16 @@
|
|||||||
- [Object](./types/object.md)
|
- [Object](./types/object.md)
|
||||||
- [Class Object](./types/class_object.md)
|
- [Class Object](./types/class_object.md)
|
||||||
- [Closure](./types/closure.md)
|
- [Closure](./types/closure.md)
|
||||||
|
- [Functions & methods](./types/functions.md)
|
||||||
|
- [Async futures](./macros/async_impl.md)
|
||||||
- [Macros](./macros/index.md)
|
- [Macros](./macros/index.md)
|
||||||
- [Module](./macros/module.md)
|
- [Module](./macros/module.md)
|
||||||
- [Module Startup Function](./macros/module_startup.md)
|
- [Module Startup Function](./macros/module_startup.md)
|
||||||
- [Function](./macros/function.md)
|
- [Function](./macros/function.md)
|
||||||
- [Classes](./macros/classes.md)
|
- [Classes](./macros/classes.md)
|
||||||
- [`impl`s](./macros/impl.md)
|
- [`impl`s](./macros/impl.md)
|
||||||
|
- [async `impl`s](./macros/async_impl.md)
|
||||||
- [Constants](./macros/constant.md)
|
- [Constants](./macros/constant.md)
|
||||||
- [`ZvalConvert`](./macros/zval_convert.md)
|
- [`ZvalConvert`](./macros/zval_convert.md)
|
||||||
- [Exceptions](./exceptions.md)
|
- [Exceptions](./exceptions.md)
|
||||||
|
- [INI Settings](./ini-settings.md)
|
||||||
|
48
guide/src/ini-settings.md
Normal file
48
guide/src/ini-settings.md
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
# INI Settings
|
||||||
|
|
||||||
|
Your PHP Extension may want to provide it's own PHP INI settings to configure behaviour. This can be done in the `#[php_startup]` annotated startup function.
|
||||||
|
|
||||||
|
## Registering INI Settings
|
||||||
|
|
||||||
|
All PHP INI definitions must be registered with PHP to get / set their values via the `php.ini` file or `ini_get() / ini_set()`.
|
||||||
|
|
||||||
|
|
||||||
|
```rust,no_run
|
||||||
|
# #![cfg_attr(windows, feature(abi_vectorcall))]
|
||||||
|
# extern crate ext_php_rs;
|
||||||
|
# use ext_php_rs::prelude::*;
|
||||||
|
# use ext_php_rs::zend::IniEntryDef;
|
||||||
|
# use ext_php_rs::flags::IniEntryPermission;
|
||||||
|
|
||||||
|
#[php_startup]
|
||||||
|
pub fn startup_function(ty: i32, module_number: i32) {
|
||||||
|
let ini_entries: Vec<IniEntryDef> = vec![
|
||||||
|
IniEntryDef::new(
|
||||||
|
"my_extension.display_emoji".to_owned(),
|
||||||
|
"yes".to_owned(),
|
||||||
|
IniEntryPermission::All,
|
||||||
|
),
|
||||||
|
];
|
||||||
|
IniEntryDef::register(ini_entries, module_number);
|
||||||
|
}
|
||||||
|
# fn main() {}
|
||||||
|
```
|
||||||
|
|
||||||
|
## Getting INI Settings
|
||||||
|
|
||||||
|
The INI values are stored as part of the `GlobalExecutor`, and can be accessed via the `ini_values()` function. To retrieve the value for a registered INI setting
|
||||||
|
|
||||||
|
```rust,no_run
|
||||||
|
# #![cfg_attr(windows, feature(abi_vectorcall))]
|
||||||
|
# extern crate ext_php_rs;
|
||||||
|
# use ext_php_rs::prelude::*;
|
||||||
|
# use ext_php_rs::zend::ExecutorGlobals;
|
||||||
|
|
||||||
|
#[php_startup]
|
||||||
|
pub fn startup_function(ty: i32, module_number: i32) {
|
||||||
|
// Get all INI values
|
||||||
|
let ini_values = ExecutorGlobals::get().ini_values(); // HashMap<String, Option<String>>
|
||||||
|
let my_ini_value = ini_values.get("my_extension.display_emoji"); // Option<Option<String>>
|
||||||
|
}
|
||||||
|
# fn main() {}
|
||||||
|
```
|
130
guide/src/macros/async_impl.md
Normal file
130
guide/src/macros/async_impl.md
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
# `#[php_async_impl]`
|
||||||
|
|
||||||
|
Using `#[php_async_impl]` instead of `#[php_impl]` allows us to expose any async Rust library to PHP, using [PHP fibers](https://www.php.net/manual/en/language.fibers.php), [php-tokio](https://github.com/danog/php-tokio) and the [PHP Revolt event loop](https://revolt.run) under the hood to handle async interoperability.
|
||||||
|
|
||||||
|
This allows full compatibility with [amphp](https://amphp.org), [PSL](https://github.com/azjezz/psl), [reactphp](https://reactphp.org) and any other async PHP library based on [Revolt](https://revolt.run).
|
||||||
|
|
||||||
|
Traits annotated with `#[php_async_impl]` can freely expose any async function, using `await` and any async Rust library.
|
||||||
|
|
||||||
|
Make sure to also expose the `php_tokio::EventLoop::init` and `php_tokio::EventLoop::wakeup` functions to PHP in order to initialize the event loop, as specified in the full example [here »](#async-example).
|
||||||
|
|
||||||
|
Also, make sure to invoke `EventLoop::shutdown` in the request shutdown handler to clean up the tokio event loop before finishing the request.
|
||||||
|
|
||||||
|
## Async example
|
||||||
|
|
||||||
|
In this example, we're exposing an async Rust HTTP client library called [reqwest](https://docs.rs/reqwest/latest/reqwest/) to PHP, using [PHP fibers](https://www.php.net/manual/en/language.fibers.php), [php-tokio](https://github.com/danog/php-tokio) and the [PHP Revolt event loop](https://revolt.run) under the hood to handle async interoperability.
|
||||||
|
|
||||||
|
This allows full compatibility with [amphp](https://amphp.org), [PSL](https://github.com/azjezz/psl), [reactphp](https://reactphp.org) and any other async PHP library based on [Revolt](https://revolt.run).
|
||||||
|
|
||||||
|
Make sure to require [php-tokio](https://github.com/danog/php-tokio) as a dependency before proceeding.
|
||||||
|
|
||||||
|
```rust,ignore
|
||||||
|
use ext_php_rs::prelude::*;
|
||||||
|
use php_tokio::{php_async_impl, EventLoop};
|
||||||
|
|
||||||
|
#[php_class]
|
||||||
|
struct Client {}
|
||||||
|
|
||||||
|
#[php_async_impl]
|
||||||
|
impl Client {
|
||||||
|
pub fn init() -> PhpResult<u64> {
|
||||||
|
EventLoop::init()
|
||||||
|
}
|
||||||
|
pub fn wakeup() -> PhpResult<()> {
|
||||||
|
EventLoop::wakeup()
|
||||||
|
}
|
||||||
|
pub async fn get(url: &str) -> anyhow::Result<String> {
|
||||||
|
Ok(reqwest::get(url).await?.text().await?)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub extern "C" fn request_shutdown(_type: i32, _module_number: i32) -> i32 {
|
||||||
|
EventLoop::shutdown();
|
||||||
|
0
|
||||||
|
}
|
||||||
|
|
||||||
|
#[php_module]
|
||||||
|
pub fn get_module(module: ModuleBuilder) -> ModuleBuilder {
|
||||||
|
module.request_shutdown_function(request_shutdown)
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Here's the async PHP code we use to interact with the Rust class we just exposed.
|
||||||
|
|
||||||
|
The `Client::init` method needs to be called only once in order to initialize the Revolt event loop and link it to the Tokio event loop, as shown by the following code.
|
||||||
|
|
||||||
|
See [here »](https://amphp.org) for more info on async PHP using [amphp](https://amphp.org) + [revolt](https://revolt.run).
|
||||||
|
|
||||||
|
```php
|
||||||
|
<?php declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Reqwest;
|
||||||
|
|
||||||
|
use Revolt\EventLoop;
|
||||||
|
|
||||||
|
use function Amp\async;
|
||||||
|
use function Amp\Future\await;
|
||||||
|
|
||||||
|
final class Client
|
||||||
|
{
|
||||||
|
private static ?string $id = null;
|
||||||
|
|
||||||
|
public static function init(): void
|
||||||
|
{
|
||||||
|
if (self::$id !== null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$f = \fopen("php://fd/".\Client::init(), 'r+');
|
||||||
|
\stream_set_blocking($f, false);
|
||||||
|
self::$id = EventLoop::onReadable($f, fn () => \Client::wakeup());
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function reference(): void
|
||||||
|
{
|
||||||
|
EventLoop::reference(self::$id);
|
||||||
|
}
|
||||||
|
public static function unreference(): void
|
||||||
|
{
|
||||||
|
EventLoop::unreference(self::$id);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function __callStatic(string $name, array $args): mixed
|
||||||
|
{
|
||||||
|
return \Client::$name(...$args);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Client::init();
|
||||||
|
|
||||||
|
function test(int $delay): void
|
||||||
|
{
|
||||||
|
$url = "https://httpbin.org/delay/$delay";
|
||||||
|
$t = time();
|
||||||
|
echo "Making async reqwest to $url that will return after $delay seconds...".PHP_EOL;
|
||||||
|
Client::get($url);
|
||||||
|
$t = time() - $t;
|
||||||
|
echo "Got response from $url after ~".$t." seconds!".PHP_EOL;
|
||||||
|
};
|
||||||
|
|
||||||
|
$futures = [];
|
||||||
|
$futures []= async(test(...), 5);
|
||||||
|
$futures []= async(test(...), 5);
|
||||||
|
$futures []= async(test(...), 5);
|
||||||
|
|
||||||
|
await($futures);
|
||||||
|
```
|
||||||
|
|
||||||
|
Result:
|
||||||
|
|
||||||
|
```
|
||||||
|
Making async reqwest to https://httpbin.org/delay/5 that will return after 5 seconds...
|
||||||
|
Making async reqwest to https://httpbin.org/delay/5 that will return after 5 seconds...
|
||||||
|
Making async reqwest to https://httpbin.org/delay/5 that will return after 5 seconds...
|
||||||
|
Got response from https://httpbin.org/delay/5 after ~5 seconds!
|
||||||
|
Got response from https://httpbin.org/delay/5 after ~5 seconds!
|
||||||
|
Got response from https://httpbin.org/delay/5 after ~5 seconds!
|
||||||
|
```
|
||||||
|
|
||||||
|
[`php_function`]: ./function.md
|
@ -8,6 +8,8 @@ implementations cannot be exported to PHP.
|
|||||||
If you do not want a function exported to PHP, you should place it in a separate
|
If you do not want a function exported to PHP, you should place it in a separate
|
||||||
`impl` block.
|
`impl` block.
|
||||||
|
|
||||||
|
If you want to use async Rust, use `#[php_async_impl]`, instead: see [here »](./async_impl.md) for more info.
|
||||||
|
|
||||||
## Methods
|
## Methods
|
||||||
|
|
||||||
Methods basically follow the same rules as functions, so read about the
|
Methods basically follow the same rules as functions, so read about the
|
||||||
@ -162,4 +164,4 @@ var_dump(Human::get_max_age()); // int(100)
|
|||||||
var_dump(Human::MAX_AGE); // int(100)
|
var_dump(Human::MAX_AGE); // int(100)
|
||||||
```
|
```
|
||||||
|
|
||||||
[`php_function`]: ./function.md
|
[`php_async_impl`]: ./async_impl.md
|
||||||
|
@ -45,3 +45,17 @@ pub fn test_bool(input: bool) -> String {
|
|||||||
var_dump(test_bool(true)); // string(4) "Yes!"
|
var_dump(test_bool(true)); // string(4) "Yes!"
|
||||||
var_dump(test_bool(false)); // string(3) "No!"
|
var_dump(test_bool(false)); // string(3) "No!"
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Rust example, taking by reference
|
||||||
|
|
||||||
|
```rust,no_run
|
||||||
|
# #![cfg_attr(windows, feature(abi_vectorcall))]
|
||||||
|
# extern crate ext_php_rs;
|
||||||
|
# use ext_php_rs::prelude::*;
|
||||||
|
# use ext_php_rs::types;
|
||||||
|
#[php_function]
|
||||||
|
pub fn test_bool(input: &mut types::Zval) {
|
||||||
|
input.reference_mut().unwrap().set_bool(false);
|
||||||
|
}
|
||||||
|
# fn main() {}
|
||||||
|
```
|
||||||
|
28
guide/src/types/functions.md
Normal file
28
guide/src/types/functions.md
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
# Functions & methods
|
||||||
|
|
||||||
|
PHP functions and methods are represented by the `Function` struct.
|
||||||
|
|
||||||
|
You can use the `try_from_function` and `try_from_method` methods to obtain a Function struct corresponding to the passed function or static method name.
|
||||||
|
It's heavily recommended you reuse returned `Function` objects, to avoid the overhead of looking up the function/method name.
|
||||||
|
|
||||||
|
```rust,no_run
|
||||||
|
# #![cfg_attr(windows, feature(abi_vectorcall))]
|
||||||
|
# extern crate ext_php_rs;
|
||||||
|
use ext_php_rs::prelude::*;
|
||||||
|
|
||||||
|
use ext_php_rs::zend::Function;
|
||||||
|
|
||||||
|
#[php_function]
|
||||||
|
pub fn test_function() -> () {
|
||||||
|
let var_dump = Function::try_from_function("var_dump").unwrap();
|
||||||
|
let _ = var_dump.try_call(vec![&"abc"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[php_function]
|
||||||
|
pub fn test_method() -> () {
|
||||||
|
let f = Function::try_from_method("ClassName", "staticMethod").unwrap();
|
||||||
|
let _ = f.try_call(vec![&"abc"]);
|
||||||
|
}
|
||||||
|
|
||||||
|
# fn main() {}
|
||||||
|
```
|
@ -15,6 +15,25 @@ object.
|
|||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
|
|
||||||
|
### Calling a method
|
||||||
|
|
||||||
|
```rust,no_run
|
||||||
|
# #![cfg_attr(windows, feature(abi_vectorcall))]
|
||||||
|
# extern crate ext_php_rs;
|
||||||
|
use ext_php_rs::{prelude::*, types::ZendObject};
|
||||||
|
|
||||||
|
// Take an object reference and also return it.
|
||||||
|
#[php_function]
|
||||||
|
pub fn take_obj(obj: &mut ZendObject) -> () {
|
||||||
|
let _ = obj.try_call_method("hello", vec![&"arg1", &"arg2"]);
|
||||||
|
}
|
||||||
|
# #[php_module]
|
||||||
|
# pub fn get_module(module: ModuleBuilder) -> ModuleBuilder {
|
||||||
|
# module
|
||||||
|
# }
|
||||||
|
# fn main() {}
|
||||||
|
```
|
||||||
|
|
||||||
### Taking an object reference
|
### Taking an object reference
|
||||||
|
|
||||||
```rust,no_run
|
```rust,no_run
|
||||||
|
@ -123,6 +123,7 @@ impl<'a> Arg<'a> {
|
|||||||
/// # Parameters
|
/// # Parameters
|
||||||
///
|
///
|
||||||
/// * `params` - A list of parameters to call the function with.
|
/// * `params` - A list of parameters to call the function with.
|
||||||
|
#[inline(always)]
|
||||||
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
||||||
self.zval.as_ref().ok_or(Error::Callable)?.try_call(params)
|
self.zval.as_ref().ok_or(Error::Callable)?.try_call(params)
|
||||||
}
|
}
|
||||||
|
@ -161,10 +161,10 @@ impl ClassBuilder {
|
|||||||
/// Panics if the class name associated with `T` is not the same as the
|
/// Panics if the class name associated with `T` is not the same as the
|
||||||
/// class name specified when creating the builder.
|
/// class name specified when creating the builder.
|
||||||
pub fn object_override<T: RegisteredClass>(mut self) -> Self {
|
pub fn object_override<T: RegisteredClass>(mut self) -> Self {
|
||||||
extern "C" fn create_object<T: RegisteredClass>(_: *mut ClassEntry) -> *mut ZendObject {
|
extern "C" fn create_object<T: RegisteredClass>(ce: *mut ClassEntry) -> *mut ZendObject {
|
||||||
// SAFETY: After calling this function, PHP will always call the constructor
|
// SAFETY: After calling this function, PHP will always call the constructor
|
||||||
// defined below, which assumes that the object is uninitialized.
|
// defined below, which assumes that the object is uninitialized.
|
||||||
let obj = unsafe { ZendClassObject::<T>::new_uninit() };
|
let obj = unsafe { ZendClassObject::<T>::new_uninit(ce.as_ref()) };
|
||||||
obj.into_raw().get_mut_zend_obj()
|
obj.into_raw().get_mut_zend_obj()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -32,7 +32,9 @@ impl<T> Deref for Vec<T> {
|
|||||||
|
|
||||||
impl<T> Drop for Vec<T> {
|
impl<T> Drop for Vec<T> {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
unsafe { Box::from_raw(std::ptr::slice_from_raw_parts_mut(self.ptr, self.len)) };
|
unsafe {
|
||||||
|
let _ = Box::from_raw(std::ptr::slice_from_raw_parts_mut(self.ptr, self.len));
|
||||||
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,7 +52,7 @@ impl ToStub for Module {
|
|||||||
// Inserts a value into the entries hashmap. Takes a key and an entry, creating
|
// Inserts a value into the entries hashmap. Takes a key and an entry, creating
|
||||||
// the internal vector if it doesn't already exist.
|
// the internal vector if it doesn't already exist.
|
||||||
let mut insert = |ns, entry| {
|
let mut insert = |ns, entry| {
|
||||||
let bucket = entries.entry(ns).or_insert_with(StdVec::new);
|
let bucket = entries.entry(ns).or_default();
|
||||||
bucket.push(entry);
|
bucket.push(entry);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
15
src/embed/embed.c
Normal file
15
src/embed/embed.c
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
#include "embed.h"
|
||||||
|
|
||||||
|
// We actually use the PHP embed API to run PHP code in test
|
||||||
|
// At some point we might want to use our own SAPI to do that
|
||||||
|
void* ext_php_rs_embed_callback(int argc, char** argv, void* (*callback)(void *), void *ctx) {
|
||||||
|
void *result = NULL;
|
||||||
|
|
||||||
|
PHP_EMBED_START_BLOCK(argc, argv)
|
||||||
|
|
||||||
|
result = callback(ctx);
|
||||||
|
|
||||||
|
PHP_EMBED_END_BLOCK()
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
4
src/embed/embed.h
Normal file
4
src/embed/embed.h
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
#include "zend.h"
|
||||||
|
#include "sapi/embed/php_embed.h"
|
||||||
|
|
||||||
|
void* ext_php_rs_embed_callback(int argc, char** argv, void* (*callback)(void *), void *ctx);
|
16
src/embed/ffi.rs
Normal file
16
src/embed/ffi.rs
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
//! Raw FFI bindings to the Zend API.
|
||||||
|
|
||||||
|
#![allow(clippy::all)]
|
||||||
|
#![allow(warnings)]
|
||||||
|
|
||||||
|
use std::ffi::{c_char, c_int, c_void};
|
||||||
|
|
||||||
|
#[link(name = "wrapper")]
|
||||||
|
extern "C" {
|
||||||
|
pub fn ext_php_rs_embed_callback(
|
||||||
|
argc: c_int,
|
||||||
|
argv: *mut *mut c_char,
|
||||||
|
func: unsafe extern "C" fn(*const c_void) -> *const c_void,
|
||||||
|
ctx: *const c_void,
|
||||||
|
) -> *mut c_void;
|
||||||
|
}
|
277
src/embed/mod.rs
Normal file
277
src/embed/mod.rs
Normal file
@ -0,0 +1,277 @@
|
|||||||
|
//! Provides implementations for running php code from rust.
|
||||||
|
//! It only works on linux for now and you should have `php-embed` installed
|
||||||
|
//!
|
||||||
|
//! This crate was only test with PHP 8.2 please report any issue with other
|
||||||
|
//! version You should only use this crate for test purpose, it's not production
|
||||||
|
//! ready
|
||||||
|
|
||||||
|
mod ffi;
|
||||||
|
|
||||||
|
use crate::boxed::ZBox;
|
||||||
|
use crate::embed::ffi::ext_php_rs_embed_callback;
|
||||||
|
use crate::ffi::{
|
||||||
|
_zend_file_handle__bindgen_ty_1, php_execute_script, zend_eval_string, zend_file_handle,
|
||||||
|
zend_stream_init_filename, ZEND_RESULT_CODE_SUCCESS,
|
||||||
|
};
|
||||||
|
use crate::types::{ZendObject, Zval};
|
||||||
|
use crate::zend::{panic_wrapper, try_catch, ExecutorGlobals};
|
||||||
|
use parking_lot::{const_rwlock, RwLock};
|
||||||
|
use std::ffi::{c_char, c_void, CString, NulError};
|
||||||
|
use std::panic::{resume_unwind, RefUnwindSafe};
|
||||||
|
use std::path::Path;
|
||||||
|
use std::ptr::null_mut;
|
||||||
|
|
||||||
|
pub struct Embed;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum EmbedError {
|
||||||
|
InitError,
|
||||||
|
ExecuteError(Option<ZBox<ZendObject>>),
|
||||||
|
ExecuteScriptError,
|
||||||
|
InvalidEvalString(NulError),
|
||||||
|
InvalidPath,
|
||||||
|
CatchError,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl EmbedError {
|
||||||
|
pub fn is_bailout(&self) -> bool {
|
||||||
|
matches!(self, EmbedError::CatchError)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static RUN_FN_LOCK: RwLock<()> = const_rwlock(());
|
||||||
|
|
||||||
|
impl Embed {
|
||||||
|
/// Run a php script from a file
|
||||||
|
///
|
||||||
|
/// This function will only work correctly when used inside the `Embed::run`
|
||||||
|
/// function otherwise behavior is unexpected
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// * `Ok(())` - The script was executed successfully
|
||||||
|
/// * `Err(EmbedError)` - An error occured during the execution of the
|
||||||
|
/// script
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use ext_php_rs::embed::Embed;
|
||||||
|
///
|
||||||
|
/// Embed::run(|| {
|
||||||
|
/// let result = Embed::run_script("src/embed/test-script.php");
|
||||||
|
///
|
||||||
|
/// assert!(result.is_ok());
|
||||||
|
/// });
|
||||||
|
/// ```
|
||||||
|
pub fn run_script<P: AsRef<Path>>(path: P) -> Result<(), EmbedError> {
|
||||||
|
let path = match path.as_ref().to_str() {
|
||||||
|
Some(path) => match CString::new(path) {
|
||||||
|
Ok(path) => path,
|
||||||
|
Err(err) => return Err(EmbedError::InvalidEvalString(err)),
|
||||||
|
},
|
||||||
|
None => return Err(EmbedError::InvalidPath),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut file_handle = zend_file_handle {
|
||||||
|
handle: _zend_file_handle__bindgen_ty_1 { fp: null_mut() },
|
||||||
|
filename: null_mut(),
|
||||||
|
opened_path: null_mut(),
|
||||||
|
type_: 0,
|
||||||
|
primary_script: false,
|
||||||
|
in_list: false,
|
||||||
|
buf: null_mut(),
|
||||||
|
len: 0,
|
||||||
|
};
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
zend_stream_init_filename(&mut file_handle, path.as_ptr());
|
||||||
|
}
|
||||||
|
|
||||||
|
let exec_result = try_catch(|| unsafe { php_execute_script(&mut file_handle) });
|
||||||
|
|
||||||
|
match exec_result {
|
||||||
|
Err(_) => Err(EmbedError::CatchError),
|
||||||
|
Ok(true) => Ok(()),
|
||||||
|
Ok(false) => Err(EmbedError::ExecuteScriptError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Start and run embed sapi engine
|
||||||
|
///
|
||||||
|
/// This function will allow to run php code from rust, the same PHP context
|
||||||
|
/// is keep between calls inside the function passed to this method.
|
||||||
|
/// Which means subsequent calls to `Embed::eval` or `Embed::run_script`
|
||||||
|
/// will be able to access variables defined in previous calls
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// * R - The result of the function passed to this method
|
||||||
|
///
|
||||||
|
/// R must implement [`Default`] so it can be returned in case of a bailout
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use ext_php_rs::embed::Embed;
|
||||||
|
///
|
||||||
|
/// Embed::run(|| {
|
||||||
|
/// let _ = Embed::eval("$foo = 'foo';");
|
||||||
|
/// let foo = Embed::eval("$foo;");
|
||||||
|
/// assert!(foo.is_ok());
|
||||||
|
/// assert_eq!(foo.unwrap().string().unwrap(), "foo");
|
||||||
|
/// });
|
||||||
|
/// ```
|
||||||
|
pub fn run<R, F: FnMut() -> R + RefUnwindSafe>(func: F) -> R
|
||||||
|
where
|
||||||
|
R: Default,
|
||||||
|
{
|
||||||
|
// @TODO handle php thread safe
|
||||||
|
//
|
||||||
|
// This is to prevent multiple threads from running php at the same time
|
||||||
|
// At some point we should detect if php is compiled with thread safety and
|
||||||
|
// avoid doing that in this case
|
||||||
|
let _guard = RUN_FN_LOCK.write();
|
||||||
|
|
||||||
|
let panic = unsafe {
|
||||||
|
ext_php_rs_embed_callback(
|
||||||
|
0,
|
||||||
|
null_mut(),
|
||||||
|
panic_wrapper::<R, F>,
|
||||||
|
&func as *const F as *const c_void,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
// This can happen if there is a bailout
|
||||||
|
if panic.is_null() {
|
||||||
|
return R::default();
|
||||||
|
}
|
||||||
|
|
||||||
|
match unsafe { *Box::from_raw(panic as *mut std::thread::Result<R>) } {
|
||||||
|
Ok(r) => r,
|
||||||
|
Err(err) => {
|
||||||
|
// we resume the panic here so it can be catched correctly by the test framework
|
||||||
|
resume_unwind(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Evaluate a php code
|
||||||
|
///
|
||||||
|
/// This function will only work correctly when used inside the `Embed::run`
|
||||||
|
/// function
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// * `Ok(Zval)` - The result of the evaluation
|
||||||
|
/// * `Err(EmbedError)` - An error occured during the evaluation
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// ```
|
||||||
|
/// use ext_php_rs::embed::Embed;
|
||||||
|
///
|
||||||
|
/// Embed::run(|| {
|
||||||
|
/// let foo = Embed::eval("$foo = 'foo';");
|
||||||
|
/// assert!(foo.is_ok());
|
||||||
|
/// });
|
||||||
|
/// ```
|
||||||
|
pub fn eval(code: &str) -> Result<Zval, EmbedError> {
|
||||||
|
let cstr = match CString::new(code) {
|
||||||
|
Ok(cstr) => cstr,
|
||||||
|
Err(err) => return Err(EmbedError::InvalidEvalString(err)),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut result = Zval::new();
|
||||||
|
|
||||||
|
let exec_result = try_catch(|| unsafe {
|
||||||
|
zend_eval_string(
|
||||||
|
cstr.as_ptr() as *const c_char,
|
||||||
|
&mut result,
|
||||||
|
b"run\0".as_ptr() as *const _,
|
||||||
|
)
|
||||||
|
});
|
||||||
|
|
||||||
|
match exec_result {
|
||||||
|
Err(_) => Err(EmbedError::CatchError),
|
||||||
|
Ok(ZEND_RESULT_CODE_SUCCESS) => Ok(result),
|
||||||
|
Ok(_) => Err(EmbedError::ExecuteError(ExecutorGlobals::take_exception())),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::Embed;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_run() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let result = Embed::eval("$foo = 'foo';");
|
||||||
|
|
||||||
|
assert!(result.is_ok());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_run_error() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let result = Embed::eval("stupid code;");
|
||||||
|
|
||||||
|
assert!(!result.is_ok());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_run_script() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let result = Embed::run_script("src/embed/test-script.php");
|
||||||
|
|
||||||
|
assert!(result.is_ok());
|
||||||
|
|
||||||
|
let zval = Embed::eval("$foo;").unwrap();
|
||||||
|
|
||||||
|
assert!(zval.is_object());
|
||||||
|
|
||||||
|
let obj = zval.object().unwrap();
|
||||||
|
|
||||||
|
assert_eq!(obj.get_class_name().unwrap(), "Test");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_run_script_error() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let result = Embed::run_script("src/embed/test-script-exception.php");
|
||||||
|
|
||||||
|
assert!(!result.is_ok());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn test_panic() {
|
||||||
|
Embed::run(|| {
|
||||||
|
panic!("test panic");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_return() {
|
||||||
|
let foo = Embed::run(|| {
|
||||||
|
return "foo";
|
||||||
|
});
|
||||||
|
|
||||||
|
assert_eq!(foo, "foo");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_eval_bailout() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let result = Embed::eval("str_repeat('a', 100_000_000_000_000);");
|
||||||
|
|
||||||
|
assert!(result.is_err());
|
||||||
|
assert!(result.unwrap_err().is_bailout());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
3
src/embed/test-script-exception.php
Normal file
3
src/embed/test-script-exception.php
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
throw new \RuntimeException('This is a test exception');
|
7
src/embed/test-script.php
Normal file
7
src/embed/test-script.php
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
class Test {
|
||||||
|
public function __construct() {}
|
||||||
|
}
|
||||||
|
|
||||||
|
$foo = new Test();
|
@ -57,6 +57,8 @@ pub enum Error {
|
|||||||
InvalidUtf8,
|
InvalidUtf8,
|
||||||
/// Could not call the given function.
|
/// Could not call the given function.
|
||||||
Callable,
|
Callable,
|
||||||
|
/// An object was expected.
|
||||||
|
Object,
|
||||||
/// An invalid exception type was thrown.
|
/// An invalid exception type was thrown.
|
||||||
InvalidException(ClassFlags),
|
InvalidException(ClassFlags),
|
||||||
/// Converting integer arguments resulted in an overflow.
|
/// Converting integer arguments resulted in an overflow.
|
||||||
@ -89,6 +91,7 @@ impl Display for Error {
|
|||||||
),
|
),
|
||||||
Error::InvalidUtf8 => write!(f, "Invalid Utf8 byte sequence."),
|
Error::InvalidUtf8 => write!(f, "Invalid Utf8 byte sequence."),
|
||||||
Error::Callable => write!(f, "Could not call given function."),
|
Error::Callable => write!(f, "Could not call given function."),
|
||||||
|
Error::Object => write!(f, "An object was expected."),
|
||||||
Error::InvalidException(flags) => {
|
Error::InvalidException(flags) => {
|
||||||
write!(f, "Invalid exception type was thrown: {flags:?}")
|
write!(f, "Invalid exception type was thrown: {flags:?}")
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,14 @@
|
|||||||
//! Types and functions used for throwing exceptions from Rust to PHP.
|
//! Types and functions used for throwing exceptions from Rust to PHP.
|
||||||
|
|
||||||
use std::ffi::CString;
|
use std::{ffi::CString, fmt::Debug};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
class::RegisteredClass,
|
class::RegisteredClass,
|
||||||
error::{Error, Result},
|
error::{Error, Result},
|
||||||
ffi::zend_throw_exception_ex,
|
ffi::zend_throw_exception_ex,
|
||||||
|
ffi::zend_throw_exception_object,
|
||||||
flags::ClassFlags,
|
flags::ClassFlags,
|
||||||
|
types::Zval,
|
||||||
zend::{ce, ClassEntry},
|
zend::{ce, ClassEntry},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -25,6 +27,7 @@ pub struct PhpException {
|
|||||||
message: String,
|
message: String,
|
||||||
code: i32,
|
code: i32,
|
||||||
ex: &'static ClassEntry,
|
ex: &'static ClassEntry,
|
||||||
|
object: Option<Zval>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PhpException {
|
impl PhpException {
|
||||||
@ -36,7 +39,12 @@ impl PhpException {
|
|||||||
/// * `code` - Integer code to go inside the exception.
|
/// * `code` - Integer code to go inside the exception.
|
||||||
/// * `ex` - Exception type to throw.
|
/// * `ex` - Exception type to throw.
|
||||||
pub fn new(message: String, code: i32, ex: &'static ClassEntry) -> Self {
|
pub fn new(message: String, code: i32, ex: &'static ClassEntry) -> Self {
|
||||||
Self { message, code, ex }
|
Self {
|
||||||
|
message,
|
||||||
|
code,
|
||||||
|
ex,
|
||||||
|
object: None,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a new default exception instance, using the default PHP
|
/// Creates a new default exception instance, using the default PHP
|
||||||
@ -59,10 +67,25 @@ impl PhpException {
|
|||||||
Self::new(message, 0, T::get_metadata().ce())
|
Self::new(message, 0, T::get_metadata().ce())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set the Zval object for the exception.
|
||||||
|
///
|
||||||
|
/// Exceptions can be based of instantiated Zval objects when you are throwing a custom exception with
|
||||||
|
/// stateful properties.
|
||||||
|
///
|
||||||
|
/// # Parameters
|
||||||
|
///
|
||||||
|
/// * `object` - The Zval object.
|
||||||
|
pub fn set_object(&mut self, object: Option<Zval>) {
|
||||||
|
self.object = object;
|
||||||
|
}
|
||||||
|
|
||||||
/// Throws the exception, returning nothing inside a result if successful
|
/// Throws the exception, returning nothing inside a result if successful
|
||||||
/// and an error otherwise.
|
/// and an error otherwise.
|
||||||
pub fn throw(self) -> Result<()> {
|
pub fn throw(self) -> Result<()> {
|
||||||
throw_with_code(self.ex, self.code, &self.message)
|
match self.object {
|
||||||
|
Some(object) => throw_object(object),
|
||||||
|
None => throw_with_code(self.ex, self.code, &self.message),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -146,3 +169,44 @@ pub fn throw_with_code(ex: &ClassEntry, code: i32, message: &str) -> Result<()>
|
|||||||
};
|
};
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Throws an exception object.
|
||||||
|
///
|
||||||
|
/// Returns a result containing nothing if the exception was successfully
|
||||||
|
/// thrown.
|
||||||
|
///
|
||||||
|
/// # Parameters
|
||||||
|
///
|
||||||
|
/// * `object` - The zval of type object
|
||||||
|
///
|
||||||
|
/// # Examples
|
||||||
|
///
|
||||||
|
/// ```no_run
|
||||||
|
/// use ext_php_rs::prelude::*;
|
||||||
|
/// use ext_php_rs::exception::throw_object;
|
||||||
|
/// use crate::ext_php_rs::convert::IntoZval;
|
||||||
|
///
|
||||||
|
/// #[php_class]
|
||||||
|
/// #[extends(ext_php_rs::zend::ce::exception())]
|
||||||
|
/// pub struct JsException {
|
||||||
|
/// #[prop(flags = ext_php_rs::flags::PropertyFlags::Public)]
|
||||||
|
/// message: String,
|
||||||
|
/// #[prop(flags = ext_php_rs::flags::PropertyFlags::Public)]
|
||||||
|
/// code: i32,
|
||||||
|
/// #[prop(flags = ext_php_rs::flags::PropertyFlags::Public)]
|
||||||
|
/// file: String,
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// #[php_module]
|
||||||
|
/// pub fn get_module(module: ModuleBuilder) -> ModuleBuilder {
|
||||||
|
/// module
|
||||||
|
/// }
|
||||||
|
///
|
||||||
|
/// let error = JsException { message: "A JS error occurred.".to_string(), code: 100, file: "index.js".to_string() };
|
||||||
|
/// throw_object( error.into_zval(true).unwrap() );
|
||||||
|
/// ```
|
||||||
|
pub fn throw_object(zval: Zval) -> Result<()> {
|
||||||
|
let mut zv = core::mem::ManuallyDrop::new(zval);
|
||||||
|
unsafe { zend_throw_exception_object(core::ptr::addr_of_mut!(zv).cast()) };
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
@ -29,6 +29,13 @@ extern "C" {
|
|||||||
pub fn ext_php_rs_process_globals() -> *mut php_core_globals;
|
pub fn ext_php_rs_process_globals() -> *mut php_core_globals;
|
||||||
pub fn ext_php_rs_sapi_globals() -> *mut sapi_globals_struct;
|
pub fn ext_php_rs_sapi_globals() -> *mut sapi_globals_struct;
|
||||||
pub fn ext_php_rs_file_globals() -> *mut php_file_globals;
|
pub fn ext_php_rs_file_globals() -> *mut php_file_globals;
|
||||||
|
pub fn ext_php_rs_sapi_module() -> *mut sapi_module_struct;
|
||||||
|
pub fn ext_php_rs_zend_try_catch(
|
||||||
|
func: unsafe extern "C" fn(*const c_void) -> *const c_void,
|
||||||
|
ctx: *const c_void,
|
||||||
|
result: *mut *mut c_void,
|
||||||
|
) -> bool;
|
||||||
|
pub fn ext_php_rs_zend_bailout() -> !;
|
||||||
}
|
}
|
||||||
|
|
||||||
include!(concat!(env!("OUT_DIR"), "/bindings.rs"));
|
include!(concat!(env!("OUT_DIR"), "/bindings.rs"));
|
||||||
|
30
src/flags.rs
30
src/flags.rs
@ -8,9 +8,10 @@ use crate::ffi::{
|
|||||||
CONST_CS, CONST_DEPRECATED, CONST_NO_FILE_CACHE, CONST_PERSISTENT, E_COMPILE_ERROR,
|
CONST_CS, CONST_DEPRECATED, CONST_NO_FILE_CACHE, CONST_PERSISTENT, E_COMPILE_ERROR,
|
||||||
E_COMPILE_WARNING, E_CORE_ERROR, E_CORE_WARNING, E_DEPRECATED, E_ERROR, E_NOTICE, E_PARSE,
|
E_COMPILE_WARNING, E_CORE_ERROR, E_CORE_WARNING, E_DEPRECATED, E_ERROR, E_NOTICE, E_PARSE,
|
||||||
E_RECOVERABLE_ERROR, E_STRICT, E_USER_DEPRECATED, E_USER_ERROR, E_USER_NOTICE, E_USER_WARNING,
|
E_RECOVERABLE_ERROR, E_STRICT, E_USER_DEPRECATED, E_USER_ERROR, E_USER_NOTICE, E_USER_WARNING,
|
||||||
E_WARNING, IS_ARRAY, IS_CALLABLE, IS_CONSTANT_AST, IS_DOUBLE, IS_FALSE, IS_LONG, IS_MIXED,
|
E_WARNING, IS_ARRAY, IS_CALLABLE, IS_CONSTANT_AST, IS_DOUBLE, IS_FALSE, IS_INDIRECT, IS_LONG,
|
||||||
IS_NULL, IS_OBJECT, IS_PTR, IS_REFERENCE, IS_RESOURCE, IS_STRING, IS_TRUE, IS_TYPE_COLLECTABLE,
|
IS_MIXED, IS_NULL, IS_OBJECT, IS_PTR, IS_REFERENCE, IS_RESOURCE, IS_STRING, IS_TRUE,
|
||||||
IS_TYPE_REFCOUNTED, IS_UNDEF, IS_VOID, ZEND_ACC_ABSTRACT, ZEND_ACC_ANON_CLASS,
|
IS_TYPE_COLLECTABLE, IS_TYPE_REFCOUNTED, IS_UNDEF, IS_VOID, PHP_INI_ALL, PHP_INI_PERDIR,
|
||||||
|
PHP_INI_SYSTEM, PHP_INI_USER, ZEND_ACC_ABSTRACT, ZEND_ACC_ANON_CLASS,
|
||||||
ZEND_ACC_CALL_VIA_TRAMPOLINE, ZEND_ACC_CHANGED, ZEND_ACC_CLOSURE, ZEND_ACC_CONSTANTS_UPDATED,
|
ZEND_ACC_CALL_VIA_TRAMPOLINE, ZEND_ACC_CHANGED, ZEND_ACC_CLOSURE, ZEND_ACC_CONSTANTS_UPDATED,
|
||||||
ZEND_ACC_CTOR, ZEND_ACC_DEPRECATED, ZEND_ACC_DONE_PASS_TWO, ZEND_ACC_EARLY_BINDING,
|
ZEND_ACC_CTOR, ZEND_ACC_DEPRECATED, ZEND_ACC_DONE_PASS_TWO, ZEND_ACC_EARLY_BINDING,
|
||||||
ZEND_ACC_FAKE_CLOSURE, ZEND_ACC_FINAL, ZEND_ACC_GENERATOR, ZEND_ACC_HAS_FINALLY_BLOCK,
|
ZEND_ACC_FAKE_CLOSURE, ZEND_ACC_FINAL, ZEND_ACC_GENERATOR, ZEND_ACC_HAS_FINALLY_BLOCK,
|
||||||
@ -174,6 +175,16 @@ bitflags! {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bitflags! {
|
||||||
|
/// Represents permissions for where a configuration setting may be set.
|
||||||
|
pub struct IniEntryPermission: u32 {
|
||||||
|
const User = PHP_INI_USER;
|
||||||
|
const PerDir = PHP_INI_PERDIR;
|
||||||
|
const System = PHP_INI_SYSTEM;
|
||||||
|
const All = PHP_INI_ALL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bitflags! {
|
bitflags! {
|
||||||
/// Represents error types when used via php_error_docref for example.
|
/// Represents error types when used via php_error_docref for example.
|
||||||
pub struct ErrorType: u32 {
|
pub struct ErrorType: u32 {
|
||||||
@ -194,6 +205,7 @@ bitflags! {
|
|||||||
const UserDeprecated = E_USER_DEPRECATED;
|
const UserDeprecated = E_USER_DEPRECATED;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(PartialEq, Eq, Hash, Debug, Clone, Copy)]
|
#[derive(PartialEq, Eq, Hash, Debug, Clone, Copy)]
|
||||||
pub enum FunctionType {
|
pub enum FunctionType {
|
||||||
Internal,
|
Internal,
|
||||||
@ -234,6 +246,7 @@ pub enum DataType {
|
|||||||
Mixed,
|
Mixed,
|
||||||
Bool,
|
Bool,
|
||||||
Ptr,
|
Ptr,
|
||||||
|
Indirect,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for DataType {
|
impl Default for DataType {
|
||||||
@ -257,6 +270,7 @@ impl DataType {
|
|||||||
DataType::Object(_) => IS_OBJECT,
|
DataType::Object(_) => IS_OBJECT,
|
||||||
DataType::Resource => IS_RESOURCE,
|
DataType::Resource => IS_RESOURCE,
|
||||||
DataType::Reference => IS_RESOURCE,
|
DataType::Reference => IS_RESOURCE,
|
||||||
|
DataType::Indirect => IS_INDIRECT,
|
||||||
DataType::Callable => IS_CALLABLE,
|
DataType::Callable => IS_CALLABLE,
|
||||||
DataType::ConstantExpression => IS_CONSTANT_AST,
|
DataType::ConstantExpression => IS_CONSTANT_AST,
|
||||||
DataType::Void => IS_VOID,
|
DataType::Void => IS_VOID,
|
||||||
@ -325,6 +339,7 @@ impl From<u32> for DataType {
|
|||||||
|
|
||||||
contains!(IS_VOID, Void);
|
contains!(IS_VOID, Void);
|
||||||
contains!(IS_PTR, Ptr);
|
contains!(IS_PTR, Ptr);
|
||||||
|
contains!(IS_INDIRECT, Indirect);
|
||||||
contains!(IS_CALLABLE, Callable);
|
contains!(IS_CALLABLE, Callable);
|
||||||
contains!(IS_CONSTANT_AST, ConstantExpression);
|
contains!(IS_CONSTANT_AST, ConstantExpression);
|
||||||
contains!(IS_REFERENCE, Reference);
|
contains!(IS_REFERENCE, Reference);
|
||||||
@ -367,6 +382,7 @@ impl Display for DataType {
|
|||||||
DataType::Bool => write!(f, "Bool"),
|
DataType::Bool => write!(f, "Bool"),
|
||||||
DataType::Mixed => write!(f, "Mixed"),
|
DataType::Mixed => write!(f, "Mixed"),
|
||||||
DataType::Ptr => write!(f, "Pointer"),
|
DataType::Ptr => write!(f, "Pointer"),
|
||||||
|
DataType::Indirect => write!(f, "Indirect"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -376,9 +392,9 @@ mod tests {
|
|||||||
use super::DataType;
|
use super::DataType;
|
||||||
use crate::ffi::{
|
use crate::ffi::{
|
||||||
IS_ARRAY, IS_ARRAY_EX, IS_CALLABLE, IS_CONSTANT_AST, IS_CONSTANT_AST_EX, IS_DOUBLE,
|
IS_ARRAY, IS_ARRAY_EX, IS_CALLABLE, IS_CONSTANT_AST, IS_CONSTANT_AST_EX, IS_DOUBLE,
|
||||||
IS_FALSE, IS_INTERNED_STRING_EX, IS_LONG, IS_NULL, IS_OBJECT, IS_OBJECT_EX, IS_PTR,
|
IS_FALSE, IS_INDIRECT, IS_INTERNED_STRING_EX, IS_LONG, IS_NULL, IS_OBJECT, IS_OBJECT_EX,
|
||||||
IS_REFERENCE, IS_REFERENCE_EX, IS_RESOURCE, IS_RESOURCE_EX, IS_STRING, IS_STRING_EX,
|
IS_PTR, IS_REFERENCE, IS_REFERENCE_EX, IS_RESOURCE, IS_RESOURCE_EX, IS_STRING,
|
||||||
IS_TRUE, IS_UNDEF, IS_VOID,
|
IS_STRING_EX, IS_TRUE, IS_UNDEF, IS_VOID,
|
||||||
};
|
};
|
||||||
use std::convert::TryFrom;
|
use std::convert::TryFrom;
|
||||||
|
|
||||||
@ -402,7 +418,7 @@ mod tests {
|
|||||||
test!(IS_RESOURCE, Resource);
|
test!(IS_RESOURCE, Resource);
|
||||||
test!(IS_REFERENCE, Reference);
|
test!(IS_REFERENCE, Reference);
|
||||||
test!(IS_CONSTANT_AST, ConstantExpression);
|
test!(IS_CONSTANT_AST, ConstantExpression);
|
||||||
test!(IS_CALLABLE, Callable);
|
test!(IS_INDIRECT, Indirect);
|
||||||
test!(IS_VOID, Void);
|
test!(IS_VOID, Void);
|
||||||
test!(IS_PTR, Ptr);
|
test!(IS_PTR, Ptr);
|
||||||
|
|
||||||
|
@ -25,6 +25,8 @@ pub mod class;
|
|||||||
pub mod closure;
|
pub mod closure;
|
||||||
pub mod constant;
|
pub mod constant;
|
||||||
pub mod describe;
|
pub mod describe;
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
pub mod embed;
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
pub mod internal;
|
pub mod internal;
|
||||||
pub mod props;
|
pub mod props;
|
||||||
@ -35,6 +37,7 @@ pub mod zend;
|
|||||||
/// A module typically glob-imported containing the typically required macros
|
/// A module typically glob-imported containing the typically required macros
|
||||||
/// and imports.
|
/// and imports.
|
||||||
pub mod prelude {
|
pub mod prelude {
|
||||||
|
|
||||||
pub use crate::builders::ModuleBuilder;
|
pub use crate::builders::ModuleBuilder;
|
||||||
#[cfg(any(docs, feature = "closure"))]
|
#[cfg(any(docs, feature = "closure"))]
|
||||||
#[cfg_attr(docs, doc(cfg(feature = "closure")))]
|
#[cfg_attr(docs, doc(cfg(feature = "closure")))]
|
||||||
|
@ -187,6 +187,33 @@ impl ZendHashTable {
|
|||||||
unsafe { zend_hash_str_find(self, str.as_ptr(), key.len() as _).as_ref() }
|
unsafe { zend_hash_str_find(self, str.as_ptr(), key.len() as _).as_ref() }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Attempts to retrieve a value from the hash table with a string key.
|
||||||
|
///
|
||||||
|
/// # Parameters
|
||||||
|
///
|
||||||
|
/// * `key` - The key to search for in the hash table.
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// * `Some(&Zval)` - A reference to the zval at the position in the hash
|
||||||
|
/// table.
|
||||||
|
/// * `None` - No value at the given position was found.
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// ```no_run
|
||||||
|
/// use ext_php_rs::types::ZendHashTable;
|
||||||
|
///
|
||||||
|
/// let mut ht = ZendHashTable::new();
|
||||||
|
///
|
||||||
|
/// ht.insert("test", "hello world");
|
||||||
|
/// assert_eq!(ht.get("test").and_then(|zv| zv.str()), Some("hello world"));
|
||||||
|
/// ```
|
||||||
|
pub fn get_mut(&self, key: &'_ str) -> Option<&mut Zval> {
|
||||||
|
let str = CString::new(key).ok()?;
|
||||||
|
unsafe { zend_hash_str_find(self, str.as_ptr(), key.len() as _).as_mut() }
|
||||||
|
}
|
||||||
|
|
||||||
/// Attempts to retrieve a value from the hash table with an index.
|
/// Attempts to retrieve a value from the hash table with an index.
|
||||||
///
|
///
|
||||||
/// # Parameters
|
/// # Parameters
|
||||||
@ -213,6 +240,32 @@ impl ZendHashTable {
|
|||||||
unsafe { zend_hash_index_find(self, key).as_ref() }
|
unsafe { zend_hash_index_find(self, key).as_ref() }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Attempts to retrieve a value from the hash table with an index.
|
||||||
|
///
|
||||||
|
/// # Parameters
|
||||||
|
///
|
||||||
|
/// * `key` - The key to search for in the hash table.
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// * `Some(&Zval)` - A reference to the zval at the position in the hash
|
||||||
|
/// table.
|
||||||
|
/// * `None` - No value at the given position was found.
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// ```no_run
|
||||||
|
/// use ext_php_rs::types::ZendHashTable;
|
||||||
|
///
|
||||||
|
/// let mut ht = ZendHashTable::new();
|
||||||
|
///
|
||||||
|
/// ht.push(100);
|
||||||
|
/// assert_eq!(ht.get_index(0).and_then(|zv| zv.long()), Some(100));
|
||||||
|
/// ```
|
||||||
|
pub fn get_index_mut(&self, key: u64) -> Option<&mut Zval> {
|
||||||
|
unsafe { zend_hash_index_find(self, key).as_mut() }
|
||||||
|
}
|
||||||
|
|
||||||
/// Attempts to remove a value from the hash table with a string key.
|
/// Attempts to remove a value from the hash table with a string key.
|
||||||
///
|
///
|
||||||
/// # Parameters
|
/// # Parameters
|
||||||
@ -715,7 +768,7 @@ impl<'a> FromZval<'a> for &'a ZendHashTable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////
|
///////////////////////////////////////////
|
||||||
//// HashMap
|
/// HashMap
|
||||||
///////////////////////////////////////////
|
///////////////////////////////////////////
|
||||||
|
|
||||||
impl<'a, V> TryFrom<&'a ZendHashTable> for HashMap<String, V>
|
impl<'a, V> TryFrom<&'a ZendHashTable> for HashMap<String, V>
|
||||||
@ -784,7 +837,7 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////
|
///////////////////////////////////////////
|
||||||
//// Vec
|
/// Vec
|
||||||
///////////////////////////////////////////
|
///////////////////////////////////////////
|
||||||
|
|
||||||
impl<'a, T> TryFrom<&'a ZendHashTable> for Vec<T>
|
impl<'a, T> TryFrom<&'a ZendHashTable> for Vec<T>
|
||||||
|
@ -99,6 +99,7 @@ impl<'a> ZendCallable<'a> {
|
|||||||
/// let result = strpos.try_call(vec![&"hello", &"e"]).unwrap();
|
/// let result = strpos.try_call(vec![&"hello", &"e"]).unwrap();
|
||||||
/// assert_eq!(result.long(), Some(1));
|
/// assert_eq!(result.long(), Some(1));
|
||||||
/// ```
|
/// ```
|
||||||
|
#[inline(always)]
|
||||||
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
||||||
if !self.0.is_callable() {
|
if !self.0.is_callable() {
|
||||||
return Err(Error::Callable);
|
return Err(Error::Callable);
|
||||||
|
@ -5,6 +5,7 @@ use std::{
|
|||||||
fmt::Debug,
|
fmt::Debug,
|
||||||
mem,
|
mem,
|
||||||
ops::{Deref, DerefMut},
|
ops::{Deref, DerefMut},
|
||||||
|
os::raw::c_char,
|
||||||
ptr::{self, NonNull},
|
ptr::{self, NonNull},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -19,6 +20,7 @@ use crate::{
|
|||||||
},
|
},
|
||||||
flags::DataType,
|
flags::DataType,
|
||||||
types::{ZendObject, Zval},
|
types::{ZendObject, Zval},
|
||||||
|
zend::ClassEntry,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Representation of a Zend class object in memory.
|
/// Representation of a Zend class object in memory.
|
||||||
@ -43,7 +45,7 @@ impl<T: RegisteredClass> ZendClassObject<T> {
|
|||||||
/// Panics if memory was unable to be allocated for the new object.
|
/// Panics if memory was unable to be allocated for the new object.
|
||||||
pub fn new(val: T) -> ZBox<Self> {
|
pub fn new(val: T) -> ZBox<Self> {
|
||||||
// SAFETY: We are providing a value to initialize the object with.
|
// SAFETY: We are providing a value to initialize the object with.
|
||||||
unsafe { Self::internal_new(Some(val)) }
|
unsafe { Self::internal_new(Some(val), None) }
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a new [`ZendClassObject`] of type `T`, with an uninitialized
|
/// Creates a new [`ZendClassObject`] of type `T`, with an uninitialized
|
||||||
@ -67,8 +69,8 @@ impl<T: RegisteredClass> ZendClassObject<T> {
|
|||||||
/// # Panics
|
/// # Panics
|
||||||
///
|
///
|
||||||
/// Panics if memory was unable to be allocated for the new object.
|
/// Panics if memory was unable to be allocated for the new object.
|
||||||
pub unsafe fn new_uninit() -> ZBox<Self> {
|
pub unsafe fn new_uninit(ce: Option<&'static ClassEntry>) -> ZBox<Self> {
|
||||||
Self::internal_new(None)
|
Self::internal_new(None, ce)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates a new [`ZendObject`] of type `T`, storing the given (and
|
/// Creates a new [`ZendObject`] of type `T`, storing the given (and
|
||||||
@ -102,10 +104,10 @@ impl<T: RegisteredClass> ZendClassObject<T> {
|
|||||||
/// # Panics
|
/// # Panics
|
||||||
///
|
///
|
||||||
/// Panics if memory was unable to be allocated for the new object.
|
/// Panics if memory was unable to be allocated for the new object.
|
||||||
unsafe fn internal_new(val: Option<T>) -> ZBox<Self> {
|
unsafe fn internal_new(val: Option<T>, ce: Option<&'static ClassEntry>) -> ZBox<Self> {
|
||||||
let size = mem::size_of::<ZendClassObject<T>>();
|
let size = mem::size_of::<ZendClassObject<T>>();
|
||||||
let meta = T::get_metadata();
|
let meta = T::get_metadata();
|
||||||
let ce = meta.ce() as *const _ as *mut _;
|
let ce = ce.unwrap_or_else(|| meta.ce()) as *const _ as *mut _;
|
||||||
let obj = ext_php_rs_zend_object_alloc(size as _, ce) as *mut ZendClassObject<T>;
|
let obj = ext_php_rs_zend_object_alloc(size as _, ce) as *mut ZendClassObject<T>;
|
||||||
let obj = obj
|
let obj = obj
|
||||||
.as_mut()
|
.as_mut()
|
||||||
@ -155,18 +157,19 @@ impl<T: RegisteredClass> ZendClassObject<T> {
|
|||||||
/// # Parameters
|
/// # Parameters
|
||||||
///
|
///
|
||||||
/// * `obj` - The zend object to get the [`ZendClassObject`] for.
|
/// * `obj` - The zend object to get the [`ZendClassObject`] for.
|
||||||
|
#[allow(clippy::needless_pass_by_ref_mut)]
|
||||||
pub fn from_zend_obj_mut(std: &mut zend_object) -> Option<&mut Self> {
|
pub fn from_zend_obj_mut(std: &mut zend_object) -> Option<&mut Self> {
|
||||||
Self::_from_zend_obj(std)
|
Self::_from_zend_obj(std)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn _from_zend_obj(std: &zend_object) -> Option<&mut Self> {
|
fn _from_zend_obj(std: &zend_object) -> Option<&mut Self> {
|
||||||
let std = std as *const zend_object as *const i8;
|
let std = std as *const zend_object as *const c_char;
|
||||||
let ptr = unsafe {
|
let ptr = unsafe {
|
||||||
let ptr = std.offset(0 - Self::std_offset() as isize) as *const Self;
|
let ptr = std.offset(0 - Self::std_offset() as isize) as *const Self;
|
||||||
(ptr as *mut Self).as_mut()?
|
(ptr as *mut Self).as_mut()?
|
||||||
};
|
};
|
||||||
|
|
||||||
if ptr.std.is_instance::<T>() {
|
if ptr.std.instance_of(T::get_metadata().ce()) {
|
||||||
Some(ptr)
|
Some(ptr)
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
|
@ -1,16 +1,17 @@
|
|||||||
//! Represents an object in PHP. Allows for overriding the internal object used
|
//! Represents an object in PHP. Allows for overriding the internal object used
|
||||||
//! by classes, allowing users to store Rust data inside a PHP object.
|
//! by classes, allowing users to store Rust data inside a PHP object.
|
||||||
|
|
||||||
use std::{convert::TryInto, fmt::Debug, ops::DerefMut};
|
use std::{convert::TryInto, fmt::Debug, ops::DerefMut, os::raw::c_char};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
boxed::{ZBox, ZBoxable},
|
boxed::{ZBox, ZBoxable},
|
||||||
class::RegisteredClass,
|
class::RegisteredClass,
|
||||||
convert::{FromZendObject, FromZval, FromZvalMut, IntoZval},
|
convert::{FromZendObject, FromZval, FromZvalMut, IntoZval, IntoZvalDyn},
|
||||||
error::{Error, Result},
|
error::{Error, Result},
|
||||||
ffi::{
|
ffi::{
|
||||||
ext_php_rs_zend_object_release, zend_call_known_function, zend_object, zend_objects_new,
|
ext_php_rs_zend_object_release, object_properties_init, zend_call_known_function,
|
||||||
HashTable, ZEND_ISEMPTY, ZEND_PROPERTY_EXISTS, ZEND_PROPERTY_ISSET,
|
zend_function, zend_hash_str_find_ptr_lc, zend_object, zend_objects_new, HashTable,
|
||||||
|
ZEND_ISEMPTY, ZEND_PROPERTY_EXISTS, ZEND_PROPERTY_ISSET,
|
||||||
},
|
},
|
||||||
flags::DataType,
|
flags::DataType,
|
||||||
rc::PhpRc,
|
rc::PhpRc,
|
||||||
@ -41,7 +42,18 @@ impl ZendObject {
|
|||||||
// SAFETY: Using emalloc to allocate memory inside Zend arena. Casting `ce` to
|
// SAFETY: Using emalloc to allocate memory inside Zend arena. Casting `ce` to
|
||||||
// `*mut` is valid as the function will not mutate `ce`.
|
// `*mut` is valid as the function will not mutate `ce`.
|
||||||
unsafe {
|
unsafe {
|
||||||
let ptr = zend_objects_new(ce as *const _ as *mut _);
|
let ptr = match ce.__bindgen_anon_2.create_object {
|
||||||
|
None => {
|
||||||
|
let ptr = zend_objects_new(ce as *const _ as *mut _);
|
||||||
|
if ptr.is_null() {
|
||||||
|
panic!("Failed to allocate memory for Zend object")
|
||||||
|
}
|
||||||
|
object_properties_init(ptr, ce as *const _ as *mut _);
|
||||||
|
ptr
|
||||||
|
}
|
||||||
|
Some(v) => v(ce as *const _ as *mut _),
|
||||||
|
};
|
||||||
|
|
||||||
ZBox::from_raw(
|
ZBox::from_raw(
|
||||||
ptr.as_mut()
|
ptr.as_mut()
|
||||||
.expect("Failed to allocate memory for Zend object"),
|
.expect("Failed to allocate memory for Zend object"),
|
||||||
@ -121,6 +133,38 @@ impl ZendObject {
|
|||||||
(self.ce as *const ClassEntry).eq(&(T::get_metadata().ce() as *const _))
|
(self.ce as *const ClassEntry).eq(&(T::get_metadata().ce() as *const _))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn try_call_method(&self, name: &str, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
||||||
|
let mut retval = Zval::new();
|
||||||
|
let len = params.len();
|
||||||
|
let params = params
|
||||||
|
.into_iter()
|
||||||
|
.map(|val| val.as_zval(false))
|
||||||
|
.collect::<Result<Vec<_>>>()?;
|
||||||
|
let packed = params.into_boxed_slice();
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
let res = zend_hash_str_find_ptr_lc(
|
||||||
|
&(*self.ce).function_table,
|
||||||
|
name.as_ptr() as *const c_char,
|
||||||
|
name.len(),
|
||||||
|
) as *mut zend_function;
|
||||||
|
if res.is_null() {
|
||||||
|
return Err(Error::Callable);
|
||||||
|
}
|
||||||
|
zend_call_known_function(
|
||||||
|
res,
|
||||||
|
self as *const _ as *mut _,
|
||||||
|
self.ce,
|
||||||
|
&mut retval,
|
||||||
|
len as _,
|
||||||
|
packed.as_ptr() as *mut _,
|
||||||
|
std::ptr::null_mut(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(retval)
|
||||||
|
}
|
||||||
/// Attempts to read a property from the Object. Returns a result containing
|
/// Attempts to read a property from the Object. Returns a result containing
|
||||||
/// the value of the property if it exists and can be read, and an
|
/// the value of the property if it exists and can be read, and an
|
||||||
/// [`Error`] otherwise.
|
/// [`Error`] otherwise.
|
||||||
|
@ -453,3 +453,23 @@ impl<'a> FromZval<'a> for &'a str {
|
|||||||
zval.str()
|
zval.str()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
mod tests {
|
||||||
|
use crate::embed::Embed;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_string() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let result = Embed::eval("'foo';");
|
||||||
|
|
||||||
|
assert!(result.is_ok());
|
||||||
|
|
||||||
|
let zval = result.as_ref().unwrap();
|
||||||
|
|
||||||
|
assert!(zval.is_string());
|
||||||
|
assert_eq!(zval.string().unwrap(), "foo");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -206,6 +206,31 @@ impl Zval {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn try_call_method(&self, name: &str, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
||||||
|
self.object()
|
||||||
|
.ok_or(Error::Object)?
|
||||||
|
.try_call_method(name, params)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns the value of the zval if it is an internal indirect reference.
|
||||||
|
pub fn indirect(&self) -> Option<&Zval> {
|
||||||
|
if self.is_indirect() {
|
||||||
|
Some(unsafe { &*(self.value.zv as *mut Zval) })
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a mutable reference to the zval if it is an internal indirect reference.
|
||||||
|
pub fn indirect_mut(&self) -> Option<&mut Zval> {
|
||||||
|
if self.is_indirect() {
|
||||||
|
Some(unsafe { &mut *(self.value.zv as *mut Zval) })
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns the value of the zval if it is a reference.
|
/// Returns the value of the zval if it is a reference.
|
||||||
pub fn reference(&self) -> Option<&Zval> {
|
pub fn reference(&self) -> Option<&Zval> {
|
||||||
if self.is_reference() {
|
if self.is_reference() {
|
||||||
@ -257,6 +282,7 @@ impl Zval {
|
|||||||
/// # Parameters
|
/// # Parameters
|
||||||
///
|
///
|
||||||
/// * `params` - A list of parameters to call the function with.
|
/// * `params` - A list of parameters to call the function with.
|
||||||
|
#[inline(always)]
|
||||||
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
||||||
self.callable().ok_or(Error::Callable)?.try_call(params)
|
self.callable().ok_or(Error::Callable)?.try_call(params)
|
||||||
}
|
}
|
||||||
@ -321,6 +347,11 @@ impl Zval {
|
|||||||
self.get_type() == DataType::Reference
|
self.get_type() == DataType::Reference
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Returns true if the zval is a reference, false otherwise.
|
||||||
|
pub fn is_indirect(&self) -> bool {
|
||||||
|
self.get_type() == DataType::Indirect
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns true if the zval is callable, false otherwise.
|
/// Returns true if the zval is callable, false otherwise.
|
||||||
pub fn is_callable(&self) -> bool {
|
pub fn is_callable(&self) -> bool {
|
||||||
let ptr: *const Self = self;
|
let ptr: *const Self = self;
|
||||||
@ -593,6 +624,7 @@ impl Debug for Zval {
|
|||||||
DataType::ConstantExpression => field!(Option::<()>::None),
|
DataType::ConstantExpression => field!(Option::<()>::None),
|
||||||
DataType::Void => field!(Option::<()>::None),
|
DataType::Void => field!(Option::<()>::None),
|
||||||
DataType::Bool => field!(self.bool()),
|
DataType::Bool => field!(self.bool()),
|
||||||
|
DataType::Indirect => field!(self.indirect()),
|
||||||
// SAFETY: We are not accessing the pointer.
|
// SAFETY: We are not accessing the pointer.
|
||||||
DataType::Ptr => field!(unsafe { self.ptr::<c_void>() }),
|
DataType::Ptr => field!(unsafe { self.ptr::<c_void>() }),
|
||||||
};
|
};
|
||||||
|
@ -52,7 +52,6 @@ php_core_globals *ext_php_rs_process_globals() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
sapi_globals_struct *ext_php_rs_sapi_globals() {
|
sapi_globals_struct *ext_php_rs_sapi_globals() {
|
||||||
#ifdef ZTS
|
#ifdef ZTS
|
||||||
#ifdef ZEND_ENABLE_STATIC_TSRMLS_CACHE
|
#ifdef ZEND_ENABLE_STATIC_TSRMLS_CACHE
|
||||||
@ -65,7 +64,6 @@ sapi_globals_struct *ext_php_rs_sapi_globals() {
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
php_file_globals *ext_php_rs_file_globals() {
|
php_file_globals *ext_php_rs_file_globals() {
|
||||||
#ifdef ZTS
|
#ifdef ZTS
|
||||||
return TSRMG_FAST_BULK(file_globals_id, php_file_globals *);
|
return TSRMG_FAST_BULK(file_globals_id, php_file_globals *);
|
||||||
@ -73,3 +71,21 @@ php_file_globals *ext_php_rs_file_globals() {
|
|||||||
return &file_globals;
|
return &file_globals;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sapi_module_struct *ext_php_rs_sapi_module() {
|
||||||
|
return &sapi_module;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ext_php_rs_zend_try_catch(void* (*callback)(void *), void *ctx, void **result) {
|
||||||
|
zend_try {
|
||||||
|
*result = callback(ctx);
|
||||||
|
} zend_catch {
|
||||||
|
return true;
|
||||||
|
} zend_end_try();
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void ext_php_rs_zend_bailout() {
|
||||||
|
zend_bailout();
|
||||||
|
}
|
||||||
|
@ -22,8 +22,9 @@
|
|||||||
#include "zend_exceptions.h"
|
#include "zend_exceptions.h"
|
||||||
#include "zend_inheritance.h"
|
#include "zend_inheritance.h"
|
||||||
#include "zend_interfaces.h"
|
#include "zend_interfaces.h"
|
||||||
#include "SAPI.h"
|
|
||||||
#include "php_variables.h"
|
#include "php_variables.h"
|
||||||
|
#include "zend_ini.h"
|
||||||
|
#include "main/SAPI.h"
|
||||||
|
|
||||||
zend_string *ext_php_rs_zend_string_init(const char *str, size_t len, bool persistent);
|
zend_string *ext_php_rs_zend_string_init(const char *str, size_t len, bool persistent);
|
||||||
void ext_php_rs_zend_string_release(zend_string *zs);
|
void ext_php_rs_zend_string_release(zend_string *zs);
|
||||||
@ -37,3 +38,6 @@ zend_executor_globals *ext_php_rs_executor_globals();
|
|||||||
php_core_globals *ext_php_rs_process_globals();
|
php_core_globals *ext_php_rs_process_globals();
|
||||||
sapi_globals_struct *ext_php_rs_sapi_globals();
|
sapi_globals_struct *ext_php_rs_sapi_globals();
|
||||||
php_file_globals *ext_php_rs_file_globals();
|
php_file_globals *ext_php_rs_file_globals();
|
||||||
|
sapi_module_struct *ext_php_rs_sapi_module();
|
||||||
|
bool ext_php_rs_zend_try_catch(void* (*callback)(void *), void *ctx, void **result);
|
||||||
|
void ext_php_rs_zend_bailout();
|
||||||
|
@ -1,6 +1,12 @@
|
|||||||
//! Builder and objects for creating classes in the PHP world.
|
//! Builder and objects for creating classes in the PHP world.
|
||||||
|
|
||||||
use crate::{ffi::zend_class_entry, flags::ClassFlags, types::ZendStr, zend::ExecutorGlobals};
|
use crate::{
|
||||||
|
boxed::ZBox,
|
||||||
|
ffi::zend_class_entry,
|
||||||
|
flags::ClassFlags,
|
||||||
|
types::{ZendObject, ZendStr},
|
||||||
|
zend::ExecutorGlobals,
|
||||||
|
};
|
||||||
use std::{convert::TryInto, fmt::Debug, ops::DerefMut};
|
use std::{convert::TryInto, fmt::Debug, ops::DerefMut};
|
||||||
|
|
||||||
/// A PHP class entry.
|
/// A PHP class entry.
|
||||||
@ -22,6 +28,17 @@ impl ClassEntry {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Creates a new [`ZendObject`], returned inside an [`ZBox<ZendObject>`]
|
||||||
|
/// wrapper.
|
||||||
|
///
|
||||||
|
/// # Panics
|
||||||
|
///
|
||||||
|
/// Panics when allocating memory for the new object fails.
|
||||||
|
#[allow(clippy::new_ret_no_self)]
|
||||||
|
pub fn new(&self) -> ZBox<ZendObject> {
|
||||||
|
ZendObject::new(self)
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns the class flags.
|
/// Returns the class flags.
|
||||||
pub fn flags(&self) -> ClassFlags {
|
pub fn flags(&self) -> ClassFlags {
|
||||||
ClassFlags::from_bits_truncate(self.ce_flags)
|
ClassFlags::from_bits_truncate(self.ce_flags)
|
||||||
|
@ -6,6 +6,8 @@ use crate::{
|
|||||||
types::{ZendClassObject, ZendObject, Zval},
|
types::{ZendClassObject, ZendObject, Zval},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use super::function::Function;
|
||||||
|
|
||||||
/// Execute data passed when a function is called from PHP.
|
/// Execute data passed when a function is called from PHP.
|
||||||
///
|
///
|
||||||
/// This generally contains things related to the call, including but not
|
/// This generally contains things related to the call, including but not
|
||||||
@ -194,6 +196,16 @@ impl ExecuteData {
|
|||||||
self.This.object_mut()
|
self.This.object_mut()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Attempt to retrieve the function that is being called.
|
||||||
|
pub fn function(&self) -> Option<&Function> {
|
||||||
|
unsafe { self.func.as_ref() }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempt to retrieve the previous execute data on the call stack.
|
||||||
|
pub fn previous(&self) -> Option<&Self> {
|
||||||
|
unsafe { self.prev_execute_data.as_ref() }
|
||||||
|
}
|
||||||
|
|
||||||
/// Translation of macro `ZEND_CALL_ARG(call, n)`
|
/// Translation of macro `ZEND_CALL_ARG(call, n)`
|
||||||
/// zend_compile.h:578
|
/// zend_compile.h:578
|
||||||
///
|
///
|
||||||
|
@ -2,7 +2,18 @@
|
|||||||
|
|
||||||
use std::{fmt::Debug, os::raw::c_char, ptr};
|
use std::{fmt::Debug, os::raw::c_char, ptr};
|
||||||
|
|
||||||
use crate::ffi::zend_function_entry;
|
use crate::{
|
||||||
|
convert::IntoZvalDyn,
|
||||||
|
error::Result,
|
||||||
|
ffi::{
|
||||||
|
zend_call_known_function, zend_fetch_function_str, zend_function, zend_function_entry,
|
||||||
|
zend_hash_str_find_ptr_lc,
|
||||||
|
},
|
||||||
|
flags::FunctionType,
|
||||||
|
types::Zval,
|
||||||
|
};
|
||||||
|
|
||||||
|
use super::ClassEntry;
|
||||||
|
|
||||||
/// A Zend function entry.
|
/// A Zend function entry.
|
||||||
pub type FunctionEntry = zend_function_entry;
|
pub type FunctionEntry = zend_function_entry;
|
||||||
@ -36,3 +47,86 @@ impl FunctionEntry {
|
|||||||
Box::into_raw(Box::new(self))
|
Box::into_raw(Box::new(self))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub type Function = zend_function;
|
||||||
|
|
||||||
|
impl Function {
|
||||||
|
pub fn function_type(&self) -> FunctionType {
|
||||||
|
FunctionType::from(unsafe { self.type_ })
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn try_from_function(name: &str) -> Option<Self> {
|
||||||
|
unsafe {
|
||||||
|
let res = zend_fetch_function_str(name.as_ptr() as *const c_char, name.len());
|
||||||
|
if res.is_null() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
Some(*res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn try_from_method(class: &str, name: &str) -> Option<Self> {
|
||||||
|
match ClassEntry::try_find(class) {
|
||||||
|
None => None,
|
||||||
|
Some(ce) => unsafe {
|
||||||
|
let res = zend_hash_str_find_ptr_lc(
|
||||||
|
&ce.function_table,
|
||||||
|
name.as_ptr() as *const c_char,
|
||||||
|
name.len(),
|
||||||
|
) as *mut zend_function;
|
||||||
|
if res.is_null() {
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
Some(*res)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempts to call the callable with a list of arguments to pass to the
|
||||||
|
/// function.
|
||||||
|
///
|
||||||
|
/// You should not call this function directly, rather through the
|
||||||
|
/// [`call_user_func`] macro.
|
||||||
|
///
|
||||||
|
/// # Parameters
|
||||||
|
///
|
||||||
|
/// * `params` - A list of parameters to call the function with.
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// Returns the result wrapped in [`Ok`] upon success. If calling the
|
||||||
|
/// callable fails, or an exception is thrown, an [`Err`] is returned.
|
||||||
|
///
|
||||||
|
/// # Example
|
||||||
|
///
|
||||||
|
/// ```no_run
|
||||||
|
/// use ext_php_rs::types::ZendCallable;
|
||||||
|
///
|
||||||
|
/// let strpos = ZendCallable::try_from_name("strpos").unwrap();
|
||||||
|
/// let result = strpos.try_call(vec![&"hello", &"e"]).unwrap();
|
||||||
|
/// assert_eq!(result.long(), Some(1));
|
||||||
|
/// ```
|
||||||
|
#[inline(always)]
|
||||||
|
pub fn try_call(&self, params: Vec<&dyn IntoZvalDyn>) -> Result<Zval> {
|
||||||
|
let mut retval = Zval::new();
|
||||||
|
let len = params.len();
|
||||||
|
let params = params
|
||||||
|
.into_iter()
|
||||||
|
.map(|val| val.as_zval(false))
|
||||||
|
.collect::<Result<Vec<_>>>()?;
|
||||||
|
let packed = params.into_boxed_slice();
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
zend_call_known_function(
|
||||||
|
self as *const _ as *mut _,
|
||||||
|
std::ptr::null_mut(),
|
||||||
|
std::ptr::null_mut(),
|
||||||
|
&mut retval,
|
||||||
|
len as _,
|
||||||
|
packed.as_ptr() as *mut _,
|
||||||
|
std::ptr::null_mut(),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(retval)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
//! Types related to the PHP executor, sapi and process globals.
|
//! Types related to the PHP executor, sapi and process globals.
|
||||||
use std::ffi::CStr;
|
|
||||||
|
use std::collections::HashMap;
|
||||||
use std::ops::{Deref, DerefMut};
|
use std::ops::{Deref, DerefMut};
|
||||||
use std::slice;
|
use std::slice;
|
||||||
use std::str;
|
use std::str;
|
||||||
@ -15,15 +16,24 @@ use crate::ffi::{
|
|||||||
sapi_globals_struct, sapi_header_struct, sapi_headers_struct, sapi_request_info,
|
sapi_globals_struct, sapi_header_struct, sapi_headers_struct, sapi_request_info,
|
||||||
zend_is_auto_global, TRACK_VARS_COOKIE, TRACK_VARS_ENV, TRACK_VARS_FILES, TRACK_VARS_GET,
|
zend_is_auto_global, TRACK_VARS_COOKIE, TRACK_VARS_ENV, TRACK_VARS_FILES, TRACK_VARS_GET,
|
||||||
TRACK_VARS_POST, TRACK_VARS_REQUEST, TRACK_VARS_SERVER,
|
TRACK_VARS_POST, TRACK_VARS_REQUEST, TRACK_VARS_SERVER,
|
||||||
|
_sapi_globals_struct, _sapi_module_struct, _zend_executor_globals, ext_php_rs_executor_globals,
|
||||||
|
ext_php_rs_sapi_globals, ext_php_rs_sapi_module, zend_ini_entry,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::types::{ZendHashTable, ZendObject, ZendStr};
|
use crate::types::{ZendHashTable, ZendObject, ZendStr};
|
||||||
|
|
||||||
use super::linked_list::ZendLinkedListIterator;
|
use super::linked_list::ZendLinkedListIterator;
|
||||||
|
use crate::types::{ZendHashTable, ZendObject};
|
||||||
|
|
||||||
/// Stores global variables used in the PHP executor.
|
/// Stores global variables used in the PHP executor.
|
||||||
pub type ExecutorGlobals = _zend_executor_globals;
|
pub type ExecutorGlobals = _zend_executor_globals;
|
||||||
|
|
||||||
|
/// Stores global SAPI variables used in the PHP executor.
|
||||||
|
pub type SapiGlobals = _sapi_globals_struct;
|
||||||
|
|
||||||
|
/// Stores the SAPI module used in the PHP executor.
|
||||||
|
pub type SapiModule = _sapi_module_struct;
|
||||||
|
|
||||||
impl ExecutorGlobals {
|
impl ExecutorGlobals {
|
||||||
/// Returns a reference to the PHP executor globals.
|
/// Returns a reference to the PHP executor globals.
|
||||||
///
|
///
|
||||||
@ -62,6 +72,41 @@ impl ExecutorGlobals {
|
|||||||
unsafe { self.class_table.as_ref() }
|
unsafe { self.class_table.as_ref() }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Attempts to retrieve the global functions hash table.
|
||||||
|
pub fn function_table(&self) -> Option<&ZendHashTable> {
|
||||||
|
unsafe { self.function_table.as_ref() }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempts to retrieve the global functions hash table as mutable.
|
||||||
|
pub fn function_table_mut(&self) -> Option<&mut ZendHashTable> {
|
||||||
|
unsafe { self.function_table.as_mut() }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Retrieves the ini values for all ini directives in the current executor
|
||||||
|
/// context..
|
||||||
|
pub fn ini_values(&self) -> HashMap<String, Option<String>> {
|
||||||
|
let hash_table = unsafe { &*self.ini_directives };
|
||||||
|
let mut ini_hash_map: HashMap<String, Option<String>> = HashMap::new();
|
||||||
|
for (_index, key, value) in hash_table.iter() {
|
||||||
|
if let Some(key) = key {
|
||||||
|
ini_hash_map.insert(key, unsafe {
|
||||||
|
let ini_entry = &*value.ptr::<zend_ini_entry>().expect("Invalid ini entry");
|
||||||
|
if ini_entry.value.is_null() {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(
|
||||||
|
(*ini_entry.value)
|
||||||
|
.as_str()
|
||||||
|
.expect("Ini value is not a string")
|
||||||
|
.to_owned(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ini_hash_map
|
||||||
|
}
|
||||||
|
|
||||||
/// Attempts to retrieve the global constants table.
|
/// Attempts to retrieve the global constants table.
|
||||||
pub fn constants(&self) -> Option<&ZendHashTable> {
|
pub fn constants(&self) -> Option<&ZendHashTable> {
|
||||||
unsafe { self.zend_constants.as_ref() }
|
unsafe { self.zend_constants.as_ref() }
|
||||||
@ -97,6 +142,87 @@ impl ExecutorGlobals {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Cancel a requested an interrupt of the PHP VM.
|
||||||
|
pub fn cancel_interrupt(&mut self) {
|
||||||
|
cfg_if::cfg_if! {
|
||||||
|
if #[cfg(php82)] {
|
||||||
|
unsafe {
|
||||||
|
zend_atomic_bool_store(&mut self.vm_interrupt, false);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
self.vm_interrupt = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SapiGlobals {
|
||||||
|
/// Returns a reference to the PHP SAPI globals.
|
||||||
|
///
|
||||||
|
/// The executor globals are guarded by a RwLock. There can be multiple
|
||||||
|
/// immutable references at one time but only ever one mutable reference.
|
||||||
|
/// Attempting to retrieve the globals while already holding the global
|
||||||
|
/// guard will lead to a deadlock. Dropping the globals guard will release
|
||||||
|
/// the lock.
|
||||||
|
pub fn get() -> GlobalReadGuard<Self> {
|
||||||
|
// SAFETY: PHP executor globals are statically declared therefore should never
|
||||||
|
// return an invalid pointer.
|
||||||
|
let globals = unsafe { ext_php_rs_sapi_globals().as_ref() }
|
||||||
|
.expect("Static executor globals were invalid");
|
||||||
|
let guard = SAPI_LOCK.read();
|
||||||
|
GlobalReadGuard { globals, guard }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a mutable reference to the PHP executor globals.
|
||||||
|
///
|
||||||
|
/// The executor globals are guarded by a RwLock. There can be multiple
|
||||||
|
/// immutable references at one time but only ever one mutable reference.
|
||||||
|
/// Attempting to retrieve the globals while already holding the global
|
||||||
|
/// guard will lead to a deadlock. Dropping the globals guard will release
|
||||||
|
/// the lock.
|
||||||
|
pub fn get_mut() -> GlobalWriteGuard<Self> {
|
||||||
|
// SAFETY: PHP executor globals are statically declared therefore should never
|
||||||
|
// return an invalid pointer.
|
||||||
|
let globals = unsafe { ext_php_rs_sapi_globals().as_mut() }
|
||||||
|
.expect("Static executor globals were invalid");
|
||||||
|
let guard = SAPI_LOCK.write();
|
||||||
|
GlobalWriteGuard { globals, guard }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SapiModule {
|
||||||
|
/// Returns a reference to the PHP SAPI module.
|
||||||
|
///
|
||||||
|
/// The executor globals are guarded by a RwLock. There can be multiple
|
||||||
|
/// immutable references at one time but only ever one mutable reference.
|
||||||
|
/// Attempting to retrieve the globals while already holding the global
|
||||||
|
/// guard will lead to a deadlock. Dropping the globals guard will release
|
||||||
|
/// the lock.
|
||||||
|
pub fn get() -> GlobalReadGuard<Self> {
|
||||||
|
// SAFETY: PHP executor globals are statically declared therefore should never
|
||||||
|
// return an invalid pointer.
|
||||||
|
let globals = unsafe { ext_php_rs_sapi_module().as_ref() }
|
||||||
|
.expect("Static executor globals were invalid");
|
||||||
|
let guard = SAPI_MODULE_LOCK.read();
|
||||||
|
GlobalReadGuard { globals, guard }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns a mutable reference to the PHP executor globals.
|
||||||
|
///
|
||||||
|
/// The executor globals are guarded by a RwLock. There can be multiple
|
||||||
|
/// immutable references at one time but only ever one mutable reference.
|
||||||
|
/// Attempting to retrieve the globals while already holding the global
|
||||||
|
/// guard will lead to a deadlock. Dropping the globals guard will release
|
||||||
|
/// the lock.
|
||||||
|
pub fn get_mut() -> GlobalWriteGuard<Self> {
|
||||||
|
// SAFETY: PHP executor globals are statically declared therefore should never
|
||||||
|
// return an invalid pointer.
|
||||||
|
let globals = unsafe { ext_php_rs_sapi_module().as_mut() }
|
||||||
|
.expect("Static executor globals were invalid");
|
||||||
|
let guard = SAPI_MODULE_LOCK.write();
|
||||||
|
GlobalWriteGuard { globals, guard }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Stores global variables used in the PHP executor.
|
/// Stores global variables used in the PHP executor.
|
||||||
@ -430,6 +556,18 @@ static PROCESS_GLOBALS_LOCK: RwLock<()> = const_rwlock(());
|
|||||||
static SAPI_GLOBALS_LOCK: RwLock<()> = const_rwlock(());
|
static SAPI_GLOBALS_LOCK: RwLock<()> = const_rwlock(());
|
||||||
static FILE_GLOBALS_LOCK: RwLock<()> = const_rwlock(());
|
static FILE_GLOBALS_LOCK: RwLock<()> = const_rwlock(());
|
||||||
|
|
||||||
|
/// SAPI globals rwlock.
|
||||||
|
///
|
||||||
|
/// PHP provides no indication if the executor globals are being accessed so
|
||||||
|
/// this is only effective on the Rust side.
|
||||||
|
static SAPI_LOCK: RwLock<()> = const_rwlock(());
|
||||||
|
|
||||||
|
/// SAPI globals rwlock.
|
||||||
|
///
|
||||||
|
/// PHP provides no indication if the executor globals are being accessed so
|
||||||
|
/// this is only effective on the Rust side.
|
||||||
|
static SAPI_MODULE_LOCK: RwLock<()> = const_rwlock(());
|
||||||
|
|
||||||
/// Wrapper guard that contains a reference to a given type `T`. Dropping a
|
/// Wrapper guard that contains a reference to a given type `T`. Dropping a
|
||||||
/// guard releases the lock on the relevant rwlock.
|
/// guard releases the lock on the relevant rwlock.
|
||||||
pub struct GlobalReadGuard<T: 'static> {
|
pub struct GlobalReadGuard<T: 'static> {
|
||||||
|
@ -238,6 +238,7 @@ impl ZendObjectHandlers {
|
|||||||
let mut zv = Zval::new();
|
let mut zv = Zval::new();
|
||||||
val.get(self_, &mut zv)?;
|
val.get(self_, &mut zv)?;
|
||||||
|
|
||||||
|
#[allow(clippy::unnecessary_mut_passed)]
|
||||||
if zend_is_true(&mut zv) == 1 {
|
if zend_is_true(&mut zv) == 1 {
|
||||||
return Ok(1);
|
return Ok(1);
|
||||||
}
|
}
|
||||||
|
58
src/zend/ini_entry_def.rs
Normal file
58
src/zend/ini_entry_def.rs
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
//! Builder for creating inis and methods in PHP.
|
||||||
|
//! See <https://www.phpinternalsbook.com/php7/extensions_design/ini_settings.html> for details.
|
||||||
|
|
||||||
|
use std::{ffi::CString, os::raw::c_char, ptr};
|
||||||
|
|
||||||
|
use crate::{ffi::zend_ini_entry_def, ffi::zend_register_ini_entries, flags::IniEntryPermission};
|
||||||
|
|
||||||
|
/// A Zend ini entry definition.
|
||||||
|
///
|
||||||
|
/// To register ini definitions for extensions, the IniEntryDef builder should
|
||||||
|
/// be used. Ini entries should be registered in your module's startup_function
|
||||||
|
/// via `IniEntryDef::register(Vec<IniEntryDef>)`.
|
||||||
|
pub type IniEntryDef = zend_ini_entry_def;
|
||||||
|
|
||||||
|
impl IniEntryDef {
|
||||||
|
/// Returns an empty ini entry, signifying the end of a ini list.
|
||||||
|
pub fn new(name: String, default_value: String, permission: IniEntryPermission) -> Self {
|
||||||
|
let mut template = Self::end();
|
||||||
|
let name = CString::new(name).expect("Unable to create CString from name");
|
||||||
|
let value = CString::new(default_value).expect("Unable to create CString from value");
|
||||||
|
template.name_length = name.as_bytes().len() as _;
|
||||||
|
template.name = name.into_raw();
|
||||||
|
template.value_length = value.as_bytes().len() as _;
|
||||||
|
template.value = value.into_raw();
|
||||||
|
template.modifiable = IniEntryPermission::PerDir.bits() as _;
|
||||||
|
template.modifiable = permission.bits() as _;
|
||||||
|
template
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns an empty ini entry def, signifying the end of a ini list.
|
||||||
|
pub fn end() -> Self {
|
||||||
|
Self {
|
||||||
|
name: ptr::null() as *const c_char,
|
||||||
|
on_modify: None,
|
||||||
|
mh_arg1: std::ptr::null_mut(),
|
||||||
|
mh_arg2: std::ptr::null_mut(),
|
||||||
|
mh_arg3: std::ptr::null_mut(),
|
||||||
|
value: std::ptr::null_mut(),
|
||||||
|
displayer: None,
|
||||||
|
modifiable: 0,
|
||||||
|
value_length: 0,
|
||||||
|
name_length: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Converts the ini entry into a raw and pointer, releasing it to the
|
||||||
|
/// C world.
|
||||||
|
pub fn into_raw(self) -> *mut Self {
|
||||||
|
Box::into_raw(Box::new(self))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn register(mut entries: Vec<Self>, module_number: i32) {
|
||||||
|
entries.push(Self::end());
|
||||||
|
let entries = Box::into_raw(entries.into_boxed_slice()) as *const Self;
|
||||||
|
|
||||||
|
unsafe { zend_register_ini_entries(entries, module_number) };
|
||||||
|
}
|
||||||
|
}
|
@ -8,14 +8,20 @@ mod function;
|
|||||||
mod globals;
|
mod globals;
|
||||||
mod handlers;
|
mod handlers;
|
||||||
mod linked_list;
|
mod linked_list;
|
||||||
|
mod ini_entry_def;
|
||||||
mod module;
|
mod module;
|
||||||
|
mod try_catch;
|
||||||
|
|
||||||
use crate::{error::Result, ffi::php_printf};
|
use crate::{
|
||||||
|
error::Result,
|
||||||
|
ffi::{php_printf, sapi_module},
|
||||||
|
};
|
||||||
use std::ffi::CString;
|
use std::ffi::CString;
|
||||||
|
|
||||||
pub use _type::ZendType;
|
pub use _type::ZendType;
|
||||||
pub use class::ClassEntry;
|
pub use class::ClassEntry;
|
||||||
pub use ex::ExecuteData;
|
pub use ex::ExecuteData;
|
||||||
|
pub use function::Function;
|
||||||
pub use function::FunctionEntry;
|
pub use function::FunctionEntry;
|
||||||
pub use globals::ExecutorGlobals;
|
pub use globals::ExecutorGlobals;
|
||||||
pub use globals::FileGlobals;
|
pub use globals::FileGlobals;
|
||||||
@ -23,7 +29,14 @@ pub use globals::ProcessGlobals;
|
|||||||
pub use globals::SapiGlobals;
|
pub use globals::SapiGlobals;
|
||||||
pub use handlers::ZendObjectHandlers;
|
pub use handlers::ZendObjectHandlers;
|
||||||
pub use linked_list::ZendLinkedList;
|
pub use linked_list::ZendLinkedList;
|
||||||
|
pub use globals::SapiGlobals;
|
||||||
|
pub use globals::SapiModule;
|
||||||
|
pub use handlers::ZendObjectHandlers;
|
||||||
|
pub use ini_entry_def::IniEntryDef;
|
||||||
pub use module::ModuleEntry;
|
pub use module::ModuleEntry;
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
pub(crate) use try_catch::panic_wrapper;
|
||||||
|
pub use try_catch::{bailout, try_catch};
|
||||||
|
|
||||||
// Used as the format string for `php_printf`.
|
// Used as the format string for `php_printf`.
|
||||||
const FORMAT_STR: &[u8] = b"%s\0";
|
const FORMAT_STR: &[u8] = b"%s\0";
|
||||||
@ -47,3 +60,9 @@ pub fn printf(message: &str) -> Result<()> {
|
|||||||
};
|
};
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get the name of the SAPI module.
|
||||||
|
pub fn php_sapi_name() -> String {
|
||||||
|
let c_str = unsafe { std::ffi::CStr::from_ptr(sapi_module.name) };
|
||||||
|
c_str.to_str().expect("Unable to parse CStr").to_string()
|
||||||
|
}
|
||||||
|
166
src/zend/try_catch.rs
Normal file
166
src/zend/try_catch.rs
Normal file
@ -0,0 +1,166 @@
|
|||||||
|
use crate::ffi::{ext_php_rs_zend_bailout, ext_php_rs_zend_try_catch};
|
||||||
|
use std::ffi::c_void;
|
||||||
|
use std::panic::{catch_unwind, resume_unwind, RefUnwindSafe};
|
||||||
|
use std::ptr::null_mut;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub struct CatchError;
|
||||||
|
|
||||||
|
pub(crate) unsafe extern "C" fn panic_wrapper<R, F: FnMut() -> R + RefUnwindSafe>(
|
||||||
|
ctx: *const c_void,
|
||||||
|
) -> *const c_void {
|
||||||
|
// we try to catch panic here so we correctly shutdown php if it happens
|
||||||
|
// mandatory when we do assert on test as other test would not run correctly
|
||||||
|
let panic = catch_unwind(|| (*(ctx as *mut F))());
|
||||||
|
|
||||||
|
Box::into_raw(Box::new(panic)) as *mut c_void
|
||||||
|
}
|
||||||
|
|
||||||
|
/// PHP propose a try catch mechanism in C using setjmp and longjmp (bailout)
|
||||||
|
/// It store the arg of setjmp into the bailout field of the global executor
|
||||||
|
/// If a bailout is triggered, the executor will jump to the setjmp and restore
|
||||||
|
/// the previous setjmp
|
||||||
|
///
|
||||||
|
/// try_catch allow to use this mechanism
|
||||||
|
///
|
||||||
|
/// # Returns
|
||||||
|
///
|
||||||
|
/// * `Ok(R)` - The result of the function
|
||||||
|
/// * `Err(CatchError)` - A bailout occurred during the execution
|
||||||
|
pub fn try_catch<R, F: FnMut() -> R + RefUnwindSafe>(func: F) -> Result<R, CatchError> {
|
||||||
|
let mut panic_ptr = null_mut();
|
||||||
|
let has_bailout = unsafe {
|
||||||
|
ext_php_rs_zend_try_catch(
|
||||||
|
panic_wrapper::<R, F>,
|
||||||
|
&func as *const F as *const c_void,
|
||||||
|
(&mut panic_ptr) as *mut *mut c_void,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
let panic = panic_ptr as *mut std::thread::Result<R>;
|
||||||
|
|
||||||
|
// can be null if there is a bailout
|
||||||
|
if panic.is_null() || has_bailout {
|
||||||
|
return Err(CatchError);
|
||||||
|
}
|
||||||
|
|
||||||
|
match unsafe { *Box::from_raw(panic as *mut std::thread::Result<R>) } {
|
||||||
|
Ok(r) => Ok(r),
|
||||||
|
Err(err) => {
|
||||||
|
// we resume the panic here so it can be catched correctly by the test framework
|
||||||
|
resume_unwind(err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Trigger a bailout
|
||||||
|
///
|
||||||
|
/// This function will stop the execution of the current script
|
||||||
|
/// and jump to the last try catch block
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// This function is unsafe because it can cause memory leaks
|
||||||
|
/// Since it will jump to the last try catch block, it will not call the
|
||||||
|
/// destructor of the current scope
|
||||||
|
///
|
||||||
|
/// When using this function you should ensure that all the memory allocated in
|
||||||
|
/// the current scope is released
|
||||||
|
pub unsafe fn bailout() -> ! {
|
||||||
|
ext_php_rs_zend_bailout();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use crate::embed::Embed;
|
||||||
|
use crate::zend::{bailout, try_catch};
|
||||||
|
use std::ptr::null_mut;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_catch() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let catch = try_catch(|| {
|
||||||
|
unsafe {
|
||||||
|
bailout();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(unreachable_code)]
|
||||||
|
{
|
||||||
|
assert!(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
assert!(catch.is_err());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_no_catch() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let catch = try_catch(|| {
|
||||||
|
assert!(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
assert!(catch.is_ok());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_bailout() {
|
||||||
|
Embed::run(|| {
|
||||||
|
unsafe {
|
||||||
|
bailout();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(unreachable_code)]
|
||||||
|
{
|
||||||
|
assert!(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[should_panic]
|
||||||
|
fn test_panic() {
|
||||||
|
Embed::run(|| {
|
||||||
|
let _ = try_catch(|| {
|
||||||
|
panic!("should panic");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_return() {
|
||||||
|
let foo = Embed::run(|| {
|
||||||
|
let result = try_catch(|| {
|
||||||
|
return "foo";
|
||||||
|
});
|
||||||
|
|
||||||
|
assert!(result.is_ok());
|
||||||
|
|
||||||
|
result.unwrap()
|
||||||
|
});
|
||||||
|
|
||||||
|
assert_eq!(foo, "foo");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_memory_leak() {
|
||||||
|
let mut ptr = null_mut();
|
||||||
|
|
||||||
|
let _ = try_catch(|| {
|
||||||
|
let mut result = "foo".to_string();
|
||||||
|
ptr = &mut result;
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
bailout();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Check that the string is never released
|
||||||
|
let result = unsafe { &*ptr as &str };
|
||||||
|
|
||||||
|
assert_eq!(result, "foo");
|
||||||
|
}
|
||||||
|
}
|
44
tests/module.rs
Normal file
44
tests/module.rs
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
#![cfg_attr(windows, feature(abi_vectorcall))]
|
||||||
|
extern crate ext_php_rs;
|
||||||
|
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
use ext_php_rs::embed::Embed;
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
use ext_php_rs::ffi::zend_register_module_ex;
|
||||||
|
use ext_php_rs::prelude::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
fn test_module() {
|
||||||
|
Embed::run(|| {
|
||||||
|
// Allow to load the module
|
||||||
|
unsafe { zend_register_module_ex(get_module()) };
|
||||||
|
|
||||||
|
let result = Embed::eval("$foo = hello_world('foo');");
|
||||||
|
|
||||||
|
assert!(result.is_ok());
|
||||||
|
|
||||||
|
let zval = result.unwrap();
|
||||||
|
|
||||||
|
assert!(zval.is_string());
|
||||||
|
|
||||||
|
let string = zval.string().unwrap();
|
||||||
|
|
||||||
|
assert_eq!(string.to_string(), "Hello, foo!");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Gives you a nice greeting!
|
||||||
|
///
|
||||||
|
/// @param string $name Your name.
|
||||||
|
///
|
||||||
|
/// @return string Nice greeting!
|
||||||
|
#[php_function]
|
||||||
|
pub fn hello_world(name: String) -> String {
|
||||||
|
format!("Hello, {}!", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[php_module]
|
||||||
|
pub fn module(module: ModuleBuilder) -> ModuleBuilder {
|
||||||
|
module
|
||||||
|
}
|
@ -55,4 +55,11 @@ impl<'a> PHPProvider<'a> for Provider {
|
|||||||
fn get_defines(&self) -> Result<Vec<(&'static str, &'static str)>> {
|
fn get_defines(&self) -> Result<Vec<(&'static str, &'static str)>> {
|
||||||
Ok(vec![])
|
Ok(vec![])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn print_extra_link_args(&self) -> Result<()> {
|
||||||
|
#[cfg(feature = "embed")]
|
||||||
|
println!("cargo:rustc-link-lib=php");
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user