001/* 002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others 003 * 004 * This program and the accompanying materials are made available under the 005 * terms of the Eclipse Public License 2.0 which is available at 006 * http://www.eclipse.org/legal/epl-2.0. 007 * 008 * SPDX-License-Identifier: EPL-2.0 009 */ 010 011package gololang; 012 013import java.lang.invoke.MethodHandle; 014import java.lang.invoke.MethodHandles; 015import java.lang.invoke.MethodType; 016import java.util.HashMap; 017import java.util.Map; 018import java.util.Set; 019import java.util.List; 020import java.util.LinkedList; 021import java.util.Objects; 022 023import static java.lang.System.arraycopy; 024import static java.lang.invoke.MethodType.genericMethodType; 025import static java.lang.invoke.MethodType.methodType; 026import static gololang.Predefined.isClosure; 027 028/** 029 * A dynamic object is an object whose properties can be dynamically added, changed and removed. Properties can be any 030 * object value or a function reference. 031 * <p> 032 * The methods <code>plug</code> and <code>propertyMissing</code> are left undocumented. They are being used 033 * by the Golo runtime to dispatch method invocations on dynamic objects. 034 */ 035public final class DynamicObject { 036 037 private final Object kind; 038 private final HashMap<String, Object> properties = new HashMap<>(); 039 private boolean frozen = false; 040 041 042 public DynamicObject() { 043 this("DynamicObject"); 044 } 045 046 public DynamicObject(Object kind) { 047 this.kind = kind; 048 } 049 050 public boolean hasKind(Object k) { 051 return Objects.equals(kind, k); 052 } 053 054 public boolean sameKind(DynamicObject other) { 055 return Objects.equals(kind, other.kind); 056 } 057 058 @Override 059 public String toString() { 060 List<String> props = new LinkedList<>(); 061 for (Map.Entry<String, Object> prop : properties.entrySet()) { 062 if (!isClosure(prop.getValue())) { 063 props.add(String.format("%s=%s", prop.getKey(), prop.getValue().toString())); 064 } 065 } 066 return String.format("%s{%s}", kind, String.join(", ", props)); 067 } 068 069 /** 070 * Defines a property. 071 * 072 * @param name the property name. 073 * @param value the property value. 074 * @return the same dynamic object. 075 * @throws IllegalStateException if the dynamic object is frozen. 076 */ 077 public DynamicObject define(String name, Object value) { 078 frozenMutationCheck(); 079 properties.put(name, value); 080 return this; 081 } 082 083 /** 084 * @return a view of all properties. 085 */ 086 public Set<Map.Entry<String, Object>> properties() { 087 return properties.entrySet(); 088 } 089 090 /** 091 * @param name the property name. 092 * @return the property value. 093 */ 094 public Object get(String name) { 095 return properties.get(name); 096 } 097 098 /** 099 * Removes a property. 100 * 101 * @param name the property name. 102 * @return the same dynamic object. 103 */ 104 public DynamicObject undefine(String name) { 105 frozenMutationCheck(); 106 properties.remove(name); 107 return this; 108 } 109 110 /** 111 * @return a new dynamic object whose properties point to the same objects. 112 */ 113 public DynamicObject copy() { 114 DynamicObject copy = new DynamicObject(this.kind); 115 for (Map.Entry<String, Object> entry : properties.entrySet()) { 116 copy.properties.put(entry.getKey(), entry.getValue()); 117 } 118 return copy; 119 } 120 121 /** 122 * Mixes all properties from another dynamic object into this one, overwriting existing properties. 123 * 124 * @param other the dynamic object to mix the properties from. 125 * @return the same dynamic object. 126 */ 127 public DynamicObject mixin(DynamicObject other) { 128 frozenMutationCheck(); 129 for (Map.Entry<String, Object> entry : other.properties.entrySet()) { 130 properties.put(entry.getKey(), entry.getValue()); 131 } 132 return this; 133 } 134 135 /** 136 * Freezes a dynamic object, meaning that its properties cannot be added, updated and removed anymore. 137 * 138 * @return the same dynamic object. 139 */ 140 public DynamicObject freeze() { 141 this.frozen = true; 142 return this; 143 } 144 145 /** 146 * Tells whether the dynamic object is frozen or not. 147 * 148 * @return {@code true} if frozen, {@code false} otherwise. 149 */ 150 public boolean isFrozen() { 151 return this.frozen; 152 } 153 154 /** 155 * Dispatch dynamic object "methods". The receiver dynamic object is expected to be the first element of {@code args}. 156 * 157 * @param property the method property in the dynamic object. 158 * @param args the arguments. 159 * @return the return value. 160 * @throws Throwable in case everything is wrong. 161 */ 162 public static Object dispatchCall(String property, Object... args) throws Throwable { 163 DynamicObject obj = (DynamicObject) args[0]; 164 Object value = obj.properties.get(property); 165 if (value != null) { 166 if (value instanceof FunctionReference) { 167 FunctionReference funRef = (FunctionReference) value; 168 if (funRef.isVarargsCollector() && args[args.length - 1] instanceof Object[]) { 169 return funRef.spread(args); 170 } 171 return funRef.invoke(args); 172 } else { 173 throw new UnsupportedOperationException("There is no dynamic object method defined for " + property); 174 } 175 } 176 if (obj.hasFallback()) { 177 FunctionReference handle = (FunctionReference) obj.properties.get("fallback"); 178 Object[] fallback_args = new Object[args.length + 1]; 179 fallback_args[0] = obj; 180 fallback_args[1] = property; 181 arraycopy(args, 1, fallback_args, 2, args.length - 1); 182 return handle.invoke(fallback_args); 183 } 184 throw new UnsupportedOperationException("There is neither a dynamic object method defined for " + property + " nor a 'fallback' method"); 185 } 186 187 /** 188 * Dispatches getter-style dynamic object methods, i.e., methods with a receiver and no argument. 189 * 190 * @param property the method property in the dynamic object. 191 * @param object the receiver object. 192 * @return the return value. 193 * @throws Throwable in case everything is wrong. 194 */ 195 public static Object dispatchGetterStyle(String property, DynamicObject object) throws Throwable { 196 Object value = object.get(property); 197 if (value != null || object.properties.containsKey(property)) { 198 if (value instanceof FunctionReference) { 199 FunctionReference funRef = (FunctionReference) value; 200 if (funRef.acceptArity(1)) { 201 return funRef.invoke(object); 202 } 203 } 204 return value; 205 } 206 if (object.hasFallback()) { 207 FunctionReference funRef = (FunctionReference) object.properties.get("fallback"); 208 return funRef.invoke(object, property); 209 } 210 return null; 211 } 212 213 /** 214 * Dispatches setter-style dynamic object methods, i.e., methods with a receiver and exactly 1 argument. 215 * 216 * @param property the method property in the dynamic object. 217 * @param object the receiver object. 218 * @param arg the arguments. 219 * @return the return value. 220 * @throws Throwable in case everything is wrong. 221 */ 222 public static Object dispatchSetterStyle(String property, DynamicObject object, Object arg) throws Throwable { 223 Object value = object.get(property); 224 if (value != null || object.properties.containsKey(property)) { 225 if (value instanceof FunctionReference) { 226 FunctionReference funRef = (FunctionReference) value; 227 if (funRef.arity() == 2) { 228 if (funRef.isVarargsCollector() && arg instanceof Object[]) { 229 return funRef.handle().invokeExact((Object) object, (Object[]) arg); 230 } 231 return funRef.invoke(object, arg); 232 } 233 } 234 } 235 // NOTE: should we try the fallback method here ? 236 return object.define(property, arg); 237 } 238 239 /** 240 * Dispatches on another dynamic object (fallback helper). 241 * 242 * @param deleguee the object to delegate to. 243 * @param receiver the receiver object. 244 * @param property the method property in the dynamic object. 245 * @param args the arguments. 246 * @return the return value. 247 * @throws Throwable in case everything is wrong. 248 */ 249 public static Object dispatchDelegate(DynamicObject deleguee, DynamicObject receiver, String property, Object... args) throws Throwable { 250 return deleguee 251 .invoker(property, genericMethodType(args.length + 1)) 252 .bindTo(deleguee) 253 .invokeWithArguments(args); 254 } 255 256 /** 257 * Creates a function suitable for the {@code fallback} property delegating to the given dynamic object. 258 * 259 * Example: 260 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 261 * let d = DynamicObject(): name("Zaphod") 262 * let o = DynamicObject(): fallback(delegate(d)) 263 * </code></pre> 264 * 265 * @param deleguee the object to delegate to. 266 * @return a function delegating to {@code deleguee} 267 */ 268 public static FunctionReference delegate(DynamicObject deleguee) { 269 return new FunctionReference( 270 DISPATCH_DELEGATE.bindTo(deleguee).asVarargsCollector(Object[].class), //.asType(genericMethodType(2, true)), 271 new String[]{"this", "name", "args"}); 272 } 273 274 /** 275 * Gives an invoker method handle for a given property. 276 * <p> 277 * While this method may be useful in itself, it is mostly relevant for the Golo runtime internals so as 278 * to allow calling "methods" on dynamic objects, as in: 279 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 280 * # obj is some dynamic object... 281 * obj: foo("bar") 282 * println(foo: bar()) 283 * 284 * obj: define("plop", |this| -> "Plop!") 285 * println(obj: plop()) 286 * </code></pre> 287 * 288 * @param property the name of a property. 289 * @param type the expected invoker type with at least one parameter (the dynamic object as a receiver). 290 * @return a method handle. 291 */ 292 public MethodHandle invoker(String property, MethodType type) { 293 switch (type.parameterCount()) { 294 case 0: 295 throw new IllegalArgumentException("A dynamic object invoker type needs at least 1 argument (the receiver)"); 296 case 1: 297 return DISPATCH_GET.bindTo(property).asType(genericMethodType(1)); 298 case 2: 299 return DISPATCH_SET.bindTo(property).asType(genericMethodType(2)); 300 default: 301 return DISPATCH_CALL.bindTo(property).asCollector(Object[].class, type.parameterCount()); 302 } 303 } 304 305 /** 306 * Verify if a method is defined for the dynamic object. 307 * 308 * @param method the method name. 309 * @return {@code true} if method is defined, {@code false} otherwise. 310 */ 311 public boolean hasMethod(String method) { 312 Object obj = properties.get(method); 313 if (obj != null) { 314 return isClosure(obj); 315 } 316 return false; 317 } 318 319 /** 320 * Let the user define a fallback behavior. 321 * 322 * @param value the fallback value 323 * @return the current object 324 */ 325 public DynamicObject fallback(Object value) { 326 return define("fallback", value); 327 } 328 329 /** 330 * Verify a fallback property exists. 331 * 332 * @return {@code true} if a fallback behavior is defined, {@code false} otherwise. 333 */ 334 private boolean hasFallback() { 335 return properties.containsKey("fallback"); 336 } 337 338 public static final MethodHandle DISPATCH_CALL; 339 public static final MethodHandle DISPATCH_GET; 340 public static final MethodHandle DISPATCH_SET; 341 public static final MethodHandle DISPATCH_DELEGATE; 342 343 static { 344 MethodHandles.Lookup lookup = MethodHandles.lookup(); 345 try { 346 DISPATCH_DELEGATE = lookup.findStatic(DynamicObject.class, "dispatchDelegate", 347 methodType(Object.class, DynamicObject.class, DynamicObject.class, String.class, Object[].class)); 348 DISPATCH_CALL = lookup.findStatic(DynamicObject.class, "dispatchCall", methodType(Object.class, String.class, Object[].class)); 349 DISPATCH_GET = lookup.findStatic(DynamicObject.class, "dispatchGetterStyle", methodType(Object.class, String.class, DynamicObject.class)); 350 DISPATCH_SET = lookup.findStatic(DynamicObject.class, "dispatchSetterStyle", methodType(Object.class, String.class, DynamicObject.class, Object.class)); 351 } catch (NoSuchMethodException | IllegalAccessException e) { 352 e.printStackTrace(); 353 throw new Error("Could not bootstrap the required method handles"); 354 } 355 } 356 357 private void frozenMutationCheck() { 358 if (this.frozen) { 359 throw new IllegalStateException("the object is frozen"); 360 } 361 } 362}