/*
GRANITE DATA SERVICES
Copyright (C) 2012 GRANITE DATA SERVICES S.A.S.
This file is part of Granite Data Services.
Granite Data Services is free software; you can redistribute it and/or modify
it under the terms of the GNU Library General Public License as published by
the Free Software Foundation; either version 2 of the License, or (at your
option) any later version.
Granite Data Services is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public License
for more details.
You should have received a copy of the GNU Library General Public License
along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
package org.granite.client.tide.javafx;
import java.nio.charset.Charset;
import java.util.Observable;
import java.util.Observer;
import java.util.concurrent.Future;
import javafx.beans.property.BooleanProperty;
import javafx.beans.property.ReadOnlyStringWrapper;
import javafx.beans.property.SimpleBooleanProperty;
import javafx.beans.property.StringProperty;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.ObservableValue;
import org.granite.client.messaging.messages.responses.FaultMessage;
import org.granite.client.messaging.messages.responses.FaultMessage.Code;
import org.granite.client.tide.Context;
import org.granite.client.tide.Identity;
import org.granite.client.tide.impl.ComponentImpl;
import org.granite.client.tide.server.ExceptionHandler;
import org.granite.client.tide.server.ServerSession;
import org.granite.client.tide.server.SimpleTideResponder;
import org.granite.client.tide.server.TideFaultEvent;
import org.granite.client.tide.server.TideResponder;
import org.granite.client.tide.server.TideResultEvent;
/**
* @author William DRAI
*/
public abstract class BaseIdentity extends ComponentImpl implements Identity, ExceptionHandler {
private BooleanProperty loggedIn = new SimpleBooleanProperty(this, "loggedIn");
private StringProperty username = new ReadOnlyStringWrapper(this, "username", null);
public BooleanProperty loggedInProperty() {
return loggedIn;
}
public boolean isLoggedIn() {
return loggedIn.get();
}
public void setLoggedIn(boolean loggedIn) {
this.loggedIn.set(loggedIn);
}
public StringProperty usernameProperty() {
return username;
}
public String getUsername() {
return username.get();
}
protected BaseIdentity() {
// CDI proxying...
}
public BaseIdentity(final ServerSession serverSession) {
super(serverSession);
this.loggedIn.set(false);
this.loggedIn.addListener(new ChangeListener<Boolean>() {
@Override
public void changed(ObservableValue<? extends Boolean> property, Boolean oldValue, Boolean newValue) {
if (Boolean.TRUE.equals(newValue)) {
initSecurityCache();
serverSession.afterLogin();
}
else {
BaseIdentity.this.username.set(null);
clearSecurityCache();
}
}
});
}
/**
* Triggers a remote call to check is user is currently logged in
* Can be used at application startup to handle browser refresh cases
*
* @param tideResponder optional responder
* @return future result: username or null
*/
public Future<String> checkLoggedIn(final TideResponder<String> tideResponder) {
return super.call("isLoggedIn", new SimpleTideResponder<String>() {
@Override
public void result(TideResultEvent<String> event) {
if (event.getResult() != null) {
BaseIdentity.this.username.set(event.getResult());
BaseIdentity.this.loggedIn.set(true);
}
else if (isLoggedIn()) {
BaseIdentity.this.loggedIn.set(false);
// Session expired, directly mark the channel as logged out
getServerSession().sessionExpired();
}
if (tideResponder != null)
tideResponder.result(event);
}
@Override
public void fault(TideFaultEvent event) {
if (event.getFault().getCode() == Code.ACCESS_DENIED) {
// Not in role for the destination
BaseIdentity.this.loggedIn.set(false);
getServerSession().logout(null);
}
if (tideResponder != null)
tideResponder.fault(event);
}
});
}
public void login(final String username, String password, final TideResponder<String> tideResponder) {
getServerSession().login(username, password);
clearSecurityCache();
try {
// Force synchronous operation to prevent issues with Spring session fixation protection
// so next remote calls use the correct session id
checkLoggedIn(tideResponder).get();
}
catch (Exception e) {
}
}
public void login(final String username, String password, Charset charset, final TideResponder<String> tideResponder) {
getServerSession().login(username, password, charset);
clearSecurityCache();
try {
// Force synchronous operation to prevent issues with Spring session fixation protection
// so next remote calls use the correct session id
checkLoggedIn(tideResponder).get();
}
catch (Exception e) {
}
}
public void logout(final TideResponder<Void> tideResponder) {
final Observer observer = new Observer() {
@SuppressWarnings("unchecked")
@Override
public void update(Observable logout, Object event) {
BaseIdentity.this.loggedIn.set(false);
if (tideResponder != null) {
if (event instanceof TideResultEvent)
tideResponder.result((TideResultEvent<Void>)event);
else if (event instanceof TideFaultEvent)
tideResponder.fault((TideFaultEvent)event);
}
}
};
getServerSession().logout(observer);
}
public abstract ObservableRole hasRole(String roleName);
protected abstract void initSecurityCache();
/**
* Clear the security cache
*/
public abstract void clearSecurityCache();
@Override
public boolean accepts(FaultMessage emsg) {
return emsg.getCode() == Code.NOT_LOGGED_IN;
}
@Override
public void handle(Context context, FaultMessage emsg, TideFaultEvent faultEvent) {
if (isLoggedIn()) {
setLoggedIn(false);
// Session expired, directly mark the channel as logged out
getServerSession().sessionExpired();
}
}
}