SharePoint JSOM KeywordQuery меняет ContentType, который он ищет

В настоящее время я использую AngularJS с SharePoint JSOM для выполнения некоторых запросов по ключевым словам к источнику результатов.

У меня есть угловая фабрика, которая принимает переменные ResultSourceName и ResultSourceLevel.

angular.module('app.services.sharepoint').factory('resultSourceDataSvc', ['$log', '$q', '$timeout', function ($log, $q, $timeout) {
    return function (resultSourceName, resultSourceLevel) {         
        return new ResultSourceDataService($log, $q, $timeout, resultSourceName, resultSourceLevel);
    }
}]);

Это вводится в мои контроллеры, и они используют это так:

var inTheNewsResultDataSource = resultSourceData('TSToday In The News Result Source', 'SPSite');
inTheNewsResultDataSource.ready(function () {
    inTheNewsResultDataSource.getData(null, IN_THE_NEWS_COLUMNS, IN_THE_NEWS_MAP_COLUMNS)
        .then(function (data) {
            $log.debug('Resolving data:', data);
            deferred.resolve(data);
        },
        function (resp) {
            $log.error('Error while attempting to retrieve In The News data:', resp);
            deferred.reject(resp);
        });
});

Фактическое мясо всего этого обрабатывается в ResultSourceDataService, который возвращается с моей фабрики, вот код из него:

function ResultSourceDataService($log, $q, $timeout, resultSourceName, resultSourceLevel) {
    var deferred = $q.defer(),
        promise = deferred.promise,
        context = undefined,
        resultSourceQuery = undefined,
        resultSourceExecutor = undefined,
        properties = undefined;

    $log = $log.getInstance('app.services.sharepoint.resultSourceDataSvc(' + resultSourceName + ')');

    function init() {           
        $log.debug('Initializing new Result Source Data Service for:', resultSourceName, 'on', resultSourceLevel);          

        SP.SOD.registerSod('sp.search.js', '/_layouts/15/sp.search.js');

        SP.SOD.loadMultiple(['sp.js', 'sp.search.js'], function () {
            context = SP.ClientContext ? SP.ClientContext.get_current() : undefined;

            // should probably add error checking to make sure context is loaded here

            resultSourceQuery = new Microsoft.SharePoint.Client.Search.Query.KeywordQuery(context);
            resultSourceExecutor = new Microsoft.SharePoint.Client.Search.Query.SearchExecutor(context);

            resultSourceQuery.set_trimDuplicates(false);

            resultSourceQuery.get_properties().set_item('SourceName', resultSourceName);
            resultSourceQuery.get_properties().set_item('SourceLevel', resultSourceLevel);

            $log.debug('Initialized');
            deferred.resolve();
        });
    }
    init();

    //#region Private Functions
    function setSelectProperties(columnsToRetrieve) {
        properties = resultSourceQuery.get_selectProperties();

        for (var i = columnsToRetrieve.length; i-- > 0;) {
            $log.debug('Adding in property to retrieve', columnsToRetrieve[i]);
            properties.add(columnsToRetrieve[i]);
        }
    }
    function getResults(data, columnsToRetrieve, columnsToMap) {
        return data.map(function (obj, i) {
            var result = columnsToRetrieve.length == 0 ? obj : {};
            for (var j = columnsToRetrieve.length; j-- > 0;) {
                result[columnsToMap[j]] = obj[columnsToRetrieve[j]];
            }
            $log.debug('getResult', result);

            return result;
        });
    }
    //#endregion

    //#region Public Functions
    this.getData = function (keyword, columnsToRetrieve, columnsToMap) {
        var deferred = $q.defer(),
            promise = deferred.promise,
            data = undefined,
            results = [];

        keyword = keyword || '*';
        columnsToRetrieve = columnsToRetrieve || [];
        columnsToMap = columnsToMap || columnsToRetrieve;           

        resultSourceQuery.set_queryText(keyword);
        setSelectProperties(columnsToRetrieve);

        $log.info('Attempting to retrieve data from result source with keyword:', keyword, resultSourceQuery);

        data = resultSourceExecutor.executeQuery(resultSourceQuery);

        context.executeQueryAsync(
            function (sender, args) {
                $log.debug('Data retrieved successfully', data.get_value().ResultTables);
                results = getResults(data.get_value().ResultTables[0].ResultRows, columnsToRetrieve, columnsToMap);                    
                $log.info('Resolving data', results);
                deferred.resolve(results);
            },
            function (sender, args) {
                $log.error('Error with resolving data in ResultSourceDataService', args.get_message());
                deferred.reject(args.get_message());
            });

        return promise;
    }
    this.ready = function (fn) {
        promise.then(fn);
    }
    //#endregion        
}

Когда я загружаю страницу, на которой работает этот код, он работает нормально. Когда я обновляю страницу, все работает нормально. Однако, если я перейду на эту страницу с любой другой страницы моего сайта, она сломается и вернет данные, как если бы источник результата не был указан.

По какой-то причине QueryModification (которая, если я прав, по существу сообщает, какой ContentType мы ищем в запросе) меняется. Когда я первоначально загружаю страницу и просматриваю данные, записанные в консоль ($log.debug('Data retrieved successfully', data.get_value().ResultTables);), я могу углубиться в них и увидеть разницу в QueryModification.

Когда он работает правильно (обновление страницы или переход на нее непосредственно из-за пределов сайта), я вижу правильное значение ContentTypeId:0x0100A9723709C6D74A77B01EA922C770FDD0*. Однако; если я перейду на страницу из SharePoint, я увижу значение * -ContentClass=urn:content-class:SPSPeople. Таким образом, он возвращает результаты, которые будут отображаться, если я не укажу имя источника результатов.

Любые мысли о том, что может быть причиной этого, когда я передаю ему одни и те же точные переменные?


person Ben Black    schedule 25.09.2015    source источник


Ответы (1)


Не знаю, почему это работает, но удаление

        resultSourceQuery.get_properties().set_item('SourceName', resultSourceName);
        resultSourceQuery.get_properties().set_item('SourceLevel', resultSourceLevel);

из метода init и добавление его к методу getData устранило проблему. Не уверен, почему это сработает, поскольку его всегда следует устанавливать перед методом getData, если вызывающий код использует метод ready.

Если кто-то видит, почему это исправлено, пожалуйста, укажите это. Тем временем моя проблема решена.

person Ben Black    schedule 25.09.2015