Swift脚本之自动更新Xcode工程BundleVersion

Swift作为一种全能语言,除了能写客户端之外其实还能用来写Server甚至还能用来写脚本的。更新Xcode工程的BudleVersion这件事情python和shell脚本都能做,并且跟Xcode的兼容性也不错。但作为iOS开发来说尝试一下用Swift来做这件事听起来也比较合乎情理了。

下面上一段Swift代码:

import Foundation

private typealias taskData = (pipe: Pipe, task: Process)

public func updateBundleVersion(_ plistPath: String) {


    print("plistPath: \(plistPath)")

    let fileManager = FileManager.default

    guard fileManager.fileExists(atPath: plistPath) else {
        print("⚠️  请输入Info.plist文件路径")
        return
    }
    
    let taskTuple = shellPipe(launchPath: "/usr/bin/git", args: ["rev-list", "head"])
    let sortTask = shellPipe(launchPath: "/usr/bin/sort", inputTask: taskTuple)
    let commitNumberTask = shellPipe(launchPath: "/usr/bin/wc", inputTask: sortTask, args: ["-l"])
    
    guard let commitNumber = shell(task: commitNumberTask) else {
        return
    }
    
    print("commitNumber:\(commitNumber)")
    
    let updateBundleVersionTask = shellPipe(launchPath: "/usr/libexec/PlistBuddy",
                                            args: ["-c","Set :CFBundleVersion \(commitNumber)", "\(plistPath)"])
    shell(task: updateBundleVersionTask)
}

private func shellPipe(launchPath: String, inputTask: taskData? = nil, args: [String]? = nil) -> taskData {
    let task = Process()
    task.launchPath = launchPath
    
    var command = launchPath
    
    if let arguments = args {
        task.arguments = arguments
        command += arguments.flatMap{ $0 }.joined(separator: " ")
    }
    print("⚠️  Executing command: \(command)")
    
    if let inputTaskTuple = inputTask {
       task.standardInput = inputTaskTuple.pipe
    }
    
    let outPutPipe = Pipe()
    task.standardOutput = outPutPipe
    
    if let inputTaskTuple = inputTask {
        inputTaskTuple.task.launch()
    }
    
    return (outPutPipe, task)
}

@discardableResult
private func shell(task: taskData) -> String? {
    task.task.launch()
    
    let data = task.pipe.fileHandleForReading.readDataToEndOfFile()
    let output: String? = String(data: data, encoding: .utf8)
    
    task.task.waitUntilExit()
    
    let status = task.task.terminationStatus
    if status == 0 {
        print("✅  Success")
    } else {
        print("❗️ Error: \(output ?? "")")
    }
    
    if let result = output {
        return result.trimmingCharacters(in: NSCharacterSet.whitespacesAndNewlines)
    } else {
        return nil
    }
}

//获取命令行参数
let arguments = CommandLine.arguments

print("⚠️  arguments: \(arguments)")

if arguments.count >= 2 {
    let infoPlistFile = arguments[1]
    updateBundleVersion(infoPlistFile)
} else {
    print("❗️  缺少Info.plist路径输入参数!")
}

这段代码做的事情就是获取git当前的commit数量并更新到Info.plist的BundleVersion中,这段Swift代码无法直接在Xcode中运行,可以在命令行中调用,比如把这段代码保存为updateBundleVersion.swift:

swift updateBundleVersion.swift ./Info.plist

我尝试了一下按这种方式放在Xcode 的Build Phases中调用的话会报错:

1512714581081-image.png

后来在网上有人将Swift源码放到Build Phases中来直接调用

1512716641796-image.png

虽然这样开起来比较丑,但毕竟算是在Swift脚本之路上迈出了坚实的一步了。

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 14,893评论 4 61
  • 漫长的公路线 没有尽头 没有目的 你看着重复的风景 却不敢闭上眼睛
    叶子e阅读 230评论 1 2
  • 简述 iOS系统提供了两套绘图框架:<strong>UIBezierPath</strong> 和 <strong...
    iLees阅读 446评论 5 5
  • 早上搬东西去新房,因为郑儿一直在哭闹,郑儿奶奶哄不了,因此我借故和郑先生吵了。放下东西,抱着郑儿回到家里。 下午去...
    简宁思静阅读 219评论 0 0
  • 有时辰,我们发明,我们的生命并非完全把握在本身的手里,连旅游城市不警惕碰到的不测,能确保安全呢?近日,让人感应不成...
    东莞康泰旅游阅读 456评论 0 0

友情链接更多精彩内容