【Apple】歴代 Apple WWDC やイベントの動画を一括にダウンロードする方法 WWDC24 も追加更新中

いわゆる ffmpeg を使った シェルスクリプトです。

以下のようなものです。


ffmpeg -i https://devstreaming-cdn.apple.com/videos/wwdc/2020/10691/2/A92788CB-81ED-4CCF-B6B1-4DD7A1F3E87D/hvc_2160p_16800/prog_index.m3u8 -c copy "Session - 10691 temp.mp4"
ffmpeg -i https://devstreaming-cdn.apple.com/videos/wwdc/2020/10691/2/A92788CB-81ED-4CCF-B6B1-4DD7A1F3E87D/audio_english_192/prog_index.m3u8 -c copy "Session - 10691 temp.aac"
ffmpeg -i "Session - 10691 temp.mp4" -i "Session - 10691 temp.aac" -c copy "Session 10691 - Monday@WWDC.mp4"
rm "Session - 10691 temp.mp4"
rm "Session - 10691 temp.aac"

しかし、なぜか QuickPlayer で映像が見えず音声だけしか再生されない。

VLCなど別の動画プレーヤーではフツーに見れます。

現在開催中の WWDC24 の動画分も更新中の様子。

まとめてみたい人は便利かもしれません。

👉 dmthomas/AppleVideoDownloadScripts: Script to download higher resolutions of Apple event videos using ffmpeg hatena-bookmark


【Swift】ファイルやディレクトリ操作するための extension をまずは作った

ファイルの操作がなんか混乱してストレス。

直感的にまずは作っておく。

extension URL {
var exists: Bool {
isFile || isDirectory
}
var isFile: Bool {
(try? resourceValues(forKeys: [.isRegularFileKey]))?.isRegularFile == true
}
var isDirectory: Bool {
(try? resourceValues(forKeys: [.isDirectoryKey]))?.isDirectory == true
}
var directoryEntryCount: Int {
(try? resourceValues(forKeys: [.directoryEntryCountKey]).directoryEntryCount) ?? 0
}
var fileSize: String {
let size = (try? resourceValues(forKeys: [.fileSizeKey]).fileSize) ?? 0
return size.formatted(.byteCount(style: .file))
}
var modified: Date {
try! resourceValues(forKeys: [.contentModificationDateKey]).contentModificationDate!
}
func shortPath(percentEncoded: Bool = true) -> String {
path(percentEncoded: percentEncoded)
.replacingOccurrences(
of: URL.homeDirectory.path(percentEncoded: percentEncoded),
with: "/HOME/"
)
}
}
extension FileManager {
private func contents(directory url: URL) -> [URL] {
(try? contentsOfDirectory(at: url, includingPropertiesForKeys: nil, options: [])) ?? []
}
func showContents(_ url: URL = .documentsDirectory) {
print(
String(format: "%@ %@", url.shortPath(), url.isFile ? "[\(url.fileSize)]" : "")
)
if url.isDirectory {
contents(directory: url)
.sorted(by: {
let lr = [$0, $1].map {
($0.path().components(separatedBy: "/").dropLast().joined(), $0.path())
}
return lr[0] < lr[1]
})
.forEach { content in
showContents(content)
}
}
}
}
// ex.
let manager = FileManager.default
let documents = URL.documentsDirectory
print(documents.shortPath())
// /HOME/Documents/
manager.showContents(documents)
// /HOME/Documents/
// /HOME/Documents/another/
// /HOME/Documents/another/copied.txt [6 bytes]
// /HOME/Documents/some/
// /HOME/Documents/some/copied.txt [6 bytes]
// /HOME/Documents/some/moved.txt [6 bytes]
print()
print(documents.exists)
// true
print(documents.isDirectory)
// true
print(documents.isFile)
// false
print(documents.isFileURL) // built-in
// true
print(documents.directoryEntryCount)
// 2
print(documents.fileSize)
// Zero kB
print(documents.modified)
// 2024-06-08 11:33:07 +0000
print(documents.shortPath(percentEncoded: true))
// /HOME/Documents/
let dummy = documents.appending(component: "dummy")
print(dummy.shortPath())
// /HOME/Documents/dummy
manager.showContents(dummy)
// /HOME/Documents/dummy
print(dummy.exists)
// false
print(dummy.isDirectory)
// false
print(dummy.isFile)
// false
print(documents.isFileURL) // built-in
// true
print(dummy.directoryEntryCount)
// 0
print(dummy.fileSize)
// Zero kB
//print(dummy.modified)
// crash


FileManager.default.showContents(.temporaryDirectory)

// /HOME/tmp/ 
// /HOME/tmp/.DS_Store [6 kB]
// /HOME/tmp/CFNetworkDownload_1Ji1Ym.tmp [1.8 MB]
// /HOME/tmp/CFNetworkDownload_3ravkG.tmp [1.8 MB]
// /HOME/tmp/CFNetworkDownload_ArezWZ.tmp [1.8 MB]
// /HOME/tmp/CFNetworkDownload_Q2vc3J.tmp [920 kB]
// /HOME/tmp/CFNetworkDownload_WnW3m7.tmp [1.8 MB]
// /HOME/tmp/CFNetworkDownload_s4wpzr.tmp [1.8 MB]

既存の URL、FileManager にぶつからないように作りたい。

あくまで、補助、簡素化。

使い勝手で Gist 更新していきたいです。

いまどきの スマホOS は必要以上の高機能で、

開発者向け SDK や public API までも初心者泣かせの仕様です。


 

🤔 参考

👉 【Swift】FileManager を使いたい hatena-bookmark
👉 【Swift】そのディレクトリ内を再帰的に確認する hatena-bookmark
👉 【Swift】ファイルやディレクトリのパスが長すぎていやだ - URL.shortPath() hatena-bookmark
👉 【Swift】URL で特定のディレクトリやファイルを指す hatena-bookmark
👉 【Swift】その URL が ファイル なのか ディレクトリ なのか 存在しないのか hatena-bookmark


【Swift】ファイルやディレクトリのパスが長すぎていやだ - URL.shortPath()

ファイルやディレクトリを操作していると、

パスの確認をしますよね。

例えば、


let documents = URL.documentsDirectory

としておいて、


print(documents.path)

あれ、Deprecated ですか。


print(documents.path())

として表示すると、


/Users/me/Library/Developer/Xcode/UserData/Previews/Simulator Devices/AA651DE-1A5C-4AA0-80D0-ADC0FF5AA467/data/Containers/Data/Application/35DFAAB4-E576-4318-9F17-DEC9F0DA259A/Documents

長い、長すぎる。

なんせ URL.homeDirectory までが長すぎる。

ファイル操作ごときが、

なぜか辛く感じるのは、

これのせいでしょうか。

短縮形のエクステンソン作ります。

ただの置換です。


extension URL {
  func shortPath(percentEncoded: Bool = true) -> String {
    path(percentEncoded: percentEncoded)
      .replacingOccurrences(
        of: URL.homeDirectory.path(percentEncoded: percentEncoded),
        with: "/HOME/"
      )
  }
}

いくつかの URL で確認。


print(documents.shortPath())
// /HOME/Documents/

print(
  documents
    .appending(component: "Documents and Settings/")
    .shortPath()
)
// /HOME/Documents/Documents%20and%20Settings/

print(
  documents
    .appending(component: "Documents and Settings", directoryHint: .isDirectory)
    .shortPath(percentEncoded: false)
)
// /HOME/Documents/Documents and Settings/

これで、ログウィンドウがスッキリしました!

ごときが !

 

🤔 参考

👉 【Swift】URL で特定のディレクトリやファイルを指す hatena-bookmark