[问题已解决,实际上我还不能选择答案作为解决方案。]
我正在使用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
答案 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答案可能确实有用。