org.opendaylight.netconf.shaded.sshd.server.StandardEnvironment Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.opendaylight.netconf.shaded.sshd.server;
import java.util.Collection;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArraySet;
import org.opendaylight.netconf.shaded.sshd.common.channel.Channel;
import org.opendaylight.netconf.shaded.sshd.common.channel.PtyMode;
import org.opendaylight.netconf.shaded.sshd.common.util.GenericUtils;
import org.opendaylight.netconf.shaded.sshd.common.util.ValidateUtils;
import org.opendaylight.netconf.shaded.sshd.common.util.logging.AbstractLoggingBean;
/**
* @author Apache MINA SSHD Project
*/
public class StandardEnvironment extends AbstractLoggingBean implements Environment {
private final Map> listeners;
private final Map env;
private final Map ptyModes;
public StandardEnvironment() {
listeners = new ConcurrentHashMap<>(Signal.SIGNALS.size());
env = new ConcurrentHashMap<>();
ptyModes = new ConcurrentHashMap<>(PtyMode.MODES.size());
}
/*
* NOTE: we don't care if the collection is a Set or not - after all, we hold the listeners inside a Set, so even if
* we add several times the same listener to the same signal set, it is harmless
*/
@Override
public void addSignalListener(SignalListener listener, Collection signals) {
SignalListener.validateListener(listener);
ValidateUtils.checkNotNullAndNotEmpty(signals, "No signals");
for (Signal s : signals) {
Collection list = getSignalListeners(s, true);
list.add(listener);
}
}
@Override
public Map getEnv() {
return env;
}
@Override
public Map getPtyModes() {
return ptyModes;
}
@Override
public void removeSignalListener(SignalListener listener) {
if (listener == null) {
return;
}
SignalListener.validateListener(listener);
for (Signal s : Signal.SIGNALS) {
Collection ls = getSignalListeners(s, false);
if (ls != null) {
ls.remove(listener);
}
}
}
public void signal(Channel channel, Signal signal) {
Collection ls = getSignalListeners(signal, false);
if (log.isDebugEnabled()) {
log.debug("signal({})[{}] - listeners={}", channel, signal, ls);
}
if (GenericUtils.isEmpty(ls)) {
return;
}
boolean traceEnabled = log.isTraceEnabled();
for (SignalListener l : ls) {
try {
l.signal(channel, signal);
if (traceEnabled) {
log.trace("signal({}) Signal {} to {}", channel, signal, l);
}
} catch (RuntimeException e) {
log.warn("signal({}) Failed ({}) to signal {} to listener={}: {}",
channel, e.getClass().getSimpleName(), signal, l, e.getMessage());
}
}
}
/**
* Adds a variable to the environment. This method is called set
according to the name of the
* appropriate posix command set
*
* @param key environment variable name - never {@code null}/empty
* @param value environment variable value
*/
public void set(String key, String value) {
// TODO: listening for property changes would be nice too.
Map environ = getEnv();
environ.put(ValidateUtils.checkNotNullAndNotEmpty(key, "Empty environment variable name"), value);
}
/**
* Retrieves the set of listeners registered for a signal
*
* @param signal The specified {@link Signal}
* @param create If {@code true} and no current listeners are mapped then creates a new {@link Collection}
* @return The {@link Collection} of listeners registered for the signal - may be {@code null} in case
* create is {@code false}
*/
protected Collection getSignalListeners(Signal signal, boolean create) {
Collection ls = listeners.get(signal);
if ((ls == null) && create) {
synchronized (listeners) {
ls = listeners.get(signal);
if (ls == null) {
ls = new CopyOnWriteArraySet<>();
listeners.put(signal, ls);
}
}
}
return ls;
}
@Override
public String toString() {
return "env=" + getEnv() + ", modes=" + getPtyModes();
}
}