//%LICENSE//////////////////////////////////////////////////////////////// // // Licensed to The Open Group (TOG) under one or more contributor license // agreements. Refer to the OpenPegasusNOTICE.txt file distributed with // this work for additional information regarding copyright ownership. // Each contributor licenses this file to you under the OpenPegasus Open // Source License; you may not use this file except in compliance with the // License. // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included // in all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. // IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY // CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, // TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE // SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // ////////////////////////////////////////////////////////////////////////// // //%///////////////////////////////////////////////////////////////////////////// #include "WQLOperationRequestDispatcher.h" #include #include PEGASUS_NAMESPACE_BEGIN PEGASUS_USING_STD; void WQLOperationRequestDispatcher::applyQueryToEnumeration( CIMResponseMessage* msg, QueryExpressionRep* query) { CIMEnumerateInstancesResponseMessage* enr = (CIMEnumerateInstancesResponseMessage*) msg; WQLSelectStatement* qs = ((WQLQueryExpressionRep*)query)->_stmt; Array& a = enr->getResponseData().getInstances(); for (int i = a.size() - 1; i >= 0; i--) { WQLInstancePropertySource ips(a[i]); try { if (qs->evaluateWhereClause(&ips)) { // // Specify that missing requested project properties are // allowed to be consistent with clarification from DMTF // qs->applyProjection(a[i], true); } else { a.remove(i); } } catch (...) { a.remove(i); } } } void WQLOperationRequestDispatcher::handleQueryRequest( CIMExecQueryRequestMessage* request) { PEG_METHOD_ENTER(TRC_DISPATCHER, "WQLOperationRequestDispatcher::handleQueryRequest"); Boolean exception=false; AutoPtr selectStatement(new WQLSelectStatement()); AutoPtr qx; CIMException cimException; CIMName className; if (request->queryLanguage!="WQL") { cimException = PEGASUS_CIM_EXCEPTION( CIM_ERR_QUERY_LANGUAGE_NOT_SUPPORTED, request->queryLanguage); exception=true; } else { try { WQLParser::parse(request->query, *selectStatement.get()); className = selectStatement->getClassName(); qx.reset(new WQLQueryExpressionRep("WQL", selectStatement.get())); selectStatement.release(); } catch (ParseError&) { cimException = PEGASUS_CIM_EXCEPTION(CIM_ERR_INVALID_QUERY, request->query); exception=true; } catch (MissingNullTerminator&) { cimException = PEGASUS_CIM_EXCEPTION(CIM_ERR_INVALID_QUERY, request->query); exception = true; } if (exception == false) { if (!_checkExistenceOfClass(request->nameSpace, className)) { cimException = PEGASUS_CIM_EXCEPTION( CIM_ERR_INVALID_CLASS, className.getString()); exception = true; } } } if (exception) { CIMResponseMessage* response = request->buildResponse(); response->cimException = cimException; _enqueueResponse(request, response); PEG_METHOD_EXIT(); return; } // // Get names of descendent classes: // ProviderInfoList providerInfos; try { providerInfos = _lookupAllInstanceProviders( request->nameSpace, className); } catch (CIMException& e) { // Return exception response if exception from getSubClasses CIMResponseMessage* response = request->buildResponse(); response->cimException = e; _enqueueResponse(request, response); PEG_METHOD_EXIT(); return; } // If no provider is registered and the repository isn't the default, // return CIM_ERR_NOT_SUPPORTED if (_rejectNoProvidersOrRepository(request,providerInfos,className)) { PEG_METHOD_EXIT(); return; } // We have instances for Providers and possibly repository. // Set up an aggregate object and save a copy of the original request. // NOTE: OperationAggregate released only when operation complete OperationAggregate* poA= new OperationAggregate( new CIMExecQueryRequestMessage(*request), className, request->nameSpace, providerInfos.providerCount, false, false, qx.release(), "WQL"); // Set the number of expected responses in the OperationAggregate Uint32 numClasses = providerInfos.size(); // Build enum request for call to repository. AutoPtr repRequest( new CIMEnumerateInstancesRequestMessage( request->messageId, request->nameSpace, CIMName(), false,false,false, CIMPropertyList(), request->queueIds, request->authType, request->userName)); if (_enumerateFromRepository(repRequest.release(), poA, providerInfos)) { CIMResponseMessage* response = poA->removeResponse(0); _forwardResponseForAggregation( new CIMExecQueryRequestMessage(*request), poA, response); } // Loop through providerInfos, forwarding requests to providers while (providerInfos.hasMore(true)) { // If this class has a provider ProviderInfo& providerInfo = providerInfos.getNext(); PEG_TRACE((TRC_DISPATCHER, Tracer::LEVEL4, "Routing ExecQuery request for class %s to " "service \"%s\" for control provider \"%s\". " "Class # %u of %u", CSTRING(providerInfo.className.getString()), lookup(providerInfo.serviceId)->getQueueName(), CSTRING(providerInfo.controlProviderName), providerInfos.getIndex(), numClasses )); ProviderIdContainer* providerIdContainer = providerInfo.providerIdContainer.get(); // if not a provider with Query capability, execute // EnumerateInstances for on the provider. if (providerInfo.hasNoQuery) { OperationContext* context = &request->operationContext; const OperationContext::Container* container = 0; container = &context->get(IdentityContainer::NAME); const IdentityContainer& identityContainer = dynamic_cast(*container); AutoPtr enumReq( new CIMEnumerateInstancesRequestMessage( request->messageId, request->nameSpace, providerInfo.className, false,false,false, CIMPropertyList(), request->queueIds, request->authType, identityContainer.getUserName())); context = &enumReq->operationContext; if (providerIdContainer) context->insert(*providerIdContainer); context->insert(identityContainer); _forwardRequestForAggregation( providerInfo.serviceId, providerInfo.controlProviderName, enumReq.release(), poA); } else { AutoPtr requestCopy( new CIMExecQueryRequestMessage(*request)); OperationContext* context = &request->operationContext; if (providerIdContainer) context->insert(*providerIdContainer); requestCopy->operationContext = *context; requestCopy->className = providerInfo.className; _forwardRequestForAggregation( providerInfo.serviceId, providerInfo.controlProviderName, requestCopy.release(), poA); } } // for all classes and derived classes PEG_METHOD_EXIT(); } PEGASUS_NAMESPACE_END