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 013/** 014 * A dynamic variable has the semantics of an inheritable thread-local reference. 015 * This class is modeled after the eponymous class from the Scala standard library. 016 * 017 * @see java.lang.InheritableThreadLocal 018 */ 019public final class DynamicVariable { 020 021 private final InheritableThreadLocal<Object> threadLocal; 022 023 /** 024 * Creates a new dynamic variable with an initial value. 025 * 026 * @param init the initial value. 027 */ 028 public DynamicVariable(final Object init) { 029 super(); 030 threadLocal = new InheritableThreadLocal<Object>() { 031 @Override 032 protected Object initialValue() { 033 return init; 034 } 035 }; 036 } 037 038 /** 039 * Returns the thread-local value of the dynamic variable. 040 * 041 * @return the value. 042 */ 043 public Object value() { 044 return threadLocal.get(); 045 } 046 047 /** 048 * Changes the dynamic variable value. The new value is only visible from the calling thread, and will be seen by 049 * future child threads. 050 * 051 * @param value the new thread-local value. 052 * @return this dynamic variable. 053 */ 054 public DynamicVariable value(Object value) { 055 threadLocal.set(value); 056 return this; 057 } 058 059 /** 060 * Given a value, calls a function {@code func}. The previous value is put pack as the dynamic variable value 061 * once {@code func} has completed. 062 * 063 * @param value the value for the course of the execution of {@code func}. 064 * @param func a 0-arity function. 065 * @return the result of the call to {@code func}. 066 * @throws Throwable in case an exception occurs. 067 */ 068 public Object withValue(Object value, FunctionReference func) throws Throwable { 069 if (!func.acceptArity(0)) { 070 throw new IllegalArgumentException("withValue requires a function with no parameters"); 071 } 072 Object oldValue = value(); 073 this.value(value); 074 try { 075 return func.invoke(); 076 } finally { 077 this.value(oldValue); 078 } 079 } 080 081 @Override 082 public String toString() { 083 return String.format("DynamicVariable{value=%s}", value()); 084 } 085}