ios的python编程软件_ios为什么有的app没有自动化

ios的python编程软件_ios为什么有的app没有自动化这段时间刚刚学习了一段时间的Python,加上自己是做iOS开发的,就想着用Python来做一个自动化打包,可以自动完成打包,上传到蒲公英,并且发送邮箱给测试人员. 一是可以减少打包功夫,二来可以练练手,结合自己的工作来输出一点东西.废话不多说,直接上代码… 就是使用xco…

ios的python编程软件_ios为什么有的app没有自动化

前言

这段时间刚刚学习了一段时间的Python,加上自己是做iOS开发的,就想着用Python来做一个自动化打包,可以自动完成打包,上传到蒲公英,并且发送邮箱给测试人员.

一是可以减少打包功夫,二来可以练练手,结合自己的工作来输出一点东西.废话不多说,直接上代码…

原理

就是使用xcodebuild来控制Xcode进行一系列的操作,从而完成打包的操作.

ios的python编程软件_ios为什么有的app没有自动化

为什么要做这个?

在我们日常开发的时候,特别是在内部测试的时间,有可能需要频繁的打包,打包的工作比较繁琐,需要等待点击下一步,选择之类,影响了开发的节奏.(开玩笑,我能有啥节奏…), 为什么不能直接运行,然后完成所有的操作呢?

思路:

从网上查找了一些关于xcodebuild来打包的资料,从而得到:

  1. 找到对应的项目
  2. clean项目
  3. archive项目
  4. export IPA
  5. 上传蒲公英
  6. 发送邮件
  7. 收工

思路有了,动手起来.

运行环境

Python, Xcode

这些需要大家直接去搭建好环境…

准备工作

  1. 下载安装pycharm(这只是我开发Python的工具而已,大家可以根据自己喜欢的来选择)
  2. 注册并认证蒲公英(不认证的话,是不能上传的)
  3. 邮箱开启POP3/SMTP服务(我使用的是QQ邮箱),记录下16位授权码
  4. 一个ExportOptions.plist文件, 这个下面会解释为什么需要还有怎么生成!
  5. 一份iOS项目代码→_→

完整代码

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# @Time : 2018/11/14 11:04 AM
# @Author : liangk
# @Site :
# @File : auto_archive_ios.py
# @Software: PyCharm


import os
import requests
import webbrowser
import subprocess
import time
import smtplib
from email.mime.text import MIMEText
from email import encoders
from email.header import Header
from email.utils import parseaddr, formataddr

project_name = 'TestArchive'    # 项目名称
archive_workspace_path = '/Users/用户/Desktop/TestArchive'    # 项目路径
export_directory = 'archive'    # 输出的文件夹
ipa_download_url = 'https://www.pgyer.com/XXX' #蒲公英的APP地址

# 蒲公英账号USER_KEY、API_KEY
USER_KEY = 'XXXXXXXXXXXXXXXXXXXX'
API_KEY = 'XXXXXXXXXXXXXXXXXXXX'

from_address = 'XXXXXXXXXXXXXXXXXXXX@qq.com'    # 发送人的地址
password = 'XXXXXXXXXXXXXXXXXXXX'  # 邮箱密码换成他提供的16位授权码
to_address = 'XXXXXXXXXXXXXXXXXXXX@qq.com'    # 收件人地址,可以是多个的
smtp_server = 'smtp.qq.com'    # 因为我是使用QQ邮箱..


class AutoArchive(object):
    """自动打包并上传到蒲公英,发邮件通知"""

    def __init__(self):
        pass

    def clean(self):
        print("\n\n===========开始clean操作===========")
        start = time.time()
        clean_command = 'xcodebuild clean -workspace %s/%s.xcworkspace -scheme %s -configuration Release' % (
            archive_workspace_path, project_name, project_name)
        clean_command_run = subprocess.Popen(clean_command, shell=True)
        clean_command_run.wait()
        end = time.time()
        # Code码
        clean_result_code = clean_command_run.returncode
        if clean_result_code != 0:
            print("=======clean失败,用时:%.2f秒=======" % (end - start))
        else:
            print("=======clean成功,用时:%.2f秒=======" % (end - start))
            self.archive()

    def archive(self):
        print("\n\n===========开始archive操作===========")

        # 删除之前的文件
        subprocess.call(['rm', '-rf', '%s/%s' % (archive_workspace_path, export_directory)])
        time.sleep(1)
        # 创建文件夹存放打包文件
        subprocess.call(['mkdir', '-p', '%s/%s' % (archive_workspace_path, export_directory)])
        time.sleep(1)

        start = time.time()
        archive_command = 'xcodebuild archive -workspace %s/%s.xcworkspace -scheme %s -configuration Release -archivePath %s/%s' % (
            archive_workspace_path, project_name, project_name, archive_workspace_path, export_directory)
        archive_command_run = subprocess.Popen(archive_command, shell=True)
        archive_command_run.wait()
        end = time.time()
        # Code码
        archive_result_code = archive_command_run.returncode
        if archive_result_code != 0:
            print("=======archive失败,用时:%.2f秒=======" % (end - start))
        else:
            print("=======archive成功,用时:%.2f秒=======" % (end - start))
            # 导出IPA
            self.export()

    def export(self):
        print("\n\n===========开始export操作===========")
        print("\n\n==========请你耐心等待一会~===========")
        start = time.time()
        # export_command = 'xcodebuild -exportArchive -archivePath /Users/liangk/Desktop/TestArchive/myArchivePath.xcarchive -exportPath /Users/liangk/Desktop/TestArchive/out -exportOptionsPlist /Users/liangk/Desktop/TestArchive/ExportOptions.plist'
        export_command = 'xcodebuild -exportArchive -archivePath %s/%s.xcarchive -exportPath %s/%s -exportOptionsPlist %s/ExportOptions.plist' % (
            archive_workspace_path, export_directory, archive_workspace_path, export_directory, archive_workspace_path)
        export_command_run = subprocess.Popen(export_command, shell=True)
        export_command_run.wait()
        end = time.time()
        # Code码
        export_result_code = export_command_run.returncode
        if export_result_code != 0:
            print("=======导出IPA失败,用时:%.2f秒=======" % (end - start))
        else:
            print("=======导出IPA成功,用时:%.2f秒=======" % (end - start))
            # 删除archive.xcarchive文件
            subprocess.call(['rm', '-rf', '%s/%s.xcarchive' % (archive_workspace_path, export_directory)])
            self.upload('%s/%s/%s.ipa' % (archive_workspace_path, export_directory, project_name))

    def upload(self, ipa_path):
        print("\n\n===========开始上传蒲公英操作===========")
        if ipa_path:
            # https://www.pgyer.com/doc/api 具体参数大家可以进去里面查看,
            url = 'http://www.pgyer.com/apiv1/app/upload'
            data = {
                'uKey': USER_KEY,
                '_api_key': API_KEY,
                'installType': '1',
                'updateDescription': description
            }
            files = {'file': open(ipa_path, 'rb')}
            r = requests.post(url, data=data, files=files)
            if r.status_code == 200:
            	# 是否需要打开浏览器
                # self.open_browser(self)
                self.send_email()
        else:
            print("\n\n===========没有找到对应的ipa===========")
            return

 @staticmethod
    def open_browser(self):
        webbrowser.open(ipa_download_url, new=1, autoraise=True)

 @staticmethod
    def _format_address(self, s):
        name, address = parseaddr(s)
        return formataddr((Header(name, 'utf-8').encode(), address))

    def send_email(self):
        # https://www.pgyer.com/XXX app地址
        # 只是单纯的发了一个文本邮箱,具体的发附件和图片大家可以自己去补充
        msg = MIMEText('<html><body><h1>Hello</h1>' +
                       '<p>╮(╯_╰)╭<a href="https://www.pgyer.com/XXX">应用已更新,请下载测试</a>╮(╯_╰)╭</p>' +
                       '<p>蒲公英的更新会有延迟,具体版本时间以邮件时间为准</p>' +
                       '</body></html>', 'html', 'utf-8')
        msg['From'] = self._format_address(self, 'iOS开发团队 <%s>' % from_address)
        msg['Subject'] = Header('来自iOS开发团队的问候……', 'utf-8').encode()
        server = smtplib.SMTP(smtp_server, 25)  # SMTP协议默认端口是25
        server.set_debuglevel(1)
        server.login(from_address, password)
        server.sendmail(from_address, [to_address], msg.as_string())
        server.quit()
        print("===========邮件发送成功===========")


if __name__ == '__main__':
    description = input("请输入内容:")
    archive = AutoArchive()
    archive.clean()

关于ExportOptions.plist文件

因为 Xcode 9+ 默认不允许访问钥匙串的内容,必须要设置 allowProvisioningUpdates 才会允许,Python的Xcode插件目前无法支持此项完成打包流程。

解决步骤如下:

1、手动Xcode10打包,导出ExportOptions.plist文件;

2、编辑ExportOptions.plist文件,配置 provisioningProfiles 对应填入Bundle identifier及证书关联配置文件(打包时自动匹配或手动填入证书,provisioningProfiles需配置的必填信息可自动生成);

3、提供ExportOptions.plist文件路径供Python脚本调用(详请参看Python脚本代码)。

具体的内容

<?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>compileBitcode</key>//是否编译bitcode
    <true/>
    <key>method</key>
    <string>ad-hoc</string>/
    <key>provisioningProfiles</key>
    <dict>
        <key>文件bundle id</key>
        <string>Adhoc_ID</string>
    </dict>
    <key>signingCertificate</key>//证书签名
    <string>这里填证书签名</string>
    <key>signingStyle</key>
    <string>manual</string>
    <key>stripSwiftSymbols</key>
    <true/>
    <key>teamID</key>
    <string>AANCCUK4M3</string>//TeamID
    <key>thinning</key>
    <string>&lt;none&gt;</string>
</dict>
</plist>

分析

xcodebuild archive -workspace XXX.xcworkspace -scheme XXX -configuration Release -archivePath XXX CONFIGURATION_BUILD_DIR ./dir ODE_SIGN_IDENTITY=证书 PROVISIONING_PROFILE=描述文件UUID
文件 说明
-workspace XXX.xcworkspace XXX.xcworkspace需要编译工程的工作空间名称,如果工程不是.xcworkspace的,可以不需要-workspace XXX.xcworkspace这段话
-scheme XXX XXX是工程名称,-scheme XXX是指定构建工程的名称
-configuration Release 填入打包的方式是Debug或Release,就跟在Xcode中编译前需要在Edit scheme的Build configuration中选择打出来的包是Debug还是Release包一样,-configuration就是配置编译的Build configuration
-archivePath XXX 配置生成.xcarchive的路径,
ODE_SIGN_IDENTITY=证书 配置打包的指定证书,如果该工程的Xcode已经配置好了证书,那么不加入这段话也可以,打包出来的证书就是Xcode中配置好的。
PROVISIONING_PROFILE=描述文件UUID 配置打包的描述文件,同上,Xcode已经配置好了就不用在填入这段话了
CONFIGURATION_BUILD_DIR 配置编译文件的输出路径,如果需要用到.xcarchive文件内部的dSYM等文件,可以使用改字段指定输出路径。

问题一

ios的python编程软件_ios为什么有的app没有自动化

配置一下compileBicode=NO即可

ios的python编程软件_ios为什么有的app没有自动化

感谢树下敲代码的超人

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/12903.html

(0)

相关推荐

  • reactnative滚动吸顶_vue滚动

    reactnative滚动吸顶_vue滚动这篇文章是三天前写就的,有大佬给我提了一些修改意见,我觉得这个意见确实中肯。所以就有了这个升级的修改版本。代码同步更新到 GitHub 了。 我入职第二家公司接到的第一个需求就是修复之前外包做的滚动吸顶效果。我当时很纳闷为何一个滚动吸顶会有 bug,后来我查看代码才发现直接用的…

    2023-07-18
    125
  • bindColumn、bindParam与bindValue的区别

    bindColumn、bindParam与bindValue的区别https://www.cnblogs.com/mrxy/p/8334287.html

    2023-02-10
    133
  • 以PyCharm设置解释器为中心的使用方法

    以PyCharm设置解释器为中心的使用方法PyCharm是在Python开发者中非常受欢迎的IDE,它具有许多功能强大的工具,使得开发者可以在其中更加高效地进行编码。而在PyCharm中设置解释器非常重要,因为解释器是用于执行Python代码的程序,PyCharm可以帮助用户创建和管理解释器,同时也支持多个解释器的配置。本文将介绍在PyCharm中如何设置解释器,以及如何为不同的项目设置不同的解释器。

    2024-05-20
    61
  • 轻松实现Python CGI页面的交互性

    轻松实现Python CGI页面的交互性CGI,全称通用网关接口(Common Gateway Interface),指的是Web服务器提供的一种机制,可以同外部程序交换数据。在Web服务器接收到客户端(如浏览器)发送的请求时,如果该请求是由CGI程序处理的,则Web服务器会将相关数据传递给指定的CGI程序,CGI程序接收到这些数据后,会处理这些数据,然后生成HTML页面返回给Web服务器,最终Web服务器将HTML页面返回给请求的客户端。

    2024-02-11
    86
  • sql-mode_mysql 42000

    sql-mode_mysql 42000处理MySQL数据库工作中遇到关于sql_mode的情况,特学习记录一下 废话不多说直接上错误提示,哈哈哈 bug开始!!!! INSERT INTO mmb_user (openid,mobile,

    2023-04-29
    146
  • Python Extend()方法:扩展列表的元素

    Python Extend()方法:扩展列表的元素在Python中,列表是一种很常见的数据类型。列表是可变的,这就意味着可以对列表进行添加、删除和修改操作。Extend()方法是Python列表中的一个非常重要的方法。这个方法用于向列表中添加一些新的元素,扩展列表的长度。

    2023-12-19
    98
  • TDengine 如何做到客户端高可用?

    TDengine 如何做到客户端高可用?小 T 导读 :经常有用户在 TDengine 的社区上递交标签为「help wanted」的问题。这些问题大都不是 Bug,只是因为不熟悉或者不了解 TDengine 的机制而让用户感到困惑的使用…

    2023-04-12
    153
  • Python Tkinter Spinbox – 创建带步进按钮的可选数字项

    Python Tkinter Spinbox – 创建带步进按钮的可选数字项Tkinter是Python的标准GUI库,可以轻松创建窗口、按钮、文本框、菜单等各种组件,提供用户与程序交互的界面。使用Tkinter开发GUI应用程序比较简单,只需要导入Tkinter模块,创建一个顶层窗口,然后在窗口对象上添加各种组件即可。

    2024-02-08
    86

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注