/*
* Copyright 2016 The Simple File Server Authors
*
* 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 org.sfs.nodes.compute.account;
import com.google.common.base.Optional;
import io.vertx.core.Handler;
import org.sfs.Server;
import org.sfs.SfsRequest;
import org.sfs.VertxContext;
import org.sfs.auth.Authenticate;
import org.sfs.elasticsearch.account.LoadAccount;
import org.sfs.elasticsearch.account.PersistAccount;
import org.sfs.elasticsearch.account.UpdateAccount;
import org.sfs.rx.ConnectionCloseTerminus;
import org.sfs.validate.ValidateAccountPath;
import org.sfs.validate.ValidateActionAdmin;
import org.sfs.validate.ValidateOptimisticAccountLock;
import org.sfs.vo.PersistentAccount;
import org.sfs.vo.TransientAccount;
import org.sfs.vo.TransientServiceDef;
import rx.Observable;
import rx.functions.Func1;
import static java.net.HttpURLConnection.HTTP_NO_CONTENT;
import static org.sfs.rx.Defer.aVoid;
import static org.sfs.rx.Defer.just;
import static org.sfs.vo.ObjectPath.fromSfsRequest;
public class PostAccount implements Handler<SfsRequest> {
@Override
public void handle(final SfsRequest httpServerRequest) {
aVoid()
.flatMap(new Authenticate(httpServerRequest))
.flatMap(new ValidateActionAdmin(httpServerRequest))
.map(aVoid -> fromSfsRequest(httpServerRequest))
.map(new ValidateAccountPath())
.map(objectPath -> objectPath.accountPath().get())
.flatMap(new LoadAccount(httpServerRequest.vertxContext()))
.flatMap(new Func1<Optional<PersistentAccount>, Observable<PersistentAccount>>() {
@Override
public Observable<PersistentAccount> call(Optional<PersistentAccount> oPersistentAccount) {
if (oPersistentAccount.isPresent()) {
PersistentAccount persistentAccount = oPersistentAccount.get();
persistentAccount.merge(httpServerRequest);
return just(persistentAccount)
.flatMap(new UpdateAccount(httpServerRequest.vertxContext()))
.map(new ValidateOptimisticAccountLock());
} else {
return just(TransientAccount.fromSfsRequest(httpServerRequest))
.doOnNext(transientAccount -> {
VertxContext<Server> vertxContext = httpServerRequest.vertxContext();
Optional<TransientServiceDef> currentMaintainerNode =
vertxContext
.verticle()
.getClusterInfo()
.getCurrentMaintainerNode();
if (currentMaintainerNode.isPresent()) {
transientAccount.setNodeId(currentMaintainerNode.get().getId());
}
})
.flatMap(new PersistAccount(httpServerRequest.vertxContext()))
.map(new ValidateOptimisticAccountLock());
}
}
})
.single()
.subscribe(new ConnectionCloseTerminus<PersistentAccount>(httpServerRequest) {
@Override
public void onNext(PersistentAccount persistentAccount) {
httpServerRequest.response().setStatusCode(HTTP_NO_CONTENT);
}
});
}
}