| /* |
| * 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; |