swim.system.lane.JoinValueLaneView Maven / Gradle / Ivy
// Copyright 2015-2024 Nstream, inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package swim.system.lane;
import java.util.AbstractCollection;
import java.util.AbstractMap;
import java.util.AbstractSet;
import java.util.Collection;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;
import swim.api.Lane;
import swim.api.Link;
import swim.api.SwimContext;
import swim.api.agent.AgentContext;
import swim.api.data.MapData;
import swim.api.downlink.ValueDownlink;
import swim.api.lane.JoinValueLane;
import swim.api.lane.function.DidDownlinkValue;
import swim.api.lane.function.WillDownlinkValue;
import swim.api.warp.function.DidCommand;
import swim.api.warp.function.DidEnter;
import swim.api.warp.function.DidLeave;
import swim.api.warp.function.DidUplink;
import swim.api.warp.function.WillCommand;
import swim.api.warp.function.WillEnter;
import swim.api.warp.function.WillLeave;
import swim.api.warp.function.WillUplink;
import swim.concurrent.Cont;
import swim.observable.function.DidClear;
import swim.observable.function.DidRemoveKey;
import swim.observable.function.DidUpdateKey;
import swim.observable.function.WillClear;
import swim.observable.function.WillRemoveKey;
import swim.observable.function.WillUpdateKey;
import swim.structure.Form;
import swim.structure.Value;
import swim.structure.collections.ValueEntryIterator;
import swim.structure.collections.ValueIterator;
import swim.system.LaneContext;
import swim.system.warp.WarpLaneView;
import swim.uri.Uri;
public class JoinValueLaneView extends WarpLaneView implements JoinValueLane {
protected final AgentContext agentContext;
protected Form keyForm;
protected Form valueForm;
protected JoinValueLaneModel laneBinding;
protected MapData dataView;
protected int flags;
JoinValueLaneView(AgentContext agentContext, Form keyForm, Form valueForm,
int flags, Object observers) {
super(observers);
this.agentContext = agentContext;
this.keyForm = keyForm;
this.valueForm = valueForm;
this.laneBinding = null;
this.dataView = null;
this.flags = flags;
}
public JoinValueLaneView(AgentContext agentContext, Form keyForm, Form valueForm) {
this(agentContext, keyForm, valueForm, JoinValueLaneView.RESIDENT, null);
}
@Override
public AgentContext agentContext() {
return this.agentContext;
}
@Override
public JoinValueLaneModel laneBinding() {
return this.laneBinding;
}
void setLaneBinding(JoinValueLaneModel laneBinding) {
this.laneBinding = laneBinding;
}
@Override
public JoinValueLaneModel createLaneBinding() {
return new JoinValueLaneModel(this.flags);
}
@Override
public final Form keyForm() {
return this.keyForm;
}
@Override
public JoinValueLaneView keyForm(Form keyForm) {
return new JoinValueLaneView(this.agentContext, keyForm, this.valueForm,
this.flags, this.typesafeObservers(this.observers));
}
@Override
public JoinValueLaneView keyClass(Class keyClass) {
return this.keyForm(Form.forClass(keyClass));
}
public void setKeyForm(Form keyForm) {
this.keyForm = keyForm;
}
@Override
public final Form valueForm() {
return this.valueForm;
}
@Override
public JoinValueLaneView valueForm(Form valueForm) {
return new JoinValueLaneView(this.agentContext, this.keyForm, valueForm,
this.flags, this.typesafeObservers(this.observers));
}
@Override
public JoinValueLaneView valueClass(Class valueClass) {
return this.valueForm(Form.forClass(valueClass));
}
public void setValueForm(Form valueForm) {
this.valueForm = valueForm;
}
protected Object typesafeObservers(Object observers) {
// TODO: filter out WillDownlinkValue, DidDownlinkValue, WillUpdateKey, DidUpdateKey,
// WillRemoveKey, DidRemoveKey, WillClear, DidClear
return observers;
}
public final boolean isResident() {
return (this.flags & JoinValueLaneView.RESIDENT) != 0;
}
@Override
public JoinValueLaneView isResident(boolean isResident) {
if (isResident) {
this.flags |= JoinValueLaneView.RESIDENT;
} else {
this.flags &= ~JoinValueLaneView.RESIDENT;
}
final JoinValueLaneModel laneBinding = this.laneBinding;
if (laneBinding != null) {
laneBinding.isResident(isResident);
}
return this;
}
void didSetResident(boolean isResident) {
if (isResident) {
this.flags |= JoinValueLaneView.RESIDENT;
} else {
this.flags &= ~JoinValueLaneView.RESIDENT;
}
}
public final boolean isTransient() {
return (this.flags & JoinValueLaneView.TRANSIENT) != 0;
}
@Override
public JoinValueLaneView isTransient(boolean isTransient) {
if (isTransient) {
this.flags |= JoinValueLaneView.TRANSIENT;
} else {
this.flags &= ~JoinValueLaneView.TRANSIENT;
}
final JoinValueLaneModel laneBinding = this.laneBinding;
if (laneBinding != null) {
laneBinding.isTransient(isTransient);
}
return this;
}
void didSetTransient(boolean isTransient) {
if (isTransient) {
this.flags |= JoinValueLaneView.TRANSIENT;
} else {
this.flags &= ~JoinValueLaneView.TRANSIENT;
}
}
@Override
protected void willLoad() {
this.dataView = this.laneBinding.data.keyForm(this.keyForm).valueForm(this.valueForm);
super.willLoad();
}
@Override
public void close() {
this.laneBinding.closeLaneView(this);
}
@Override
public JoinValueLaneView observe(Object observer) {
super.observe(observer);
return this;
}
@Override
public JoinValueLaneView unobserve(Object observer) {
super.unobserve(observer);
return this;
}
@Override
public JoinValueLaneView willDownlink(WillDownlinkValue willDownlink) {
return this.observe(willDownlink);
}
@Override
public JoinValueLaneView didDownlink(DidDownlinkValue didDownlink) {
return this.observe(didDownlink);
}
@Override
public JoinValueLaneView willUpdate(WillUpdateKey willUpdate) {
return this.observe(willUpdate);
}
@Override
public JoinValueLaneView didUpdate(DidUpdateKey didUpdate) {
return this.observe(didUpdate);
}
@Override
public JoinValueLaneView willRemove(WillRemoveKey willRemove) {
return this.observe(willRemove);
}
@Override
public JoinValueLaneView didRemove(DidRemoveKey didRemove) {
return this.observe(didRemove);
}
@Override
public JoinValueLaneView willClear(WillClear willClear) {
return this.observe(willClear);
}
@Override
public JoinValueLaneView didClear(DidClear didClear) {
return this.observe(didClear);
}
@Override
public JoinValueLaneView willCommand(WillCommand willCommand) {
return this.observe(willCommand);
}
@Override
public JoinValueLaneView didCommand(DidCommand didCommand) {
return this.observe(didCommand);
}
@Override
public JoinValueLaneView willUplink(WillUplink willUplink) {
return this.observe(willUplink);
}
@Override
public JoinValueLaneView didUplink(DidUplink didUplink) {
return this.observe(didUplink);
}
@Override
public JoinValueLaneView willEnter(WillEnter willEnter) {
return this.observe(willEnter);
}
@Override
public JoinValueLaneView didEnter(DidEnter didEnter) {
return this.observe(didEnter);
}
@Override
public JoinValueLaneView willLeave(WillLeave willLeave) {
return this.observe(willLeave);
}
@Override
public JoinValueLaneView didLeave(DidLeave didLeave) {
return this.observe(didLeave);
}
@SuppressWarnings("unchecked")
public Map.Entry dispatchWillUpdate(Link link, K key, V newValue, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
final Link oldLink = SwimContext.getLink();
try {
SwimContext.setLane(this);
SwimContext.setLink(link);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof WillUpdateKey, ?>) {
if (((WillUpdateKey, ?>) observers).isPreemptive() == preemptive) {
try {
newValue = ((WillUpdateKey) observers).willUpdate(key, newValue);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof WillUpdateKey, ?>) {
if (((WillUpdateKey, ?>) observer).isPreemptive() == preemptive) {
try {
newValue = ((WillUpdateKey) observer).willUpdate(key, newValue);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return new AbstractMap.SimpleImmutableEntry(complete, newValue);
} finally {
SwimContext.setLink(oldLink);
SwimContext.setLane(oldLane);
}
}
@SuppressWarnings("unchecked")
public boolean dispatchDidUpdate(Link link, K key, V newValue, V oldValue, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
final Link oldLink = SwimContext.getLink();
try {
SwimContext.setLane(this);
SwimContext.setLink(link);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof DidUpdateKey, ?>) {
if (((DidUpdateKey, ?>) observers).isPreemptive() == preemptive) {
try {
((DidUpdateKey) observers).didUpdate(key, newValue, oldValue);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof DidUpdateKey, ?>) {
if (((DidUpdateKey, ?>) observer).isPreemptive() == preemptive) {
try {
((DidUpdateKey) observer).didUpdate(key, newValue, oldValue);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return complete;
} finally {
SwimContext.setLink(oldLink);
SwimContext.setLane(oldLane);
}
}
@SuppressWarnings("unchecked")
public boolean dispatchWillRemove(Link link, K key, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
final Link oldLink = SwimContext.getLink();
try {
SwimContext.setLane(this);
SwimContext.setLink(link);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof WillRemoveKey>) {
if (((WillRemoveKey>) observers).isPreemptive() == preemptive) {
try {
((WillRemoveKey) observers).willRemove(key);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof WillRemoveKey>) {
if (((WillRemoveKey>) observer).isPreemptive() == preemptive) {
try {
((WillRemoveKey) observer).willRemove(key);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return complete;
} finally {
SwimContext.setLink(oldLink);
SwimContext.setLane(oldLane);
}
}
@SuppressWarnings("unchecked")
public boolean dispatchDidRemove(Link link, K key, V oldValue, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
final Link oldLink = SwimContext.getLink();
try {
SwimContext.setLane(this);
SwimContext.setLink(link);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof DidRemoveKey, ?>) {
if (((DidRemoveKey, ?>) observers).isPreemptive() == preemptive) {
try {
((DidRemoveKey) observers).didRemove(key, oldValue);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof DidRemoveKey, ?>) {
if (((DidRemoveKey, ?>) observer).isPreemptive() == preemptive) {
try {
((DidRemoveKey) observer).didRemove(key, oldValue);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return complete;
} finally {
SwimContext.setLink(oldLink);
SwimContext.setLane(oldLane);
}
}
public boolean dispatchWillClear(Link link, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
final Link oldLink = SwimContext.getLink();
try {
SwimContext.setLane(this);
SwimContext.setLink(link);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof WillClear) {
if (((WillClear) observers).isPreemptive() == preemptive) {
try {
((WillClear) observers).willClear();
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof WillClear) {
if (((WillClear) observer).isPreemptive() == preemptive) {
try {
((WillClear) observer).willClear();
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return complete;
} finally {
SwimContext.setLink(oldLink);
SwimContext.setLane(oldLane);
}
}
public boolean dispatchDidClear(Link link, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
final Link oldLink = SwimContext.getLink();
try {
SwimContext.setLane(this);
SwimContext.setLink(link);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof DidClear) {
if (((DidClear) observers).isPreemptive() == preemptive) {
try {
((DidClear) observers).didClear();
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof DidClear) {
if (((DidClear) observer).isPreemptive() == preemptive) {
try {
((DidClear) observer).didClear();
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return complete;
} finally {
SwimContext.setLink(oldLink);
SwimContext.setLane(oldLane);
}
}
@SuppressWarnings("unchecked")
public Map.Entry> dispatchWillDownlink(K key, ValueDownlink> downlink, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
try {
SwimContext.setLane(this);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof WillDownlinkValue>) {
if (((WillDownlinkValue>) observers).isPreemptive() == preemptive) {
try {
downlink = ((WillDownlinkValue) observers).willDownlink(key, downlink);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof WillDownlinkValue>) {
if (((WillDownlinkValue>) observer).isPreemptive() == preemptive) {
try {
downlink = ((WillDownlinkValue) observer).willDownlink(key, downlink);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return new AbstractMap.SimpleImmutableEntry>(complete, downlink);
} finally {
SwimContext.setLane(oldLane);
}
}
@SuppressWarnings("unchecked")
public boolean dispatchDidDownlink(K key, ValueDownlink> downlink, boolean preemptive) {
final Lane oldLane = SwimContext.getLane();
try {
SwimContext.setLane(this);
final Object observers = this.observers;
boolean complete = true;
if (observers instanceof DidDownlinkValue>) {
if (((DidDownlinkValue>) observers).isPreemptive() == preemptive) {
try {
((DidDownlinkValue) observers).didDownlink(key, downlink);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
} else if (observers instanceof Object[]) {
final Object[] array = (Object[]) observers;
for (int i = 0, n = array.length; i < n; i += 1) {
final Object observer = array[i];
if (observer instanceof DidDownlinkValue>) {
if (((DidDownlinkValue>) observer).isPreemptive() == preemptive) {
try {
((DidDownlinkValue) observer).didDownlink(key, downlink);
} catch (Throwable error) {
if (Cont.isNonFatal(error)) {
this.laneDidFail(error);
}
throw error;
}
} else if (preemptive) {
complete = false;
}
}
}
}
return complete;
} finally {
SwimContext.setLane(oldLane);
}
}
public ValueDownlink laneWillDownlink(K key, ValueDownlink downlink) {
return downlink;
}
public void laneDidDownlink(K key, ValueDownlink downlink) {
// hook
}
public V laneWillUpdate(K key, V newValue) {
return newValue;
}
public void laneDidUpdate(K key, V newValue, V oldValue) {
// hook
}
public void laneWillRemove(K key) {
// hook
}
public void laneDidRemove(K key, V oldValue) {
// hook
}
public void laneWillClear() {
// hook
}
public void laneDidClear() {
// hook
}
@Override
public ValueDownlink downlink(K key) {
final LaneContext laneContext = this.laneBinding.laneContext();
return new JoinValueLaneDownlink(laneContext, laneContext.stage(), this.laneBinding,
this.keyForm.mold(key).toValue(), this.laneBinding.meshUri(),
Uri.empty(), Uri.empty(), Uri.empty(), 0.0f, 0.0f,
Value.absent(), this.valueForm);
}
@Override
public boolean isEmpty() {
return this.dataView.isEmpty();
}
@Override
public int size() {
return this.dataView.size();
}
@Override
public boolean containsKey(Object key) {
return this.dataView.containsKey(key);
}
@Override
public boolean containsValue(Object value) {
return this.dataView.containsValue(value);
}
@Override
public V get(Object key) {
return this.dataView.get(key);
}
@SuppressWarnings("unchecked")
@Override
public ValueDownlink> getDownlink(Object key) {
return this.laneBinding.getDownlink(this.keyForm.mold((K) key));
}
@Override
public V put(K key, V value) {
return this.laneBinding.put(this, key, value);
}
@Override
public void putAll(Map extends K, ? extends V> map) {
for (Map.Entry extends K, ? extends V> entry : map.entrySet()) {
this.laneBinding.put(this, entry.getKey(), entry.getValue());
}
}
@SuppressWarnings("unchecked")
@Override
public V remove(Object key) {
final Class> keyType = this.keyForm.type();
if (keyType == null || keyType.isInstance(key)) {
return this.laneBinding.remove(this, (K) key);
}
return this.valueForm.unit();
}
@Override
public void clear() {
this.laneBinding.clear(this);
}
@SuppressWarnings("unchecked")
@Override
public Set> entrySet() {
if (this.keyForm != Form.forValue() || this.valueForm != Form.forValue()) {
return new JoinValueLaneViewEntrySet(this);
} else {
return (Set>) (Set>) this.laneBinding.entrySet();
}
}
@SuppressWarnings("unchecked")
@Override
public Set keySet() {
if (this.keyForm != Form.forValue()) {
return this.dataView.keySet();
} else {
return (Set) (Set>) this.laneBinding.keySet();
}
}
@SuppressWarnings("unchecked")
@Override
public Collection values() {
if (this.valueForm != Form.forValue()) {
return new JoinValueLaneViewValues(this);
} else {
return (Collection) this.laneBinding.values();
}
}
@SuppressWarnings("unchecked")
@Override
public Iterator> iterator() {
if (this.keyForm != Form.forValue() || this.valueForm != Form.forValue()) {
return new ValueEntryIterator(this.laneBinding.iterator(), this.keyForm, this.valueForm);
} else {
return (Iterator>) (Iterator>) this.laneBinding.iterator();
}
}
@Override
public Iterator keyIterator() {
return this.dataView.keyIterator();
}
@SuppressWarnings("unchecked")
@Override
public Iterator valueIterator() {
if (this.valueForm != Form.forValue()) {
return new ValueIterator(this.laneBinding.valueIterator(), this.valueForm);
} else {
return (Iterator) this.laneBinding.valueIterator();
}
}
@SuppressWarnings("unchecked")
@Override
public Iterator>> downlinkIterator() {
return (Iterator>>) (Iterator>) this.laneBinding.downlinks.iterator();
}
static final int RESIDENT = 1 << 0;
static final int TRANSIENT = 1 << 1;
}
final class JoinValueLaneViewEntrySet extends AbstractSet> {
final JoinValueLaneView view;
JoinValueLaneViewEntrySet(JoinValueLaneView view) {
this.view = view;
}
@Override
public int size() {
return this.view.size();
}
@Override
public Iterator> iterator() {
return this.view.iterator();
}
}
final class JoinValueLaneViewValues extends AbstractCollection {
final JoinValueLaneView view;
JoinValueLaneViewValues(JoinValueLaneView view) {
this.view = view;
}
@Override
public int size() {
return this.view.size();
}
@Override
public Iterator iterator() {
return this.view.valueIterator();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy