blob: 87034d18cfe4d585607f503e668eb4ca965d82bd [file]
/*
* 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.
*/
import com.fasterxml.jackson.databind.ObjectMapper
import com.fasterxml.jackson.databind.node.ArrayNode
import com.fasterxml.jackson.databind.node.ObjectNode
import jakarta.ws.rs.core.Response
import org.apache.cxf.jaxrs.client.WebClient
import org.identityconnectors.common.security.GuardedString
import org.identityconnectors.framework.common.objects.OperationOptions
// Parameters:
// The connector sends the following:
// client : CXF WebClient
// objectClass: a String describing the Object class (__ACCOUNT__ / __GROUP__ / other)
// action: a string describing the action ("SEARCH" here)
// log: a handler to the Log facility
// options: a handler to the OperationOptions Map
// query: a handler to the Query Map
//
// The Query map describes the filter used (via FIQL's ConditionType):
//
// query = [ conditionType: "EQUALS", left: attribute, right: "value" ]
// query = [ conditionType: "GREATER_THAN", left: attribute, right: "value" ]
// query = [ conditionType: "GREATER_OR_EQUALS", left: attribute, right: "value" ]
// query = [ conditionType: "LESS_THAN", left: attribute, right: "value" ]
// query = [ conditionType: "LESS_OR_EQUALS", left: attribute, right: "value" ]
// query = null : then we assume we fetch everything
//
// AND and OR filter just embed a left/right couple of queries.
// query = [ conditionType: "AND", left: query1, right: query2 ]
// query = [ conditionType: "OR", left: query1, right: query2 ]
//
// Returns: A list of Maps. Each map describing one row.
// !!!! Each Map must contain a '__UID__' and '__NAME__' attribute.
// This is required to build a ConnectorObject.
def buildConnectorObject(node) {
return [
__UID__:node.get("key").textValue(),
__NAME__:node.get("key").textValue(),
__ENABLE__:node.get("status").textValue().equals("ACTIVE"),
__PASSWORD__:new GuardedString(node.get("password").textValue().toCharArray()),
key:node.get("key").textValue(),
username:node.get("username").textValue(),
firstName:node.get("firstName").textValue(),
surname:node.get("surname").textValue(),
email:node.get("email").textValue()
];
}
log.info("Entering " + action + " Script");
// ----------------
// Manage pagination
// ----------------
def offset = options[OperationOptions.OP_PAGED_RESULTS_COOKIE] == null
? 0
: options[OperationOptions.OP_PAGED_RESULTS_COOKIE].toInteger();
def pageSize = options[OperationOptions.OP_PAGE_SIZE] == null
? 100
: options[OperationOptions.OP_PAGE_SIZE].toInteger();
def limit = offset + pageSize;
log.ok("pagedResultsCookie: " + offset);
log.ok("pageSize: " + pageSize);
log.ok("limit: " + limit);
// ----------------
WebClient webClient = client;
ObjectMapper mapper = new ObjectMapper();
def result = []
switch (objectClass) {
case "__ACCOUNT__":
if (query == null
|| (!query.get("left").equals("__UID__") && !query.get("left").equals("key")
&& !query.get("conditionType").name().equals("EQUALS"))) {
webClient.path("/users");
log.ok("Sending GET to {0}", webClient.getCurrentURI().toASCIIString());
Response response = webClient.get();
log.ok("LIST response: {0} {1}", response.getStatus(), response.getHeaders());
ArrayNode nodes = mapper.readTree(response.getEntity());
// beware: this is not enforcing any server-side pagination feature
for (int i = offset; i < (limit < nodes.size() ? limit: nodes.size()); i++) {
result.add(buildConnectorObject(nodes.get(i)));
}
} else {
webClient.path("/users/" + query.get("right"));
log.ok("Sending GET to {0}", webClient.getCurrentURI().toASCIIString());
Response response = webClient.get();
log.ok("READ response: {0} {1}", response.getStatus(), response.getHeaders());
if (response.getStatus() == 200) {
ObjectNode node = mapper.readTree(response.getEntity());
result.add(buildConnectorObject(node));
} else {
log.warn("Could not read object {0}", query.get("right"));
}
}
break
default:
result;
}
// ----------------
// Return paged result cookie
// ----------------
def pagedResultCookieLine = [:]
if (pageSize > result.size()) {
// no more results
pagedResultCookieLine.put(OperationOptions.OP_PAGED_RESULTS_COOKIE, null);
} else {
pagedResultCookieLine.put(OperationOptions.OP_PAGED_RESULTS_COOKIE, "" + limit);
}
result.add(pagedResultCookieLine);
// ----------------
return result;