mirror of
				https://github.com/robertkrimen/otto
				synced 2025-10-19 19:55:30 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			399 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			399 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package otto
 | |
| 
 | |
| type _objectClass struct {
 | |
| 	getOwnProperty    func(*_object, string) *_property
 | |
| 	getProperty       func(*_object, string) *_property
 | |
| 	get               func(*_object, string) Value
 | |
| 	canPut            func(*_object, string) bool
 | |
| 	put               func(*_object, string, Value, bool)
 | |
| 	hasProperty       func(*_object, string) bool
 | |
| 	hasOwnProperty    func(*_object, string) bool
 | |
| 	defineOwnProperty func(*_object, string, _property, bool) bool
 | |
| 	delete            func(*_object, string, bool) bool
 | |
| 	enumerate         func(*_object, bool, func(string) bool)
 | |
| 	clone             func(*_object, *_object, *_clone) *_object
 | |
| }
 | |
| 
 | |
| func objectEnumerate(self *_object, all bool, each func(string) bool) {
 | |
| 	for _, name := range self.propertyOrder {
 | |
| 		if all || self.property[name].enumerable() {
 | |
| 			if !each(name) {
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| var (
 | |
| 	_classObject,
 | |
| 	_classArray,
 | |
| 	_classString,
 | |
| 	_classArguments,
 | |
| 	_classGoStruct,
 | |
| 	_classGoMap,
 | |
| 	_classGoArray,
 | |
| 	_classGoSlice,
 | |
| 	_ *_objectClass
 | |
| )
 | |
| 
 | |
| func init() {
 | |
| 	_classObject = &_objectClass{
 | |
| 		objectGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		objectDefineOwnProperty,
 | |
| 		objectDelete,
 | |
| 		objectEnumerate,
 | |
| 		objectClone,
 | |
| 	}
 | |
| 
 | |
| 	_classArray = &_objectClass{
 | |
| 		objectGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		arrayDefineOwnProperty,
 | |
| 		objectDelete,
 | |
| 		objectEnumerate,
 | |
| 		objectClone,
 | |
| 	}
 | |
| 
 | |
| 	_classString = &_objectClass{
 | |
| 		stringGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		objectDefineOwnProperty,
 | |
| 		objectDelete,
 | |
| 		stringEnumerate,
 | |
| 		objectClone,
 | |
| 		//stringClone,
 | |
| 	}
 | |
| 
 | |
| 	_classArguments = &_objectClass{
 | |
| 		argumentsGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		argumentsGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		argumentsDefineOwnProperty,
 | |
| 		argumentsDelete,
 | |
| 		//argumentsEnumerate,
 | |
| 		objectEnumerate,
 | |
| 		//argumentsClone
 | |
| 		objectClone,
 | |
| 	}
 | |
| 
 | |
| 	_classGoStruct = &_objectClass{
 | |
| 		goStructGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		goStructCanPut,
 | |
| 		goStructPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		objectDefineOwnProperty,
 | |
| 		objectDelete,
 | |
| 		goStructEnumerate,
 | |
| 		objectClone,
 | |
| 	}
 | |
| 
 | |
| 	_classGoMap = &_objectClass{
 | |
| 		goMapGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		goMapDefineOwnProperty,
 | |
| 		goMapDelete,
 | |
| 		goMapEnumerate,
 | |
| 		objectClone,
 | |
| 	}
 | |
| 
 | |
| 	_classGoArray = &_objectClass{
 | |
| 		goArrayGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		goArrayDefineOwnProperty,
 | |
| 		goArrayDelete,
 | |
| 		goArrayEnumerate,
 | |
| 		objectClone,
 | |
| 	}
 | |
| 
 | |
| 	_classGoSlice = &_objectClass{
 | |
| 		goSliceGetOwnProperty,
 | |
| 		objectGetProperty,
 | |
| 		objectGet,
 | |
| 		objectCanPut,
 | |
| 		objectPut,
 | |
| 		objectHasProperty,
 | |
| 		objectHasOwnProperty,
 | |
| 		goSliceDefineOwnProperty,
 | |
| 		goSliceDelete,
 | |
| 		goSliceEnumerate,
 | |
| 		objectClone,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Allons-y
 | |
| 
 | |
| // 8.12.1
 | |
| func objectGetOwnProperty(self *_object, name string) *_property {
 | |
| 	// Return a _copy_ of the property
 | |
| 	property, exists := self._read(name)
 | |
| 	if !exists {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return &property
 | |
| }
 | |
| 
 | |
| // 8.12.2
 | |
| func objectGetProperty(self *_object, name string) *_property {
 | |
| 	property := self.getOwnProperty(name)
 | |
| 	if property != nil {
 | |
| 		return property
 | |
| 	}
 | |
| 	if self.prototype != nil {
 | |
| 		return self.prototype.getProperty(name)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // 8.12.3
 | |
| func objectGet(self *_object, name string) Value {
 | |
| 	property := self.getProperty(name)
 | |
| 	if property != nil {
 | |
| 		return property.get(self)
 | |
| 	}
 | |
| 	return UndefinedValue()
 | |
| }
 | |
| 
 | |
| // 8.12.4
 | |
| func objectCanPut(self *_object, name string) bool {
 | |
| 
 | |
| 	property := self.getOwnProperty(name)
 | |
| 	if property != nil {
 | |
| 		switch propertyValue := property.value.(type) {
 | |
| 		case Value:
 | |
| 			return property.writable()
 | |
| 		case _propertyGetSet:
 | |
| 			return propertyValue[1] != nil
 | |
| 		default:
 | |
| 			panic(newTypeError())
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if self.prototype == nil {
 | |
| 		return self.extensible
 | |
| 	}
 | |
| 
 | |
| 	property = self.prototype.getOwnProperty(name)
 | |
| 	if property == nil {
 | |
| 		return self.extensible
 | |
| 	}
 | |
| 
 | |
| 	switch propertyValue := property.value.(type) {
 | |
| 	case Value:
 | |
| 		if !self.extensible {
 | |
| 			return false
 | |
| 		}
 | |
| 		return property.writable()
 | |
| 	case _propertyGetSet:
 | |
| 		return propertyValue[1] != nil
 | |
| 	default:
 | |
| 		panic(newTypeError())
 | |
| 	}
 | |
| 
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // 8.12.5
 | |
| func objectPut(self *_object, name string, value Value, throw bool) {
 | |
| 	if !self.canPut(name) {
 | |
| 		typeErrorResult(throw)
 | |
| 		return
 | |
| 	}
 | |
| 	// TODO This is not exacting to the specification
 | |
| 	// TODO Shortcut?
 | |
| 	property := self.getOwnProperty(name)
 | |
| 	if property == nil {
 | |
| 		self.defineProperty(name, value, 0111, throw)
 | |
| 	} else {
 | |
| 		switch propertyValue := property.value.(type) {
 | |
| 		case Value:
 | |
| 			property.value = value
 | |
| 			self.defineOwnProperty(name, *property, throw)
 | |
| 		case _propertyGetSet:
 | |
| 			if propertyValue[1] != nil {
 | |
| 				propertyValue[1].callSet(toValue(self), value)
 | |
| 				return
 | |
| 			}
 | |
| 			if throw {
 | |
| 				panic(newTypeError())
 | |
| 			}
 | |
| 		default:
 | |
| 			panic(newTypeError())
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // 8.12.6
 | |
| func objectHasProperty(self *_object, name string) bool {
 | |
| 	return self.getProperty(name) != nil
 | |
| }
 | |
| 
 | |
| func objectHasOwnProperty(self *_object, name string) bool {
 | |
| 	return self.getOwnProperty(name) != nil
 | |
| }
 | |
| 
 | |
| // 8.12.9
 | |
| func objectDefineOwnProperty(self *_object, name string, descriptor _property, throw bool) bool {
 | |
| 	property, exists := self._read(name)
 | |
| 	{
 | |
| 		if !exists {
 | |
| 			if !self.extensible {
 | |
| 				goto Reject
 | |
| 			}
 | |
| 			self._write(name, descriptor.value, descriptor.mode)
 | |
| 			return true
 | |
| 		}
 | |
| 		if descriptor.isEmpty() {
 | |
| 			return true
 | |
| 		}
 | |
| 
 | |
| 		// TODO Per 8.12.9.6 - We should shortcut here (returning true) if
 | |
| 		// the current and new (define) properties are the same
 | |
| 
 | |
| 		configurable := property.configurable()
 | |
| 		if !configurable {
 | |
| 			if descriptor.configurable() {
 | |
| 				goto Reject
 | |
| 			}
 | |
| 			// Test that, if enumerable is set on the property descriptor, then it should
 | |
| 			// be the same as the existing property
 | |
| 			if descriptor.enumerateSet() && descriptor.enumerable() != property.enumerable() {
 | |
| 				goto Reject
 | |
| 			}
 | |
| 		}
 | |
| 		value, isDataDescriptor := property.value.(Value)
 | |
| 		getSet, _ := property.value.(_propertyGetSet)
 | |
| 		if descriptor.isGenericDescriptor() {
 | |
| 			// GenericDescriptor
 | |
| 		} else if isDataDescriptor != descriptor.isDataDescriptor() {
 | |
| 			// FIXME This branch is non-functioning
 | |
| 			panic("isDataDescriptor != descriptor.isDataDescriptor")
 | |
| 			var interface_ interface{}
 | |
| 			if isDataDescriptor {
 | |
| 				property.writeOff()
 | |
| 				property.value = interface_
 | |
| 			} else {
 | |
| 				property.writeOn()
 | |
| 				property.value = interface_
 | |
| 			}
 | |
| 		} else if isDataDescriptor && descriptor.isDataDescriptor() {
 | |
| 			if !configurable {
 | |
| 				if !property.writable() && descriptor.writable() {
 | |
| 					goto Reject
 | |
| 				}
 | |
| 				if !property.writable() {
 | |
| 					if descriptor.value != nil && !sameValue(value, descriptor.value.(Value)) {
 | |
| 						goto Reject
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		} else {
 | |
| 			if !configurable {
 | |
| 				defineGetSet, _ := descriptor.value.(_propertyGetSet)
 | |
| 				if getSet[0] != defineGetSet[0] || getSet[1] != defineGetSet[1] {
 | |
| 					goto Reject
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		{
 | |
| 			// This section will preserve attributes of
 | |
| 			// the original property, if necessary
 | |
| 			value1 := descriptor.value
 | |
| 			if value1 == nil {
 | |
| 				value1 = property.value
 | |
| 			}
 | |
| 			mode1 := descriptor.mode
 | |
| 			if mode1&0222 != 0 {
 | |
| 				// TODO Factor this out into somewhere testable
 | |
| 				// (Maybe put into switch ...)
 | |
| 				mode0 := property.mode
 | |
| 				if mode1&0200 != 0 {
 | |
| 					mode1 |= (mode0 & 0100)
 | |
| 				}
 | |
| 				if mode1&020 != 0 {
 | |
| 					mode1 |= (mode0 & 010)
 | |
| 				}
 | |
| 				if mode1&02 != 0 {
 | |
| 					mode1 |= (mode0 & 01)
 | |
| 				}
 | |
| 				mode1 &= 0111
 | |
| 			}
 | |
| 			self._write(name, value1, mode1)
 | |
| 		}
 | |
| 		return true
 | |
| 	}
 | |
| Reject:
 | |
| 	if throw {
 | |
| 		panic(newTypeError())
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| func objectDelete(self *_object, name string, throw bool) bool {
 | |
| 	property_ := self.getOwnProperty(name)
 | |
| 	if property_ == nil {
 | |
| 		return true
 | |
| 	}
 | |
| 	if property_.configurable() {
 | |
| 		self._delete(name)
 | |
| 		return true
 | |
| 	}
 | |
| 	return typeErrorResult(throw)
 | |
| }
 | |
| 
 | |
| func objectClone(self0 *_object, self1 *_object, clone *_clone) *_object {
 | |
| 	*self1 = *self0
 | |
| 
 | |
| 	self1.runtime = clone.runtime
 | |
| 	if self1.prototype != nil {
 | |
| 		self1.prototype = clone.object(self0.prototype)
 | |
| 	}
 | |
| 	self1.property = make(map[string]_property, len(self0.property))
 | |
| 	self1.propertyOrder = make([]string, len(self0.propertyOrder))
 | |
| 	copy(self1.propertyOrder, self0.propertyOrder)
 | |
| 	for index, property := range self0.property {
 | |
| 		self1.property[index] = clone.property(property)
 | |
| 	}
 | |
| 
 | |
| 	switch value := self0.value.(type) {
 | |
| 	case _functionObject:
 | |
| 		self1.value = value.clone(clone)
 | |
| 	case _argumentsObject:
 | |
| 		self1.value = value.clone(clone)
 | |
| 	}
 | |
| 
 | |
| 	return self1
 | |
| }
 | 
