相信大部分程序员都对开发环境的工具都有一些特殊的执念(???),如果在自己不习惯的环境中工作完全无法开展,怎么这个工具没有那个字体难受,我本人就是,换了新的 Mac 电脑后如何快速恢复到之前的开发工具呢?

开发工具包括 App 和命令行工具。用移动硬盘+时间机器自动备份当然能够完美地解决备份和恢复问题,不过接下来讨论的是一种用 Python 实现的“便宜的”方案。

App 信息备份

对于 App Store 安装的 App,通过 Apple ID 可以方便地重新下载,而对于非 App Store 安装的小众 App(如通过 Git Release)怎么快速恢复?通过 homebrew、gem 安装的依赖包和小工具呢?如果不是高频使用的 App,我们大多也记不住名字,只有到使用时才能想起它。下面介绍我使用 Python 编写程序备份 App 信息,并定期自动发送到邮件。

  • 实现思路

    Mac OS 的 /Applications 文件夹中,保存了从 App Store 下载的 App,我们其他途径得到的 .dmg 等安装包也会引导我们将 App 安装到 /Applications中,每一个 App 对应一个 .app 后缀名的文件夹,右键选择任一个 App -> 显示包内容打开文件夹,打开 Contents/Info.plist,该 Info.plist 记录了 App 的名字、开发者、Bundle ID、版本信息,我们备份了这些信息,在需要时 就能准确地在 Git 重新搜索到该 App。如下是 Python 实现
class AppInfo():
def __init__(self,name,version,bundleid,folder):
self.name = name
self.version = version
self.bundleid = bundleid
self.folder = folder
def __str__(self):
return "<td>{3}</td>\n\t<td>{0}</td>\n\t<td>{1}</td>\n\t<td>{2}</td>".format(self.name,self.version,self.bundleid,self.folder) def application_list():
app_folder = "/Applications"
def pharse_plist(path_dir,path_name):
plist = {}
with open(path_dir,'rb') as rbf:
plist = plistlib.load(rbf)
version = plist.get('CFBundleShortVersionString','-')
bundleid = plist.get('CFBundleIdentifier','-')
name = plist.get('CFBundleName','-')
return AppInfo(name=name,version=version,bundleid=bundleid,folder=path_name)
def list_in_dir(dir,level=0):
dirs = os.listdir(dir) if os.path.isdir(dir) else []
apps = []
level -= 1
for app in dirs:
pre_path = os.path.join(dir,app)
info_plist_path = os.path.join(pre_path,'Contents/Info.plist')
if os.path.isfile(info_plist_path):
apps.append(pharse_plist(info_plist_path,app))
elif level >= 0:
apps = apps + list_in_dir(pre_path,level=level)
return apps
app_str = ''
for app in list_in_dir(app_folder,level=2):
app_str += ('<tr>' + str(app) + '</tr>\n')
table_define = """<table frame='hsides'>\n{}{}'</table>'""".format('<tr>\n<th align="left">App</th>\n<th align="left">名字</th>\n<th align="left">版本</th>\n<th align="left">BundleID</th>\n</tr>',app_str)
return table_define
  • 主要是遍历了 /Applications 文件夹,解析每个 App 的 Info.plist 文件,得到 App 列表信息,并加上了 html 标签进行格式化。注意 /Applications 里面可能包含文件夹,所以 list_in_dir 包含两层遍历

命令行程序备份

Mac 上的命令行工具大部分是从 homebrewrubygems 两个地方安装,接下来的 Python 代码演示备份此三处的命令行信息:

def exe_command(list):
result = subprocess.run(list,stdout=subprocess.PIPE)
return result.stdout.decode("utf-8").strip('\n') def gem_list():
return exe_command(['/Users/$(whoami)/.rvm/rubies/ruby-2.4.1/bin/gem','list']).replace("\\n",'<br>\n') def brew_list():
return exe_command(['/usr/local/bin/brew','list']).replace("\\n",'<br>\n')

使用 subprocess.run 执行 shell 命令并得到标准输出,然后对输出做格式化处理,方便后续做阅读

发送邮件

上面得到了 App 和工具信息,现在将其发到邮箱保存。下面的代码使用了内置的邮件服务器,可能会被当做垃圾邮件或有风险的邮件而被拒收

def temp_attachment_path():
file_p = exe_command(['mktemp'])
html_ext = file_p + '.html'
os.rename(file_p,'file_p' + '.html')
# MARK: 打印临时 html 的目录,可以预览发送的格式
print(html_ext)
return html_ext def sendEmail(html_content):
msg = MIMEText(html_content,'html','utf-8')
msg['From'] = 'APP_BACKUP@localhost.com'
# TODO: 改成自己的邮箱地址
msg['To'] = ' xxx@xxx.xxx'
msg['Subject'] = "XXXXX's MBP App 列表"
p = Popen(["/usr/sbin/sendmail", "-t","-oi"], stdin=PIPE)
p.communicate(msg.as_bytes()) if __name__ == '__main__':
app_str = application_list()
pip3_str = '<br><h2>Pip3 Apps</h2><p>%s</p>'%pip3_list()
gem_str = '<br><h2>Gem Apps</h2><p>%s</p>'%gem_list()
brew_str = '<br><h2>Homebrew Apps</h2><p>%s</p>'%brew_list()
content_str = app_str + pip3_str + gem_str + brew_str
attachment_path = temp_attachment_path()
with open(attachment_path,'w') as wf:
wf.write(content_str)
sendEmail(content_str)

上述邮件发送的内容示例:

Mac 开发工具信息的备份的更多相关文章

  1. Mac开发工具

    便捷管理你的Mac App Homebrew:https://brew.sh/index_zh-cn 强大的文本编辑器 Sublime Text:http://www.sublimetext.com ...

  2. MAC 开发工具

    web开发编辑器 Espresso下载地址   密码: i9hr

  3. Mac开发工具汇总

    1: Json Parser Mac版 http://www.pc6.com/mac/180470.html

  4. mac 常用的开发工具

    http://www.oschina.net/news/53946/mac-dev-tools 要清楚的认识到,我们寻找的不是开始按钮,而是程序入口,任何一个操作系统,用户要做的事情并不是找到开始菜单 ...

  5. Mac下的开发工具

    1.webstrom 淘宝上2块钱就能买一个 WebStorm 是jetbrains公司旗下一款JavaScript 开发工具.被广大中国JS开发者誉为“Web前端开发神器”.“最强大的HTML5编辑 ...

  6. Android 常用开发工具以及Mac常用软件

    Android 常用的开发工具记录.其中包括AndroidStudio(IDEA)插件.Mac 上好用的软件以及国内知名Android开发者博客等. Android Studio 插件 codota ...

  7. mac版微信web开发者工具(小程序开发工具)无法显示二维码 解决方案

    微信小程序概念的提出,绝对可以算得上中国IT界惊天动地的一件大事,这可能意味着一场新的开发热潮即将到来, 我也怀着激动的心情准备全身心投入其中,不过截止目前,在官方网站上下载的最新版本都无法使用,打开 ...

  8. JetBrains IntelliJ IDEA for Mac 15.0 破解版 – Mac 上强大的 Java 集成开发工具

    应网友要求更新. IntelliJ IDEA 是最强大的 Java IDE 之一,由知名的Jetbrainsg公司出品,最新版本增加了大量强大易用的特性,比如 Java 8 的Lambda 表达式调试 ...

  9. Mac开发必备工具(一)—— Homebrew

    Homebrew 简介 macOS 缺失的软件包管理器.使用 Homebrew 安装 Apple 没有预装但 你需要的东西.官网有中文说明. 安装与配置 Homebrew 的安装非常简单,将下面这条命 ...

随机推荐

  1. Flutter的盒子约束

    由Expanded widget引发的思考 设计稿如下 布局widget分解 很常见的一种布局方式:Column的子widget中包含ListView @override Widget build(B ...

  2. c++中比较好用的黑科技

    切入正题,上黑科技 一.黑科技函数(常用的我就不写了,例如sort函数) 1.next_permutation(a+1,a+1+n) a[1-n]全排列 2.reverse(a+1,a+1+n) 将a ...

  3. wentiqingdan

    1. Python不用在行尾加分号,也不要用分号将两条命令放在同一行,但加上分号也能执行,不像C/C++分号是必须加的,缺了就会出错. 2. C属于编译型语言,Python属解型语言 编译型的优点是& ...

  4. flask 部署外部访问

    在 app.run(host='0.0.0.0',port=5000) 可以让外部客户端进行访问,访问地址是flask服务器的ip地址和你设置的端口(端口注意不要占用其他端口,如果是阿里云有可能要设置 ...

  5. python爬虫的数据库连接问题

    1.需要导的包 import pymysql 2.# mysql连接信息(字典形式) db_config ={ 'host': '127.0.0.1',#连接的主机id(107.0.0.1是本机id) ...

  6. 配置XNA以适用VS2017进行开发

    Win10似乎已不再支持使用XNA进行游戏开发,安装XNA Game Studio经常会出现错误,显示不兼容,即使安装VS2010也不行.下面给出方法,可以使用VS2017配合XNA进行游戏开发. 1 ...

  7. Django中ORM中的get与filter区别

    本文出自 “orangleliu笔记本” 博客,出处http://blog.csdn.net/orangleliu/article/details/38597593 Django的orm中get和fi ...

  8. 当AI遇上K8S:使用Rancher安装机器学习必备工具JupyterHub

    Jupyter Notebook是用于科学数据分析的利器,JupyterHub可以在服务器环境下为多个用户托管Jupyter运行环境.本文将详细介绍如何使用Rancher安装JupyterHub来为数 ...

  9. Spring框架——IOC 工厂方法

    IoC 是典型的⼯厂模式,如何使⽤用⼯厂模式创建 bean, IoC 通过⼯厂模式创建 bean 有以下两种⽅式 xml <?xml version="1.0" encodi ...

  10. Spring Boot + LayUI 批量修改数据 数据包含着对象

    页面展示 HTML 代码 <blockquote class="layui-elem-quote demoTable"> <div class="lay ...