//%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 "CQLOperationRequestDispatcher.h" #include #include #include #include PEGASUS_NAMESPACE_BEGIN PEGASUS_USING_STD; void CQLOperationRequestDispatcher::applyQueryToEnumeration( CIMResponseMessage* msg, QueryExpressionRep* query) { PEG_METHOD_ENTER(TRC_DISPATCHER, "CQLOperationRequestDispatcher::applyQueryToEnumeration"); CIMEnumerateInstancesResponseMessage* enr = (CIMEnumerateInstancesResponseMessage*) msg; CQLSelectStatement* qs = ((CQLQueryExpressionRep*)query)->_stmt; Array& a = enr->getResponseData().getInstances(); for (int i = a.size() - 1; i >= 0; i--) { try { if (qs->evaluate(a[i])) { // // 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); } } PEG_METHOD_EXIT(); } void CQLOperationRequestDispatcher::handleQueryRequest( CIMExecQueryRequestMessage* request) { PEG_METHOD_ENTER(TRC_DISPATCHER, "CQLOperationRequestDispatcher::handleQueryRequest"); Boolean exception=false; CIMOMHandle _ch; CIMOMHandleQueryContext _queryOrig(request->nameSpace,_ch); AutoPtr selectStatement( new CQLSelectStatement(request->queryLanguage, request->query, _queryOrig)); AutoPtr qx; CIMException cimException; CIMName className; if (request->queryLanguage != "DMTF:CQL") { cimException = PEGASUS_CIM_EXCEPTION( CIM_ERR_QUERY_LANGUAGE_NOT_SUPPORTED, request->queryLanguage); exception=true; } else { try { CQLParser::parse(request->query, *selectStatement.get()); Array classPath = selectStatement->getClassPathList(); className = classPath[0].getClassName(); qx.reset( new CQLQueryExpressionRep("DMTF:CQL", 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; // This exception should not be required or we should apply for // all _lookupInstanceProvider Calls. 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.providerCount, 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. OperationAggregate* poA= new OperationAggregate( new CIMExecQueryRequestMessage(*request), request->getType(), request->messageId, request->queueIds.top(), className, CIMNamespaceName(), qx.release(), "DMTF:CQL"); // Set the number of expected responses in the OperationAggregate Uint32 numClasses = providerInfos.size(); poA->_nameSpace=request->nameSpace; if (_repository->isDefaultInstanceProvider()) { // Loop through providerInfos, forwarding requests to repository for (Uint32 i = 0; i < numClasses; i++) { ProviderInfo& providerInfo = providerInfos[i]; // this class is registered to a provider - skip if (providerInfo.hasProvider) { continue; } // If this class does not have a provider PEG_TRACE((TRC_DISPATCHER, Tracer::LEVEL4, "Routing ExecQuery request for class %s to the " "repository. Class # %u of %u", CSTRING(providerInfo.className.getString()), (i + 1), numClasses )); // Create an EnumerateInstances response from an ExecQuery request AutoPtr response( new CIMEnumerateInstancesResponseMessage( request->messageId, CIMException(), request->queueIds.copyAndPop())); response->syncAttributes(request); try { // Enumerate instances only for this class response->getResponseData().setInstances( _repository->enumerateInstancesForClass( request->nameSpace, providerInfo.className)); } catch (CIMException& e) { response->cimException = e; } catch (Exception& e) { response->cimException = PEGASUS_CIM_EXCEPTION( CIM_ERR_FAILED, e.getMessage()); } catch (...) { response->cimException = PEGASUS_CIM_EXCEPTION( CIM_ERR_FAILED, String::EMPTY); } poA->appendResponse(response.release()); } // for all classes and derived classes Uint32 numberResponses = poA->numberResponses(); Uint32 totalIssued = providerInfos.providerCount + (numberResponses > 0 ? 1 : 0); poA->setTotalIssued(totalIssued); if (numberResponses > 0) { handleOperationResponseAggregation(poA,false,false); CIMResponseMessage* response = poA->removeResponse(0); _forwardRequestForAggregation( getQueueId(), String(), new CIMExecQueryRequestMessage(*request), poA, response); } } // if isDefaultInstanceProvider else { // Set the number of expected responses in the OperationAggregate PEG_TRACE((TRC_DISPATCHER, Tracer::LEVEL4, "providerCount:%u",(unsigned int)providerInfos.providerCount)); poA->setTotalIssued(providerInfos.providerCount); } // Loop through providerInfos, forwarding requests to providers for (Uint32 i = 0; i < numClasses; i++) { // If this class has a provider ProviderInfo& providerInfo = providerInfos[i]; // this class is NOT registered to a provider - skip if (!providerInfo.hasProvider) { continue; } 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), (i + 1), numClasses )); ProviderIdContainer* providerIdContainer = providerInfo.providerIdContainer.get(); 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