Удаленный сервер возвратил ошибку: (401) Несанкционированный - CSOM - OAuth - через определенное время - PullRequest
0 голосов
/ 11 мая 2018

Я пытаюсь получить разрешения, назначенные каждому документу из библиотеки документов, используя csom в консольном приложении. Консольное приложение работает нормально в течение почти одного часа и выдает сообщение об ошибке: «Удаленный сервер возвратил ошибку: (401) Несанкционировано».

Я зарегистрировал приложение в SharePoint и использую идентификатор клиента и секретный ключ клиента для создания контекста клиента. Вот код для получения контекста клиента

public static ClientContext getClientContext()
    {
        ClientContext ctx = null;
        try
        {
            ctx = new AuthenticationManager().GetAppOnlyAuthenticatedContext(SiteUrl, ClientId, ClientSecret);
            ctx.RequestTimeout = 6000000;

            ctx.ExecutingWebRequest += delegate (object sender, WebRequestEventArgs e)
            {
                e.WebRequestExecutor.WebRequest.UserAgent = "NONISV|Contoso|GovernanceCheck/1.0";
            };
        }
        catch (Exception ex)
        {
            WriteLog("Method-getClientContext, Error - " + ex.Message);
        }
        return ctx;
    }

Вот код, который получает все назначения, назначенные документу в библиотеке документов.

static StringBuilder excelData = new StringBuilder();            
static void ProcessWebnLists()
    {
        try
        {
            string[] lists = { "ABC", "XYZ", "DEF", "GHI"};

            foreach (string strList in lists)
            {
                using (var ctx = getClientContext())
                {
                    if (ctx != null)
                    {
                        Web web = ctx.Site.RootWeb;
                        ListCollection listCollection = web.Lists;
                        ctx.Load(web);
                        ctx.Load(listCollection);
                        ctx.ExecuteQuery();

                        List list = web.Lists.GetByTitle(strList);
                        ctx.Load(list);
                        ctx.ExecuteQuery();

                        Console.WriteLine("List Name:{0}", list.Title);

                        var query = new CamlQuery { ViewXml = "<View/>" };

                        var items = list.GetItems(query);
                        ctx.Load(items);
                        ctx.ExecuteQuery();

                        foreach (var item in items)
                        {
                            var itemAssignments = item.RoleAssignments;
                            ctx.Load(item, i => i.DisplayName);
                            ctx.Load(itemAssignments);
                            ctx.ExecuteQuery();
                            Console.WriteLine(item.DisplayName);
                            string groupNames = "";
                            foreach (var assignment in itemAssignments)
                            {
                                try
                                {
                                    ctx.Load(assignment.Member);
                                    ctx.ExecuteQuery();
                                    groupNames += "[" + assignment.Member.Title.Replace(",", " ") + "] ";
                                    Console.WriteLine($"--> {assignment.Member.Title}");
                                }
                                catch (Exception e)
                                {
                                    WriteLog("Method-ProcessWebnLists, List-" + list.Title + ", Document Title - " + item.DisplayName + ", Error : " + e.Message);
                                    WriteLog("Method-ProcessWebnLists, StackTrace : " + e.StackTrace);
                                }
                            }
                            excelData.AppendFormat("{0},{1},ITEM,{2}", list.Title, (item.DisplayName).Replace(",", " "), groupNames);
                            excelData.AppendLine();
                        }
                    }
                    excelData.AppendLine();
                }
            }
        }
        catch (Exception ex)
        {
            WriteLog("Method-ProcessWebnLists, Error : " + ex.Message);
            WriteLog("Method-ProcessWebnLists, StackTrace : " + ex.StackTrace.ToString());
        }
    }

Кто-нибудь может подсказать, что еще мне не хватает в этом или какие изменения мне нужно сделать, чтобы избежать этой ошибки?

Процесс занимает слишком много времени, поэтому я хочу, чтобы мое приложение продолжало работать в течение нескольких часов.

Заранее спасибо!

1 Ответ

0 голосов
/ 20 июля 2018

Есть ли причина, по которой getClientContext() находится внутри цикла foreach?

Я сталкивался с этой ошибкой раньше. Я добавил это, чтобы решить мой.

public static void ExecuteQueryWithIncrementalRetry(this ClientContext context, int retryCount, int delay)
{
    int retryAttempts = 0;
    int backoffInterval = delay;
    if (retryCount <= 0)
        throw new ArgumentException("Provide a retry count greater than zero.");
   if (delay <= 0)
        throw new ArgumentException("Provide a delay greater than zero.");
   while (retryAttempts < retryCount)
    {
        try
        {
            context.ExecuteQuery();
            return;
        }
        catch (WebException wex)
        {
            var response = wex.Response as HttpWebResponse;
            if (response != null &amp;&amp; response.StatusCode == (HttpStatusCode)429)
            {
                Console.WriteLine(string.Format("CSOM request exceeded usage limits. Sleeping for {0} seconds before retrying.", backoffInterval));
                //Add delay.
                System.Threading.Thread.Sleep(backoffInterval);
                //Add to retry count and increase delay.
                retryAttempts++;
                backoffInterval = backoffInterval * 2;
            }
            else
            {
                throw;
            }
        }
    }
    throw new MaximumRetryAttemptedException(string.Format("Maximum retry attempts {0}, have been attempted.", retryCount));
}

По сути, этот код откатывается на определенный промежуток времени / интервал перед выполнением другого запроса, чтобы предотвратить удушение.

Таким образом, вместо использования ctx.ExecuteQuery(), используйте ctx.ExecuteQueryWithIncrementalRetry(5, 1000);

Дальнейшее объяснение здесь https://docs.microsoft.com/en-us/sharepoint/dev/general-development/how-to-avoid-getting-throttled-or-blocked-in-sharepoint-online

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...