執行php文件:
- func Test_exec(t *testing.T) {
- engine.Initialize()
- ctx := &engine.Context{
- Output: os.Stdout,
- }
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- err = ctx.Exec("/tmp/index.php")
- iferr != nil {
- fmt.Println(err)
- }
- }
其中 /tmp/index.php 的內容為:
- echo("hello/n");
- Eval,返回值
- func Test_eval(t *testing.T) {
- engine.Initialize()
- ctx := &engine.Context{}
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- val, err := ctx.Eval("return 'hello';")
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.DestroyValue(val)
- ifengine.ToString(val) !="hello"{
- t.FailNow()
- }
- }
返回的value的生命周期所有權是golang程序,所以我們要負責DestroyValue
設置全局變量來傳參
- func Test_argument(t *testing.T) {
- engine.Initialize()
- ctx := &engine.Context{}
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- err = ctx.Bind("greeting","hello")
- iferr != nil {
- fmt.Println(err)
- }
- val, err := ctx.Eval("return $greeting;")
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.DestroyValue(val)
- ifengine.ToString(val) !="hello"{
- t.FailNow()
- }
- }
傳遞進去的參數的生命周期是php控制的,在request shutdown的時候內存會被釋放。
PHP 回調 Golang
- type greetingProvider struct {
- greeting string
- }
- func (provider *greetingProvider) GetGreeting() string {
- returnprovider.greeting
- }
- func newGreetingProvider(args []interface{})interface{} {
- return&greetingProvider{
- greeting: args[0].(string),
- }
- }
- func Test_callback(t *testing.T) {
- engine.Initialize()
- ctx := &engine.Context{}
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- err = engine.Define("GreetingProvider", newGreetingProvider)
- iferr != nil {
- fmt.Println(err)
- }
- val, err := ctx.Eval(`
- $greetingProvider=newGreetingProvider('hello');
- return$greetingProvider->GetGreeting();`)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.DestroyValue(val)
- ifengine.ToString(val) !="hello"{
- t.FailNow()
- }
- }
PHP 錯誤日志
- func Test_log(t *testing.T) {
- engine.PHP_INI_PATH_OVERRIDE ="/tmp/php.ini"
- engine.Initialize()
- ctx := &engine.Context{
- Log: os.Stderr,
- }
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- _, err = ctx.Eval("error_log('hello', 4); trigger_error('sent from golang', E_USER_ERROR);")
- iferr != nil {
- fmt.Println(err)
- }
- }
其中 /tmp/php.ini 的內容為
error_reporting= E_ALL
error_log="/tmp/php-error.log"
錯誤會被輸出到 /tmp/php-error.log。直接調用error_log會同時再輸出一份到stderr
HTTP 輸入輸出
- func Test_http(t *testing.T) {
- engine.Initialize()
- recorder := httptest.NewRecorder()
- ctx := &engine.Context{
- Request: httptest.NewRequest("GET","/hello", nil),
- ResponseWriter: recorder,
- }
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- _, err = ctx.Eval("echo($_SERVER['REQUEST_URI']);")
- iferr != nil {
- fmt.Println(err)
- }
- body, err := ioutil.ReadAll(recorder.Result().Body)
- iferr != nil {
- fmt.Println(err)
- }
- ifstring(body) !="/hello"{
- t.FailNow()
- }
- }
所有的PHP超級全局變量都會被初始化為傳遞進去的Request的值,包括:
- $_SERVER
- $_GET
- $_POST
- $_FILE
- $_COOKIE
- $_ENV
echo的內容,http code和http header會被寫回到傳入的ResponseWriter
fastcgi_finish_request
PHP-FPM 很常用的一個功能是 fastcgi_finish_request ,用于在php里做一些異步完成的事情。這個特殊的全局函數必須支持
- func Test_fastcgi_finish_reqeust(t *testing.T) {
- engine.Initialize()
- buffer := &bytes.Buffer{}
- ctx := &engine.Context{
- Output: buffer,
- }
- err := engine.RequestStartup(ctx)
- iferr != nil {
- fmt.Println(err)
- }
- defer engine.RequestShutdown(ctx)
- ctx.Eval("ob_start(); echo ('hello');")
- ifbuffer.String() !=""{
- t.FailNow()
- }
- ctx.Eval("fastcgi_finish_request();")
- ifbuffer.String() !="hello"{
- t.FailNow()
- }
- }
實際的作用就是把output提前輸出到 ResposneWriter 里去,讓調用方知道結果。對于當前進程的執行其實是沒有影響的,只是影響了output。
新聞熱點
疑難解答