質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.51%

  • Swift

    8410questions

    Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

  • iOS

    4487questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Twitter

    752questions

    Twitterは、140文字以内の「ツイート」と呼ばれる短文を投稿できるサービスです。Twitter上のほぼ全ての機能に対応するAPIが存在し、その関連サービスが多く公開されています。

TwitterにAVCaptureSessionで撮影した動画をアップロードできない

受付中

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 594

tarofess

score 118

iPhoneのカメラで15秒間動画を撮影し、その動画をTwitterにアップロードしようとすると「Request status was 413」とエラーが返ってきて動画をアップロードできません。
アップロードしている最中にログに「SLRequestBodyInputStream <SLRequestBodyInputStream: 0x1016e7e10> read:maxLength: Total 6692864 bytes streamed, total length is 15382475」というようなものが表示され続けています。(表示されるたびにTotal 6692864の数字が増えていっています)
Twitterにアップロードしている部分のコードは以下になります。

class TwitterSharer: NSObject {

    var videoPathURL: URL
    var account: ACAccount!
    let twitterUploadURL = URL(string: "https://upload.twitter.com/1.1/media/upload.json")
    let twitterStatusURL = URL(string: "https://api.twitter.com/1.1/statuses/update.json")

    init(url: URL) {
        videoPathURL = url
    }

    func authAccount(completion: @escaping (_ error: Bool)->Void) {
        let accountStore = ACAccountStore()
        let accountType:ACAccountType = accountStore.accountType(withAccountTypeIdentifier: ACAccountTypeIdentifierTwitter)
        accountStore.requestAccessToAccounts(with: accountType, options: nil) { (granted, error) -> Void in
            if error != nil {
                print("error! \(error)")
                completion(false)
                return
            }

            if !granted {
                print("error! Twitterアカウントの利用が許可されていません")
                completion(false)
                return
            }

            let accounts = accountStore.accounts(with: accountType) as! [ACAccount]
            if accounts.count == 0 {
                print("error! 設定画面からアカウントを設定してください")
                completion(false)
                return
            }

            print("アカウント取得完了")

            self.account = accounts.first!

            guard let mediaData = NSData(contentsOf: self.videoPathURL) else {
                completion(false)
                return
            }

            self.postMedia(tweet: "test", mediaData: mediaData as Data, fileSize: String(mediaData.length), completion: completion)
        }
    }

    func post(completion: @escaping (_ error: Bool)->Void) {
        authAccount(completion: completion)
    }

    private func postMedia(tweet: String, mediaData: Data, fileSize: String, completion: @escaping (_ error: Bool)->Void) {
        var json: [String: Any]!

        // INIT リクエスト
        uploadVideoInitRequest(fileSize: fileSize, success: { (_ responseData: Data)->Void in
            do {
                json = try JSONSerialization.jsonObject(with: responseData, options: .allowFragments) as! [String: Any]
            } catch {
                completion(false)
                return
            }

            let mediaIdString = json["media_id_string"] as! String

            // APPEND リクエスト
            self.uploadVideoAppendRequest(mediaData: mediaData, mediaIdString: mediaIdString, success: { () -> Void in

                // FINALIZE リクエスト
                self.uploadVideoFinalizeRequest(mediaIdString: mediaIdString, success: { (_ responseData: Data) -> Void in

                    let statusRequest = SLRequest(forServiceType: SLServiceTypeTwitter, requestMethod: .POST, url: self.twitterStatusURL, parameters: ["status" : tweet, "media_ids" : mediaIdString])

                    statusRequest?.account = self.account

                    // 動画をつけてツイート
                    statusRequest?.perform { (responseData, urlResponse, error) -> Void in
                        if error == nil {
                            completion(true)
                        } else {
                            completion(false)
                        }
                    }
                }, completion: { (_ success: Bool) -> Void in
                    if !success {
                        completion(false)
                    }
                })
            }, completion: { (_ success: Bool) -> Void in
                if !success {
                    completion(false)
                }
            })
        }, completion: { (_ success: Bool) -> Void in
            if !success {
                completion(false)
            }
        })
    }

    // INIT リクエスト
    private func uploadVideoInitRequest(fileSize: String, success: @escaping (_ responseData: Data)->Void, completion: @escaping (_ error: Bool)->Void) {
        let initParams: [String: Any] = ["command": "INIT", "media_type": "video/mp4", "total_bytes": fileSize]
        let initRequest = SLRequest(forServiceType: SLServiceTypeTwitter, requestMethod: .POST, url: self.twitterUploadURL, parameters: initParams)
        initRequest!.account = account
        initRequest!.perform { (responseData, urlResponse, error) -> Void in

            if error == nil {
                success(responseData!)
            } else {
                completion(false)
                return
            }
        }
    }

    // APPEND リクエスト
    private func uploadVideoAppendRequest(mediaData: Data, mediaIdString: String, success: @escaping () -> Void, completion: @escaping (_ error: Bool)->Void) {
        let appendParam: [NSString: Any] = ["command": "APPEND", "media_id": mediaIdString, "segment_index": "0"]
        let appendRequest = SLRequest(forServiceType: SLServiceTypeTwitter, requestMethod: .POST, url: self.twitterUploadURL, parameters: appendParam)
        appendRequest?.addMultipartData(mediaData, withName: "media", type: "video/mov", filename: nil)
        appendRequest?.account = account
        appendRequest?.perform { (responseData, urlResponse, error) -> Void in

            if urlResponse!.statusCode < 300 && urlResponse!.statusCode >= 200 {
                success()
            } else {
                completion(false)
                return
            }
        }
    }

    // FINALIZE リクエスト
    private func uploadVideoFinalizeRequest(mediaIdString: String, success: @escaping (_ responseData: Data) -> Void, completion: @escaping (_ error: Bool)->Void) {
        let finalizeParam: [NSString: Any] = ["command": "FINALIZE", "media_id": mediaIdString]
        let finalizeRequest = SLRequest(forServiceType: SLServiceTypeTwitter, requestMethod: .POST, url: self.twitterUploadURL, parameters: finalizeParam)
        finalizeRequest?.account = account
        finalizeRequest?.perform { (responseData, urlResponse, error) -> Void in

            if error == nil {
                success(responseData!)
            } else {
                completion(false)
                return
            }
        }
    }

}

試しに撮影する動画を軽くしようと思い、以下のようにAVCaptureSessionにAVCaptureSessionPresetLowを設定すると動画をTwitterにアップロードできるようになります。

func setUpCamera() {
        let devices = AVCaptureDeviceDiscoverySession(deviceTypes: [.builtInWideAngleCamera], mediaType: AVMediaTypeVideo, position: .front).devices

        do {
            let videoInput = try AVCaptureDeviceInput(device: devices?.first!) as AVCaptureDeviceInput
            captureSession.addInput(videoInput)
            let audioInput = try AVCaptureDeviceInput(device: audioDevice) as AVCaptureDeviceInput
            captureSession.addInput(audioInput);
        } catch {

        }
        captureSession.commitConfiguration()
        captureSession.addOutput(fileOutput)
        captureSession.sessionPreset = AVCaptureSessionPresetLow    //この行を追加するとアップロードできる
        captureSession.startRunning()
    }

しかし動画がかなり荒いです。
普通に撮影した状態で動画をTwitterにアップロードしたいのですが、どうすればいいでしょうか?
Twitterに動画をアップロードする際に画質が少し荒くなるのは仕方ないことなのでしょうか?
いろいろ調べて見たもののこれ以上どうすればいいかわからないので、どなたかわかる方がいれば教えていただきたいです。よろしくお願いします。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

0

原因はデータ量のようなので、動画を短くすれば良いはずです。
止まっているところの量を見る限り3つに分割でしょうか。
一度にどれだけのデータ量を上げられるのかを確認していますか?

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/10/09 22:29

    15Mまでと言う風に書いているのは確認しました。
    やはり動画を短くしたりするしかないのですね......。

    キャンセル

同じタグがついた質問を見る

  • Swift

    8410questions

    Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

  • iOS

    4487questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Twitter

    752questions

    Twitterは、140文字以内の「ツイート」と呼ばれる短文を投稿できるサービスです。Twitter上のほぼ全ての機能に対応するAPIが存在し、その関連サービスが多く公開されています。