mirror of
https://github.com/oven-sh/bun
synced 2026-02-11 11:29:02 +00:00
* Fix stack overflow in large files * Add test for stack overflow * wip * Disable cache in debug build * Remove our extra `captureStackTrace` call * Update RuntimeTranspilerCache.zig * Update RuntimeTranspilerCache.zig * Fix issues with integer environment variables * Add missing ref * Add missing null check * Update bindings.cpp * Update transpiler-cache.test.ts * Add version check --------- Co-authored-by: Jarred Sumner <709451+Jarred-Sumner@users.noreply.github.com>
171 lines
6.3 KiB
C++
171 lines
6.3 KiB
C++
#include "root.h"
|
|
#include "ZigGlobalObject.h"
|
|
|
|
#include "helpers.h"
|
|
|
|
#include <JavaScriptCore/JSObject.h>
|
|
#include <JavaScriptCore/ObjectConstructor.h>
|
|
#include "BunClientData.h"
|
|
using namespace JSC;
|
|
|
|
extern "C" size_t Bun__getEnvCount(JSGlobalObject* globalObject, void** list_ptr);
|
|
extern "C" size_t Bun__getEnvKey(void* list, size_t index, unsigned char** out);
|
|
|
|
extern "C" bool Bun__getEnvValue(JSGlobalObject* globalObject, ZigString* name, ZigString* value);
|
|
|
|
namespace Bun {
|
|
|
|
using namespace WebCore;
|
|
|
|
JSC_DEFINE_CUSTOM_GETTER(jsGetterEnvironmentVariable, (JSGlobalObject * globalObject, JSC::EncodedJSValue thisValue, PropertyName propertyName))
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
auto* thisObject = jsDynamicCast<JSObject*>(JSValue::decode(thisValue));
|
|
if (UNLIKELY(!thisObject))
|
|
return JSValue::encode(jsUndefined());
|
|
|
|
ZigString name = toZigString(propertyName.publicName());
|
|
ZigString value = { nullptr, 0 };
|
|
|
|
if (UNLIKELY(name.len == 0))
|
|
return JSValue::encode(jsUndefined());
|
|
|
|
if (!Bun__getEnvValue(globalObject, &name, &value)) {
|
|
return JSValue::encode(jsUndefined());
|
|
}
|
|
|
|
JSValue result = jsString(vm, Zig::toStringCopy(value));
|
|
thisObject->putDirect(vm, propertyName, result, 0);
|
|
return JSValue::encode(result);
|
|
}
|
|
|
|
JSC_DEFINE_CUSTOM_SETTER(jsSetterEnvironmentVariable, (JSGlobalObject * globalObject, JSC::EncodedJSValue thisValue, JSC::EncodedJSValue value, PropertyName propertyName))
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
JSC::JSObject* object = JSValue::decode(thisValue).getObject();
|
|
if (!object)
|
|
return false;
|
|
|
|
object->putDirect(vm, propertyName, JSValue::decode(value), 0);
|
|
return true;
|
|
}
|
|
|
|
JSC_DEFINE_CUSTOM_GETTER(jsTimeZoneEnvironmentVariableGetter, (JSGlobalObject * globalObject, JSC::EncodedJSValue thisValue, PropertyName propertyName))
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
auto* thisObject = jsDynamicCast<JSObject*>(JSValue::decode(thisValue));
|
|
if (UNLIKELY(!thisObject))
|
|
return JSValue::encode(jsUndefined());
|
|
|
|
auto* clientData = WebCore::clientData(vm);
|
|
|
|
ZigString name = toZigString(propertyName.publicName());
|
|
ZigString value = { nullptr, 0 };
|
|
|
|
if (auto hasExistingValue = thisObject->getIfPropertyExists(globalObject, clientData->builtinNames().dataPrivateName())) {
|
|
return JSValue::encode(hasExistingValue);
|
|
}
|
|
|
|
if (!Bun__getEnvValue(globalObject, &name, &value) || value.len == 0) {
|
|
return JSValue::encode(jsUndefined());
|
|
}
|
|
|
|
JSValue out = jsString(vm, Zig::toStringCopy(value));
|
|
thisObject->putDirect(vm, clientData->builtinNames().dataPrivateName(), out, 0);
|
|
|
|
return JSValue::encode(out);
|
|
}
|
|
|
|
// In Node.js, the "TZ" environment variable is special.
|
|
// Setting it automatically updates the timezone.
|
|
// We also expose an explicit setTimeZone function in bun:jsc
|
|
JSC_DEFINE_CUSTOM_SETTER(jsTimeZoneEnvironmentVariableSetter, (JSGlobalObject * globalObject, JSC::EncodedJSValue thisValue, JSC::EncodedJSValue value, PropertyName propertyName))
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
JSC::JSObject* object = JSValue::decode(thisValue).getObject();
|
|
if (!object)
|
|
return false;
|
|
|
|
JSValue decodedValue = JSValue::decode(value);
|
|
if (decodedValue.isString()) {
|
|
auto timeZoneName = decodedValue.toWTFString(globalObject);
|
|
if (timeZoneName.length() < 32) {
|
|
if (WTF::setTimeZoneOverride(timeZoneName)) {
|
|
vm.dateCache.resetIfNecessarySlow();
|
|
}
|
|
}
|
|
}
|
|
|
|
auto* clientData = WebCore::clientData(vm);
|
|
auto* builtinNames = &clientData->builtinNames();
|
|
auto privateName = builtinNames->dataPrivateName();
|
|
object->putDirect(vm, privateName, JSValue::decode(value), 0);
|
|
|
|
// TODO: this is an assertion failure
|
|
// Recreate this because the property visibility needs to be set correctly
|
|
// object->putDirectWithoutTransition(vm, propertyName, JSC::CustomGetterSetter::create(vm, jsTimeZoneEnvironmentVariableGetter, jsTimeZoneEnvironmentVariableSetter), JSC::PropertyAttribute::CustomAccessor | 0);
|
|
return true;
|
|
}
|
|
|
|
JSValue createEnvironmentVariablesMap(Zig::GlobalObject* globalObject)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
void* list;
|
|
size_t count = Bun__getEnvCount(globalObject, &list);
|
|
JSC::JSObject* object = nullptr;
|
|
if (count < 63) {
|
|
object = constructEmptyObject(globalObject, globalObject->objectPrototype(), count);
|
|
} else {
|
|
object = constructEmptyObject(globalObject, globalObject->objectPrototype());
|
|
}
|
|
|
|
static NeverDestroyed<String> TZ = MAKE_STATIC_STRING_IMPL("TZ");
|
|
bool hasTZ = false;
|
|
for (size_t i = 0; i < count; i++) {
|
|
unsigned char* chars;
|
|
size_t len = Bun__getEnvKey(list, i, &chars);
|
|
auto name = String::fromUTF8(chars, len);
|
|
if (name == TZ) {
|
|
hasTZ = true;
|
|
continue;
|
|
}
|
|
ASSERT(len > 0);
|
|
|
|
Identifier identifier = Identifier::fromString(vm, name);
|
|
|
|
// CustomGetterSetter doesn't support indexed properties yet.
|
|
// This causes strange issues when the environment variable name is an integer.
|
|
if (UNLIKELY(chars[0] >= '0' && chars[0] <= '9')) {
|
|
if (auto index = parseIndex(identifier)) {
|
|
ZigString valueString = { nullptr, 0 };
|
|
ZigString nameStr = toZigString(name);
|
|
JSValue value = jsUndefined();
|
|
if (Bun__getEnvValue(globalObject, &nameStr, &valueString)) {
|
|
value = jsString(vm, Zig::toStringCopy(valueString));
|
|
}
|
|
object->putDirectIndex(globalObject, *index, value, 0, PutDirectIndexLikePutDirect);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
object->putDirectCustomAccessor(vm, identifier, JSC::CustomGetterSetter::create(vm, jsGetterEnvironmentVariable, jsSetterEnvironmentVariable), JSC::PropertyAttribute::CustomAccessor | 0);
|
|
}
|
|
|
|
unsigned int TZAttrs = JSC::PropertyAttribute::CustomAccessor | 0;
|
|
if (!hasTZ) {
|
|
TZAttrs |= JSC::PropertyAttribute::DontEnum;
|
|
}
|
|
object->putDirectCustomAccessor(
|
|
vm,
|
|
Identifier::fromString(vm, TZ), JSC::CustomGetterSetter::create(vm, jsTimeZoneEnvironmentVariableGetter, jsTimeZoneEnvironmentVariableSetter), TZAttrs);
|
|
|
|
return object;
|
|
}
|
|
}
|