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.concurrent.async; 012 013import java.util.HashSet; 014 015/** 016 * A promise object is used to abstract over possibly asynchronous computations. 017 * 018 * You should consult the "golodoc" of the {@code gololang.Async} module. 019 * 020 * @see gololang.concurrent.async.Future 021 * @see gololang.concurrent.async.AssignedFuture 022 */ 023public final class Promise { 024 025 private volatile boolean resolved = false; 026 private volatile Object value; 027 028 private final Object lock = new Object(); 029 private final HashSet<Future.Observer> setObservers = new HashSet<>(); 030 private final HashSet<Future.Observer> failObservers = new HashSet<>(); 031 032 /** 033 * Checks whether the promise has been resolved. 034 * 035 * @return {@code true} if it has been resolved, {@code false} otherwise. 036 */ 037 public boolean isResolved() { 038 return resolved; 039 } 040 041 /** 042 * Checks whether the promise has failed. 043 * 044 * @return {@code true} if it has been resolved and failed, {@code false} otherwise. 045 */ 046 public boolean isFailed() { 047 return value instanceof Throwable; 048 } 049 050 /** 051 * Non-blocking get. 052 * 053 * @return the promise value, which may be {@code null} if it has not been resolved yet. 054 */ 055 public Object get() { 056 return value; 057 } 058 059 /** 060 * Blocking get, waiting until the promise is resolved. 061 * 062 * @return the promise value. 063 * @throws InterruptedException if the current thread gets interrupted. 064 */ 065 public Object blockingGet() throws InterruptedException { 066 synchronized (lock) { 067 while (!resolved) { 068 lock.wait(); 069 } 070 return value; 071 } 072 } 073 074 /** 075 * Sets the promise value. This has no effect if the promise has already been resolved. 076 * 077 * @param value the value. 078 * @return this promise. 079 */ 080 public Promise set(Object value) { 081 if (resolved) { 082 return this; 083 } 084 synchronized (lock) { 085 if (!resolved) { 086 this.value = value; 087 this.resolved = true; 088 lock.notifyAll(); 089 } 090 } 091 HashSet<Future.Observer> observers = isFailed() ? failObservers : setObservers; 092 for (Future.Observer observer : observers) { 093 observer.apply(value); 094 } 095 return this; 096 } 097 098 /** 099 * Fails the promise. This has no effect if the promise has already been resolved. 100 * 101 * @param throwable the failure. 102 * @return this promise. 103 */ 104 public Promise fail(Throwable throwable) { 105 return set(throwable); 106 } 107 108 /** 109 * Creates a new future to observe the eventual resolution of this promise. 110 * 111 * @return a new future object. 112 */ 113 public Future future() { 114 return new Future() { 115 @Override 116 public Object get() { 117 return Promise.this.get(); 118 } 119 120 @Override 121 public Object blockingGet() throws InterruptedException { 122 return Promise.this.blockingGet(); 123 } 124 125 @Override 126 public boolean isResolved() { 127 return Promise.this.isResolved(); 128 } 129 130 @Override 131 public boolean isFailed() { 132 return Promise.this.isFailed(); 133 } 134 135 @Override 136 public Future onSet(Observer observer) { 137 synchronized (lock) { 138 if (resolved && !Promise.this.isFailed()) { 139 observer.apply(value); 140 } else { 141 setObservers.add(observer); 142 } 143 } 144 return this; 145 } 146 147 @Override 148 public Future onFail(Observer observer) { 149 synchronized (lock) { 150 if (resolved && Promise.this.isFailed()) { 151 observer.apply(value); 152 } else { 153 failObservers.add(observer); 154 } 155 } 156 return this; 157 } 158 }; 159 } 160 161 @Override 162 public String toString() { 163 return String.format("Promise{resolved=%s, value=%s}", resolved, value); 164 } 165}