如何解决YouTube API查询中的401错误?

时间:2019-06-03 20:24:52

标签: swift youtube-api youtube-data-api

[问题已解决,实际上我还不能选择答案作为解决方案。]

我正在使用YouTube数据API开发移动应用的视频上传功能;但我在执行查询时始终遇到401错误。

我正在使用Firebase向我提供Google Developer的OAuth和API密钥。


我试图查看Firebase提供的PLIST文件是否发生了更改,甚至还有其他更改。

我试图通过GIDSignIn.sharedInstance().disconnect()重设iOS模拟器,卸载应用程序并终止我的帐户对应用程序的权限。

我试图与其他StackOverFlow问题一起查看,但是它们都与其他语言有关,甚至与“原始” HTTP请求有关。


在我的AppDelegate文件中,我具有与Firebase和Google连接的那些行:

FirebaseApp.configure()
GIDSignIn.sharedInstance().clientID = FirebaseApp.app()?.options.clientID

和这种方法(还有一个非常相似的方法,我将不在这里显示):

func application( _ application: UIApplication, open url: URL, sourceApplication: String?, annotation: Any ) -> Bool {
        return GIDSignIn.sharedInstance().handle( url,
                sourceApplication: sourceApplication,
                annotation: annotation )
}

在与控制器相关的类中,我具有创建查询的方法:

private func uploadVideo( withVideoObject video: GTLRYouTube_Video, resumeUploadLocationURL locationURL: URL? ) throws {
    let query = try uploadQuery( _video: video, resumeUploadLocationURL: locationURL )!

    // [ progress related feedback ]

    YouTubeUploadController.uploadFileTicket = self.service.executeQuery( query,
            delegate: self.controller!,
            didFinish: #selector( self.controller!.displayResultWithTicket(ticket:finishedWithObject:error:) ) )
}
private func uploadQuery( _video: GTLRYouTube_Video, resumeUploadLocationURL locationURL: URL? ) throws -> GTLRYouTubeQuery_VideosInsert? {
    let fileToUploadURL = URL( fileURLWithPath: "\(self.controller!.uploadPathField)" )
    if !( try! fileToUploadURL.checkPromisedItemIsReachable() ) {
        throw YouTubeCustomError.init( _kind: .fileNotFound )
    }
    // Get a file handle for the upload data.
    let uploadParameters = prepareUploadParameters( fileToUploadURL: fileToUploadURL, locationURL: locationURL )

    return GTLRYouTubeQuery_VideosInsert.query( withObject: _video, part: "snippet,status", uploadParameters: uploadParameters )
}

它通过displayResultWithTicket(ticket:finishedWithObject:error:)发送错误:
此操作无法完成。(com.google.HTTPStatus错误401。)

没有给定的失败原因,也没有恢复建议。

谢谢。


响应正文:
这是GTMSessionFetcher的HTTP回答日志,用于尝试上传:

Response body: (178 bytes)
{
  "error" : {
    "message" : "Login Required",
    "errors" : [
      {
        "message" : "Login Required",
        "locationType" : "header",
        "reason" : "required",
        "domain" : "global",
        "location" : "Authorization"
      }
    ],
    "code" : 401
  }
}

请求标头

Request: POST https://www.googleapis.com/resumable/upload/youtube/v3/videos?part=snippet%2Cstatus&prettyPrint=false
Request headers:
  Accept: application/json
  Cache-Control: no-cache
  Content-Type: application/json; charset=utf-8
  User-Agent: org.cocoapods.GoogleAPIClientForREST/1.3.8 google-api-objc-client/3.0 iPhone/12.2 hw/sim (gzip) (GTMSUF/1)
  X-Goog-Upload-Command: start
  X-Goog-Upload-Content-Length: 2614231
  X-Goog-Upload-Content-Type: video/mp4
  X-Goog-Upload-Protocol: resumable

1 个答案:

答案 0 :(得分:1)

因此,在我的情况下,我的GIDSignInDelegate signIn方法缺少GTRLService授权者影响:
service.authorizer = user.authentication.fetcherAuthorizer()

我认为它也可以用其他方法完成,但这是初始化它的核心。


我通过ObjC写访问器做到了这一点,因为我没有找到create/obtain a GTRLService instance in Swift的方法,并且我使用ObjC单例类来获取它,而且我不确定是否可以访问GTRLService实例方法。如果是在ObjC中初始化的,则为Swift。

+ (void)setSharedInstanceAuthorizer:(id <GTMFetcherAuthorizationProtocol>)_authorizer {
    if (sharedInstance == nil) {
        sharedInstance = [GTLRYouTubeService new];
    }
    sharedInstance.authorizer = _authorizer;
};

如果这不能解决问题,则问题注释中的@stvar答案可能确实有用。