mirror of
https://github.com/dunglas/frankenphp.git
synced 2025-12-24 13:38:11 +08:00
153 lines
3.9 KiB
Go
153 lines
3.9 KiB
Go
package frankenphp
|
|
|
|
import (
|
|
"errors"
|
|
"log/slog"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"go.uber.org/zap/exp/zapslog"
|
|
"go.uber.org/zap/zaptest"
|
|
)
|
|
|
|
// execute the function on a PHP thread directly
|
|
// this is necessary if tests make use of PHP's internal allocation
|
|
func testOnDummyPHPThread(t *testing.T, cb func()) {
|
|
t.Helper()
|
|
logger = slog.New(zapslog.NewHandler(zaptest.NewLogger(t).Core()))
|
|
assert.NoError(t, Init(
|
|
WithWorkers("tw", "./testdata/tasks/task-worker.php", 1, AsTaskWorker(true, 0)),
|
|
WithNumThreads(2),
|
|
WithLogger(logger),
|
|
))
|
|
defer Shutdown()
|
|
|
|
task, err := executeOnPHPThread(cb, "tw")
|
|
assert.NoError(t, err)
|
|
|
|
task.WaitForCompletion()
|
|
}
|
|
|
|
// executeOnPHPThread executes the callback func() directly on a task worker thread
|
|
// Currently only used in tests
|
|
func executeOnPHPThread(callback func(), taskWorkerName string) (*PendingTask, error) {
|
|
tw := getTaskWorkerByName(taskWorkerName)
|
|
if tw == nil {
|
|
return nil, errors.New("no task worker found with name " + taskWorkerName)
|
|
}
|
|
|
|
pt := &PendingTask{callback: callback}
|
|
err := pt.dispatch(tw)
|
|
|
|
return pt, err
|
|
}
|
|
|
|
func TestGoString(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalString := "Hello, World!"
|
|
|
|
phpString := PHPString(originalString, false)
|
|
defer zendStringRelease(phpString)
|
|
|
|
assert.Equal(t, originalString, GoString(phpString), "string -> zend_string -> string should yield an equal string")
|
|
})
|
|
}
|
|
|
|
func TestPHPMap(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalMap := map[string]any{
|
|
"foo1": "bar1",
|
|
"foo2": "bar2",
|
|
}
|
|
|
|
phpArray := PHPMap(originalMap)
|
|
defer zvalPtrDtor(phpArray)
|
|
|
|
assert.Equal(t, originalMap, GoMap(phpArray), "associative array should be equal after conversion")
|
|
})
|
|
}
|
|
|
|
func TestOrderedPHPAssociativeArray(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalArray := AssociativeArray{
|
|
Map: map[string]any{
|
|
"foo1": "bar1",
|
|
"foo2": "bar2",
|
|
},
|
|
Order: []string{"foo2", "foo1"},
|
|
}
|
|
|
|
phpArray := PHPAssociativeArray(originalArray)
|
|
defer zvalPtrDtor(phpArray)
|
|
|
|
assert.Equal(t, originalArray, GoAssociativeArray(phpArray), "associative array should be equal after conversion")
|
|
})
|
|
}
|
|
|
|
func TestPHPPackedArray(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalSlice := []any{"bar1", "bar2"}
|
|
|
|
phpArray := PHPPackedArray(originalSlice)
|
|
defer zvalPtrDtor(phpArray)
|
|
|
|
assert.Equal(t, originalSlice, GoPackedArray(phpArray), "slice should be equal after conversion")
|
|
})
|
|
}
|
|
|
|
func TestPHPPackedArrayToGoMap(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalSlice := []any{"bar1", "bar2"}
|
|
expectedMap := map[string]any{
|
|
"0": "bar1",
|
|
"1": "bar2",
|
|
}
|
|
|
|
phpArray := PHPPackedArray(originalSlice)
|
|
defer zvalPtrDtor(phpArray)
|
|
|
|
assert.Equal(t, expectedMap, GoMap(phpArray), "convert a packed to an associative array")
|
|
})
|
|
}
|
|
|
|
func TestPHPAssociativeArrayToPacked(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalArray := AssociativeArray{
|
|
Map: map[string]any{
|
|
"foo1": "bar1",
|
|
"foo2": "bar2",
|
|
},
|
|
Order: []string{"foo1", "foo2"},
|
|
}
|
|
expectedSlice := []any{"bar1", "bar2"}
|
|
|
|
phpArray := PHPAssociativeArray(originalArray)
|
|
defer zvalPtrDtor(phpArray)
|
|
|
|
assert.Equal(t, expectedSlice, GoPackedArray(phpArray), "convert an associative array to a slice")
|
|
})
|
|
}
|
|
|
|
func TestNestedMixedArray(t *testing.T) {
|
|
testOnDummyPHPThread(t, func() {
|
|
originalArray := map[string]any{
|
|
"string": "value",
|
|
"int": int64(123),
|
|
"float": float64(1.2),
|
|
"true": true,
|
|
"false": false,
|
|
"nil": nil,
|
|
"packedArray": []any{"bar1", "bar2"},
|
|
"associativeArray": AssociativeArray{
|
|
Map: map[string]any{"foo1": "bar1", "foo2": "bar2"},
|
|
Order: []string{"foo2", "foo1"},
|
|
},
|
|
}
|
|
|
|
phpArray := PHPMap(originalArray)
|
|
defer zvalPtrDtor(phpArray)
|
|
|
|
assert.Equal(t, originalArray, GoMap(phpArray), "nested mixed array should be equal after conversion")
|
|
})
|
|
}
|