iOS开发Swift

iOS 企业内部应用(In-House)

企业内部应用

企业内部应用,即只在企业部门和员工内部使用、不对外公开的应用。苹果提供了专门的In-House证书用来发布这种应用,可以分发给任意的手机,只要通过一个URL即可下载安装,不用上传到App Stroe审核。我把企业内部应用也叫做In-House应用。

In-House应用,有时需要根据部门需求进行版本的快速迭代,因为不需要App Store审核,所以可以做到随时修改,随时发布,节省了大量的时间。In-House证书还可以用于应用的内测分发。

必须具备的两个条件:

  • 企业开发者账号。99$的普通开发者账号不行,必须以企业的名义申请一个299$的企业开发者账号Apple Developer Enterprise Program
  • 带SSL证书的域名。企业内部应用需要把ipa文件上传到服务器,然后通过一个链接来下载安装,而苹果很重视安全性,要求这个链接的域名必须具有SSL证书,支持 https ,否则无法安装。

SSL证书其实并不是必需的,可以使用一些知名的云存储服务,比如亚马逊的AWS,阿里云等,这些大公司的云存储都支持Https,我用的就是AWS的S3云存储,但299$的企业开发者账号就避免不了。

准备的文件

  • ipa文件。
  • plist文件。名称必须与ipa文件一致,用于配置bundle id、版本号、ipa文件的URL、应用图标等。
  • @1x 和 @2x 的Icon。下载安装时显示应用图标。

打包

1. 创建发布证书(Production Certificates),选择In-House类型的,过程我就不赘述了,和其他证书一样。

2. 创建配置文件(Distribution Provisioning Profiles)

3. 在Xcode选择对应的Code Signing 和 Provisioning Profile, Archive

4. 导出 ipa 文件

plist 文件

Xcode 5 及其以前打包In-House应用会一起生成ipaplist文件,但Xcode 6 以后就只有ipa文件了,所以要手动生成 plist文件。在plist文件中输入ipa的URL、安装时显示的 icon 的url、bundle id、版本号、安装前的提示信息。
文件格式如下:

    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
    <plist version="1.0">
    <dict>
        <key>items</key>
        <array>
            <dict>
                <key>assets</key>
                <array>
                    <dict>
                        <key>kind</key>
                        <string>software-package</string>
                        <key>url</key>
                        <string>[INSERT THE URL FOR YOUR IPA HERE. e.g : https://s3-us-west-2.amazonaws.com/folder/appName-version.ipa]</string>
                    </dict>
                    <dict>
                        <key>kind</key>
                        <string>full-size-image</string>
                        <key>needs-shine</key>
                        <true/>
                        <key>url</key>
                        <string>[INSERT THE URL FOR INSTALLATION @2x ICON HERE. e.g : https://s3-us-west-2.amazonaws.com/folder/images/Icon@2x.png]</string>
                    </dict>
                    <dict>
                        <key>kind</key>
                        <string>display-image</string>
                        <key>needs-shine</key>
                        <true/>
                        <key>url</key>
                        <string>[INSERT THE URL FOR INSTALLATION ICON HERE. e.g : https://s3-us-west-2.amazonaws.com/folder/images/Icon.png]</string>
                    </dict>
                </array>
                <key>metadata</key>
                <dict>
                    <key>bundle-identifier</key>
                    <string>[INSERT BUNDLE ID HERE]</string>
                    <key>bundle-version</key>
                    <string>[INSERT VERSION HERE]</string>
                    <key>kind</key>
                    <string>software</string>
                    <key>title</key>
                    <string>[INSERT APP TITLE HERE. The Title will present to the user installing the app]</string>
                </dict>
            </dict>
        </array>
    </dict>
    </plist>

发布与安装

ipa、配置好的plist 文件和图标一起上传AWS的S3云存储上即可。

安装

iOS的企业内部应用是通过访问plist文件来安装的,因为plist文件中包含了对应的ipa文件和图标的URL,iPhone会根据URL自动下载并安装应用程序。

在iPhone的Safari浏览器中输入:

itms-services://?action=download-manifest&url=https://s3-us-west-2.amazonaws.com/folder/appName-version.plist

首先会询问是否打开要打开链接,点击“打开”

然后询问是否要安装App,点击“安装”

自动更新

为了避免每次发布后都需要通知别人更新App的麻烦事,自动更新是必备的。与后台沟通,设计一个更新接口GET /updates

Response

Status-Code: 200 OK

{
    "data": {
        "update" : true,
        "lastest" : "1.0.1",
        "url" : "itms-services://?action=download-manifest&url=https://s3-us-west-2.amazonaws.com/folder/appName-1.0.1.plist"
    },
    "error": {}
}

App一启动时,调用GET /updates接口传递平台参数和当前版本号给后台进行检查,后台判断当前版本是否为最新版,如果不是最新版,则返回最新版本号和对应的下载链接,然后用浏览器打开返回的URL进行安装即可。

func upgrade()
    {
        let params = [
            "platform" : "ios", 
            "version" : CKAppInfo.releaseVersion()
        ]
        ObjectRequest(URLRequest: UpdateRouter.CheckUpdate(params)).load(
            successHandler: { (promotion: Promotion?) -> Void in
                SWLog(promotion)
                
                if promotion?.update == true { // different version
                    
                    let alert = DVAlertController(title: "Upgrade", message: "update to latest version", preferredStyle: UIAlertControllerStyle.Alert)
                    alert.addAction(UIAlertAction(title: "Upgrade", style: UIAlertActionStyle.Default, handler: { (_) -> Void in
                        let url = NSURL(string:promotion!.url!)
                        UIApplication.sharedApplication().openURL(url!)
                    }))
                    
                    alert.show(animated: true, completion: nil)
                    SWLog("different version")
                }
            }, failHandler: { (error) -> Void in
                self.creatAlert("checkout upgrade failed")
        })
    }