mirror of
				https://github.com/robertkrimen/otto
				synced 2025-10-26 20:28:49 +08:00 
			
		
		
		
	 7163cd7841
			
		
	
	
		7163cd7841
		
	
	
	
	
		
			
			* The "reflect" package is willing to expose unexported fields in structs (a little suprising). * Skip unexported fields when enumerating or trying to access (pretend they don't exist). * Allow structs that implement json.Marshaler to work as expected in a JSON.stringify call. If we let the runtime try to access unexported fields, it'll cause a panic. This fixes #64
		
			
				
	
	
		
			286 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			286 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package otto
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"encoding/json"
 | |
| 	"math"
 | |
| 	"strings"
 | |
| )
 | |
| 
 | |
| type _builtinJSON_parseContext struct {
 | |
| 	call    FunctionCall
 | |
| 	reviver Value
 | |
| }
 | |
| 
 | |
| func builtinJSON_parse(call FunctionCall) Value {
 | |
| 	ctx := _builtinJSON_parseContext{
 | |
| 		call: call,
 | |
| 	}
 | |
| 	revive := false
 | |
| 	if reviver := call.Argument(1); reviver.isCallable() {
 | |
| 		revive = true
 | |
| 		ctx.reviver = reviver
 | |
| 	}
 | |
| 
 | |
| 	var root interface{}
 | |
| 	err := json.Unmarshal([]byte(toString(call.Argument(0))), &root)
 | |
| 	if err != nil {
 | |
| 		panic(newSyntaxError(err.Error()))
 | |
| 	}
 | |
| 	value, exists := builtinJSON_parseWalk(ctx, root)
 | |
| 	if !exists {
 | |
| 		value = UndefinedValue()
 | |
| 	}
 | |
| 	if revive {
 | |
| 		root := ctx.call.runtime.newObject()
 | |
| 		root.put("", value, false)
 | |
| 		return builtinJSON_reviveWalk(ctx, root, "")
 | |
| 	}
 | |
| 	return value
 | |
| }
 | |
| 
 | |
| func builtinJSON_reviveWalk(ctx _builtinJSON_parseContext, holder *_object, name string) Value {
 | |
| 	value := holder.get(name)
 | |
| 	if object := value._object(); object != nil {
 | |
| 		if isArray(object) {
 | |
| 			length := int64(objectLength(object))
 | |
| 			for index := int64(0); index < length; index += 1 {
 | |
| 				name := arrayIndexToString(index)
 | |
| 				value := builtinJSON_reviveWalk(ctx, object, name)
 | |
| 				if value.IsUndefined() {
 | |
| 					object.delete(name, false)
 | |
| 				} else {
 | |
| 					object.defineProperty(name, value, 0111, false)
 | |
| 				}
 | |
| 			}
 | |
| 		} else {
 | |
| 			object.enumerate(false, func(name string) bool {
 | |
| 				value := builtinJSON_reviveWalk(ctx, object, name)
 | |
| 				if value.IsUndefined() {
 | |
| 					object.delete(name, false)
 | |
| 				} else {
 | |
| 					object.defineProperty(name, value, 0111, false)
 | |
| 				}
 | |
| 				return true
 | |
| 			})
 | |
| 		}
 | |
| 	}
 | |
| 	return ctx.reviver.call(toValue_object(holder), name, value)
 | |
| }
 | |
| 
 | |
| func builtinJSON_parseWalk(ctx _builtinJSON_parseContext, rawValue interface{}) (Value, bool) {
 | |
| 	switch value := rawValue.(type) {
 | |
| 	case nil:
 | |
| 		return NullValue(), true
 | |
| 	case bool:
 | |
| 		return toValue_bool(value), true
 | |
| 	case string:
 | |
| 		return toValue_string(value), true
 | |
| 	case float64:
 | |
| 		return toValue_float64(value), true
 | |
| 	case []interface{}:
 | |
| 		arrayValue := make([]Value, len(value))
 | |
| 		for index, rawValue := range value {
 | |
| 			if value, exists := builtinJSON_parseWalk(ctx, rawValue); exists {
 | |
| 				arrayValue[index] = value
 | |
| 			}
 | |
| 		}
 | |
| 		return toValue_object(ctx.call.runtime.newArrayOf(arrayValue)), true
 | |
| 	case map[string]interface{}:
 | |
| 		object := ctx.call.runtime.newObject()
 | |
| 		for name, rawValue := range value {
 | |
| 			if value, exists := builtinJSON_parseWalk(ctx, rawValue); exists {
 | |
| 				object.put(name, value, false)
 | |
| 			}
 | |
| 		}
 | |
| 		return toValue_object(object), true
 | |
| 	}
 | |
| 	return Value{}, false
 | |
| }
 | |
| 
 | |
| type _builtinJSON_stringifyContext struct {
 | |
| 	call             FunctionCall
 | |
| 	stack            []*_object
 | |
| 	propertyList     []string
 | |
| 	replacerFunction *Value
 | |
| 	gap              string
 | |
| }
 | |
| 
 | |
| func builtinJSON_stringify(call FunctionCall) Value {
 | |
| 	ctx := _builtinJSON_stringifyContext{
 | |
| 		call:  call,
 | |
| 		stack: []*_object{nil},
 | |
| 	}
 | |
| 	replacer := call.Argument(1)._object()
 | |
| 	if replacer != nil {
 | |
| 		if isArray(replacer) {
 | |
| 			length := objectLength(replacer)
 | |
| 			seen := map[string]bool{}
 | |
| 			propertyList := make([]string, length)
 | |
| 			length = 0
 | |
| 			for index, _ := range propertyList {
 | |
| 				value := replacer.get(arrayIndexToString(int64(index)))
 | |
| 				switch value._valueType {
 | |
| 				case valueObject:
 | |
| 					switch value.value.(*_object).class {
 | |
| 					case "String":
 | |
| 					case "Number":
 | |
| 					default:
 | |
| 						continue
 | |
| 					}
 | |
| 				case valueString:
 | |
| 				case valueNumber:
 | |
| 				default:
 | |
| 					continue
 | |
| 				}
 | |
| 				name := toString(value)
 | |
| 				if seen[name] {
 | |
| 					continue
 | |
| 				}
 | |
| 				seen[name] = true
 | |
| 				length += 1
 | |
| 				propertyList[index] = name
 | |
| 			}
 | |
| 			ctx.propertyList = propertyList[0:length]
 | |
| 		} else if replacer.class == "Function" {
 | |
| 			value := toValue_object(replacer)
 | |
| 			ctx.replacerFunction = &value
 | |
| 		}
 | |
| 	}
 | |
| 	if spaceValue, exists := call.getArgument(2); exists {
 | |
| 		if spaceValue._valueType == valueObject {
 | |
| 			switch spaceValue.value.(*_object).class {
 | |
| 			case "String":
 | |
| 				spaceValue = toValue_string(toString(spaceValue))
 | |
| 			case "Number":
 | |
| 				spaceValue = toNumber(spaceValue)
 | |
| 			}
 | |
| 		}
 | |
| 		switch spaceValue._valueType {
 | |
| 		case valueString:
 | |
| 			value := toString(spaceValue)
 | |
| 			if len(value) > 10 {
 | |
| 				ctx.gap = value[0:10]
 | |
| 			} else {
 | |
| 				ctx.gap = value
 | |
| 			}
 | |
| 		case valueNumber:
 | |
| 			value := toInteger(spaceValue).value
 | |
| 			if value > 10 {
 | |
| 				value = 10
 | |
| 			} else if value < 0 {
 | |
| 				value = 0
 | |
| 			}
 | |
| 			ctx.gap = strings.Repeat(" ", int(value))
 | |
| 		}
 | |
| 	}
 | |
| 	holder := call.runtime.newObject()
 | |
| 	holder.put("", call.Argument(0), false)
 | |
| 	value, exists := builtinJSON_stringifyWalk(ctx, "", holder)
 | |
| 	if !exists {
 | |
| 		return UndefinedValue()
 | |
| 	}
 | |
| 	valueJSON, err := json.Marshal(value)
 | |
| 	if err != nil {
 | |
| 		panic(newTypeError(err.Error()))
 | |
| 	}
 | |
| 	if ctx.gap != "" {
 | |
| 		valueJSON1 := bytes.Buffer{}
 | |
| 		json.Indent(&valueJSON1, valueJSON, "", ctx.gap)
 | |
| 		valueJSON = valueJSON1.Bytes()
 | |
| 	}
 | |
| 	return toValue_string(string(valueJSON))
 | |
| }
 | |
| 
 | |
| func builtinJSON_stringifyWalk(ctx _builtinJSON_stringifyContext, key string, holder *_object) (interface{}, bool) {
 | |
| 	value := holder.get(key)
 | |
| 
 | |
| 	if value.IsObject() {
 | |
| 		object := value._object()
 | |
| 		if toJSON := object.get("toJSON"); toJSON.IsFunction() {
 | |
| 			value = toJSON.call(value, key)
 | |
| 		} else {
 | |
| 			// If the object is a GoStruct or something that implements json.Marshaler
 | |
| 			if object.objectClass.marshalJSON != nil {
 | |
| 				marshaler := object.objectClass.marshalJSON(object)
 | |
| 				if marshaler != nil {
 | |
| 					return marshaler, true
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if ctx.replacerFunction != nil {
 | |
| 		value = (*ctx.replacerFunction).call(toValue_object(holder), key, value)
 | |
| 	}
 | |
| 
 | |
| 	if value._valueType == valueObject {
 | |
| 		switch value.value.(*_object).class {
 | |
| 		case "Boolean":
 | |
| 			value = value._object().value.(Value)
 | |
| 		case "String":
 | |
| 			value = toValue_string(toString(value))
 | |
| 		case "Number":
 | |
| 			value = toNumber(value)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	switch value._valueType {
 | |
| 	case valueBoolean:
 | |
| 		return toBoolean(value), true
 | |
| 	case valueString:
 | |
| 		return toString(value), true
 | |
| 	case valueNumber:
 | |
| 		value := toFloat(value)
 | |
| 		if math.IsNaN(value) || math.IsInf(value, 0) {
 | |
| 			return nil, true
 | |
| 		}
 | |
| 		return value, true
 | |
| 	case valueNull:
 | |
| 		return nil, true
 | |
| 	case valueObject:
 | |
| 		holder := value._object()
 | |
| 		if value := value._object(); nil != value {
 | |
| 			for _, object := range ctx.stack {
 | |
| 				if holder == object {
 | |
| 					panic(newTypeError("Converting circular structure to JSON"))
 | |
| 				}
 | |
| 			}
 | |
| 			ctx.stack = append(ctx.stack, value)
 | |
| 			defer func() { ctx.stack = ctx.stack[:len(ctx.stack)-1] }()
 | |
| 		}
 | |
| 		if isArray(holder) {
 | |
| 			length := holder.get("length").value.(uint32)
 | |
| 			array := make([]interface{}, length)
 | |
| 			for index, _ := range array {
 | |
| 				name := arrayIndexToString(int64(index))
 | |
| 				value, _ := builtinJSON_stringifyWalk(ctx, name, holder)
 | |
| 				array[index] = value
 | |
| 			}
 | |
| 			return array, true
 | |
| 		} else if holder.class != "Function" {
 | |
| 			object := map[string]interface{}{}
 | |
| 			if ctx.propertyList != nil {
 | |
| 				for _, name := range ctx.propertyList {
 | |
| 					value, exists := builtinJSON_stringifyWalk(ctx, name, holder)
 | |
| 					if exists {
 | |
| 						object[name] = value
 | |
| 					}
 | |
| 				}
 | |
| 			} else {
 | |
| 				// Go maps are without order, so this doesn't conform to the ECMA ordering
 | |
| 				// standard, but oh well...
 | |
| 				holder.enumerate(false, func(name string) bool {
 | |
| 					value, exists := builtinJSON_stringifyWalk(ctx, name, holder)
 | |
| 					if exists {
 | |
| 						object[name] = value
 | |
| 					}
 | |
| 					return true
 | |
| 				})
 | |
| 			}
 | |
| 			return object, true
 | |
| 		}
 | |
| 	}
 | |
| 	return nil, false
 | |
| }
 |