【问题标题】:Using gmail api with asp.net web api将gmail api与asp.net web api一起使用
【发布时间】:2016-01-09 05:24:23
【问题描述】:

我想在 asp.net web api 项目中使用 gmail api。这是我到目前为止所拥有的:

        UserCredential credential;
        var cleantSecretPath = HostingEnvironment.MapPath("~/client_secret.json");
        var rootPath = HostingEnvironment.MapPath("~/");

        if (rootPath != null)
        {
            var credentialPath = Path.Combine(rootPath, "Credentials");
            var directoryInfo = Directory.CreateDirectory(credentialPath);
        }

        var credentialsPath = HostingEnvironment.MapPath("~/Credentials/");

        using (var stream = new FileStream(cleantSecretPath, FileMode.Open, FileAccess.Read))
        {
            var secret = GoogleClientSecrets.Load(stream).Secrets;
            var dataStore = new FileDataStore("test");
            credential = GoogleWebAuthorizationBroker.AuthorizeAsync(secret, Scopes, _credentials.Email, CancellationToken.None, dataStore).Result;
        }

        var service = new GmailService(new BaseClientService.Initializer()
        {
            HttpClientInitializer = credential,
            ApplicationName = ApplicationName,
        });
        return

它可以在本地主机上运行,​​但是一旦我将它部署到 azure,我就会收到一个错误:

<Error>
    <Message>An error has occurred.</Message>
    <ExceptionMessage>One or more errors occurred.</ExceptionMessage>
    <ExceptionType>System.AggregateException</ExceptionType>
    <StackTrace>
    at System.Threading.Tasks.Task.ThrowIfExceptional(Boolean includeTaskCanceledExceptions) at System.Threading.Tasks.Task`1.GetResultCore(Boolean waitCompletionNotification) at System.Threading.Tasks.Task`1.get_Result() at BL.GmailServiceBl.GetGmailService() in D:\Development\DotNetProjects\GmailAnalysis\BL\GmailServiceBl.cs:line 53 at BL.AnalysisService..ctor(PersonCredentials credentials) in D:\Development\DotNetProjects\GmailAnalysis\BL\AnalysisService.cs:line 22 at WebAPI.Controllers.ValuesController.Get(PersonCredentials credentials) in D:\Development\DotNetProjects\GmailAnalysis\WebAPI\Controllers\ValuesController.cs:line 63 at lambda_method(Closure , Object , Object[] ) at System.Web.Http.Controllers.ReflectedHttpActionDescriptor.ActionExecutor.<>c__DisplayClass10.<GetExecutor>b__9(Object instance, Object[] methodParameters) at System.Web.Http.Controllers.ReflectedHttpActionDescriptor.ActionExecutor.Execute(Object instance, Object[] arguments) at System.Web.Http.Controllers.ReflectedHttpActionDescriptor.ExecuteAsync(HttpControllerContext controllerContext, IDictionary`2 arguments, CancellationToken cancellationToken) --- End of stack trace from previous location where exception was thrown --- at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) at System.Web.Http.Controllers.ApiControllerActionInvoker.<InvokeActionAsyncCore>d__0.MoveNext() --- End of stack trace from previous location where exception was thrown --- at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) at System.Web.Http.Controllers.ActionFilterResult.<ExecuteAsync>d__2.MoveNext() --- End of stack trace from previous location where exception was thrown --- at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) at System.Web.Http.Controllers.AuthenticationFilterResult.<ExecuteAsync>d__0.MoveNext() --- End of stack trace from previous location where exception was thrown --- at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task) at System.Web.Http.Dispatcher.HttpControllerDispatcher.<SendAsync>d__1.MoveNext()
    </StackTrace>
    <InnerException>
        <Message>An error has occurred.</Message>
        <ExceptionMessage>Access is denied</ExceptionMessage>
        <ExceptionType>System.Net.HttpListenerException</ExceptionType>
        <StackTrace>
        at Microsoft.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task) at Microsoft.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccess(Task task) at Google.Apis.Auth.OAuth2.GoogleWebAuthorizationBroker.<AuthorizeAsync>d__1.MoveNext() in c:\code\github\google-api-dotnet-client\Tools\Google.Apis.Release\bin\Release\1.9.2\default\Src\GoogleApis.Auth.DotNet4\OAuth2\GoogleWebAuthorizationBroker.cs:line 59
        </StackTrace>
    </InnerException>
</Error>

我将 client_secret.json 用于 Web 应用程序。谁能给我一个例子或指出正确的方向。

【问题讨论】:

  • 您实际存储 json 的物理路径是什么?
  • 它是 azure HostingEnvironment.MapPath("~/client_secret.json") 上 WebApi 项目的根,正如我在调试器中看到的,client_secret.json 不是问题
  • 尝试新的 FileDataStore(".",true);如果@DimaDaron 说的不起作用。
  • @DaImTo 我收到了同样的错误:(
  • 服务器上的物理路径是什么(不是web项目)?登录 Azure 门户并浏览到您的 Web 应用 >> 工具 >> Kudu >> 前往 >> 调试控制台 >> CMD >> 站点 >> wwwroot --- 然后查看 client_secret.json 是否真的存在。

标签: c# azure asp.net-web-api gmail-api google-api-dotnet-client


【解决方案1】:

这是一个有效的例子:

var filePath = HostingEnvironment.MapPath("~/build-info.json");
var fileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read);
string fileContent;
using (var reader = new StreamReader(fileStream))
{
    fileContent = reader.ReadToEnd();
}

我已在本地运行并部署到 Azure WebApp,它在这两个位置都有效。

所以类似下面的东西应该可以工作:

UserCredential credential;
var filePath = HostingEnvironment.MapPath("~/client_secret.json");
using (var stream = new FileStream(filePath , FileMode.Open, FileAccess.Read))
{
    credential = await GoogleWebAuthorizationBroker.AuthorizeAsync(GoogleClientSecrets.Load(stream).Secrets,
        new[] { GmailService.Scope.GmailReadonly },
        "user", CancellationToken.None);
}

【讨论】:

  • 嘿 @viperguynaz 谢谢你的帮助,但如果你使用 await,凭证不会返回。如果你改变这样的代码:credential =GoogleWebAuthorizationBroker.AuthorizeAsync(GoogleClientSecrets.Load(stream).Secrets, new[] { GmailService.Scope.GmailReadonly }, "user", CancellationToken.None).Result; 你会得到访问被拒绝错误
【解决方案2】:

在处理类似问题几天后,我发现这不是你应该这样做的:

credential = GoogleWebAuthorizationBroker.AuthorizeAsync(secret, Scopes, _credentials.Email, CancellationToken.None, dataStore).Result;

相反,您应该使用 FlowMetadata 方法进行身份验证,demonstrated here.

【讨论】:

    猜你喜欢
    • 2014-10-29
    • 2015-10-10
    • 2018-07-20
    • 1970-01-01
    • 1970-01-01
    • 2012-03-14
    • 2012-12-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多