一、需求

app打包需要打入一些H5进去,以便更快的加载页面。这些H5文件是散落在各个文件夹中的【如下列所示】,偶尔各个文件夹还需新增文件,每次新增一个文件,需要改动jenkins上job脚本,比较麻烦,所以换一种思路来解决这个问题。

tmp/
├── c
│   ├── cmd.js.d
│   ├── TZT2..js.d
│   ├── TZT.js.d
│   └── TZT\\\\\\\\\\\\.json.d
├── c_modules
│   ├── config.js.d
│   ├── css
│   │   ├── base-min2..css.d
│   │   ├── base-min.css.d
│   │   ├── base-min-white.css.d
│   │   ├── images
│   │   │   └── arrow.png.d
│   │   ├── jskj-elementUI.css.d
│   │   ├── scroller.css.d
│   │   ├── skin
│   │   │   └── bg.css.d
│   │   └── skin01
│   │   └── bg.css.d
│   ├── debuggap.js.d
│   ├── fastclick.js.d
│   ├── heatmap.min.js.d
│   ├── jquery-1.7..min.js.d
│   ├── jquery.min.js.d
│   ├── loadData.js.d
│   ├── loadSel.js.d
│   ├── onscroll.js.d
│   ├── pop
│   │   ├── images
│   │   │   └── icon_closeBtn.png.d
│   │   ├── pop.css.d
│   │   ├── pop.js.d
│   │   ├── pop-zdycx.css.d
│   │   └── pop-zdycx.js.d
│   ├── proint.js.d
│   ├── sea.js.d
│   └── sensorsdata.min.js.d
├── config
│   └── index.js.d
├── dist
│   ├── src
│   │   ├── common
│   │   │   ├── iconClick.js.d
│   │   │   └── md5.js.d
│   │   ├── home
│   │   │   ├── app.js.d
│   │   │   ├── images
│   │   │   │   ├── close.png.d
│   │   │   │   ├── default.png.d
│   │   │   │   ├── down.png.d
│   │   │   │   ├── dxsq.png.d
│   │   │   │   ├── fxng.png.d
│   │   │   │   ├── gpcc.png.d
│   │   │   │   ├── gpkh.png.d
│   │   │   │   ├── guide1.png.d
│   │   │   │   ├── guide2.png.d
│   │   │   │   ├── guide3.png.d
│   │   │   │   ├── guide4.png.d
│   │   │   │   ├── header_bg.png.d
│   │   │   │   ├── jftoast.png.d
│   │   │   │   ├── mid_xgsgts.png.d
│   │   │   │   ├── myzc.png.d
│   │   │   │   ├── new.png.d
│   │   │   │   ├── news.png.d
│   │   │   │   ├── nodata.png.d
│   │   │   │   ├── oper_activity_pop.png.d
│   │   │   │   ├── register_tel.png.d
│   │   │   │   ├── right.png.d
│   │   │   │   ├── rmzt_one.png.d
│   │   │   │   ├── rmzt_three.png.d
│   │   │   │   ├── rmzt_two.png.d
│   │   │   │   ├── selected.png.d
│   │   │   │   ├── select.png.d
│   │   │   │   ├── shadow.png.d
│   │   │   │   ├── tougu_bg.png.d
│   │   │   │   ├── up.png.d
│   │   │   │   ├── user.png.d
│   │   │   │   ├── voice.gif.d
│   │   │   │   ├── yszb.png.d
│   │   │   │   └── ywbl.png.d
│   │   │   ├── index.html.d
│   │   │   ├── js
│   │   │   │   ├── home_index.js.d
│   │   │   │   └── Sortable.js.d
│   │   │   ├── routes
│   │   │   │   └── routes.js.d
│   │   │   ├── skin01
│   │   │   │   └── home_index.css.d
│   │   │   └── template
│   │   │   └── home_index.html.d
│   │   ├── mine
│   │   │   ├── api
│   │   │   │   └── function.js.d
│   │   │   ├── app.js.d
│   │   │   ├── css
│   │   │   │   └── index.css.d
│   │   │   ├── img
│   │   │   │   ├── arrow_account.png.d
│   │   │   │   ├── cao.png.d
│   │   │   │   ├── common_used.png.d
│   │   │   │   ├── crown.png.d
│   │   │   │   ├── detail.png.d
│   │   │   │   ├── dui.png.d
│   │   │   │   ├── flow.png.d
│   │   │   │   ├── guide1.png.d
│   │   │   │   ├── guide2.png.d
│   │   │   │   ├── guide3.png.d
│   │   │   │   ├── guide4.png.d
│   │   │   │   ├── guide5.png.d
│   │   │   │   ├── hide_assets.png.d
│   │   │   │   ├── jcsc.png.d
│   │   │   │   ├── ka.png.d
│   │   │   │   ├── not_login.png.d
│   │   │   │   ├── show_assets.png.d
│   │   │   │   ├── sign.png.d
│   │   │   │   ├── transfer_accounts.png.d
│   │   │   │   ├── tui.png.d
│   │   │   │   ├── vip_arrow.png.d
│   │   │   │   ├── vip.png.d
│   │   │   │   ├── wei.png.d
│   │   │   │   └── wen.png.d
│   │   │   ├── index.html.d
│   │   │   ├── js
│   │   │   │   └── index.js.d
│   │   │   ├── routes
│   │   │   │   └── routes.js.d
│   │   │   ├── skin
│   │   │   │   └── index.css.d
│   │   │   ├── skin01
│   │   │   │   └── index.css.d
│   │   │   └── template
│   │   │   └── index.html.d
│   │   └── public-images
│   │   ├── cccb.png.d
│   │   ├── dkxh.png.d
│   │   ├── dpjl.png.d
│   │   ├── dtzq.png.d
│   │   ├── dxsq.png.d
│   │   ├── dzjy.png.d
│   │   ├── fhsp.png.d
│   │   ├── fxng.png.d
│   │   ├── fxspg.png.d
│   │   ├── gdrs.png.d
│   │   ├── ggcg.png.d
│   │   ├── ggtjy.png.d
│   │   ├── ggt.png.d
│   │   ├── gpcc.png.d
│   │   ├── gpkh.png.d
│   │   ├── gpqq.png.d
│   │   ├── gsrl.png.d
│   │   ├── gszx.png.d
│   │   ├── gznhg.png.d
│   │   ├── jcsc.png.d
│   │   ├── jfsc.png.d
│   │   ├── jgdy.png.d
│   │   ├── jjzt.png.d
│   │   ├── lcsc_new.png.d
│   │   ├── lhb.png.d
│   │   ├── mfjg.png.d
│   │   ├── myzc.png.d
│   │   ├── nrzt.png.d
│   │   ├── qbfw.png.d
│   │   ├── rmzt.png.d
│   │   ├── rzrq.png.d
│   │   ├── sczd.png.d
│   │   ├── sjjj.png.d
│   │   ├── ssnc.png.d
│   │   ├── tfp.png.d
│   │   ├── tgsq.png.d
│   │   ├── xsjj.png.d
│   │   ├── xskx.png.d
│   │   ├── xyjy.png.d
│   │   ├── yjyg.png.d
│   │   ├── ywbl_old.png.d
│   │   ├── ywbl.png.d
│   │   ├── yzzz.png.d
│   │   ├── zlcc.png.d
│   │   ├── znxg.png.d
│   │   ├── znyj.png.d
│   │   └── znzg.png.d
│   └── static
│   ├── css
│   │   ├── base.css.d
│   │   ├── base-min.css.d
│   │   ├── base-min-sea-vue.css.d
│   │   ├── mobiscroll.custom-2.6..min.css.d
│   │   └── swiper.css.d
│   ├── jquery
│   │   ├── jquery-1.11..min.js.d
│   │   ├── jquery-1.8..min.js.d
│   │   └── jquery-range.js.d
│   ├── lib
│   │   ├── action.js.d
│   │   ├── flexible.js.d
│   │   ├── mobiscroll.custom-2.6..min.js.d
│   │   └── swiper.js.d
│   ├── sea
│   │   ├── sea.js.d
│   │   └── seajs-text.js.d
│   └── vue
│   ├── vue.js.d
│   ├── vue-router.min.js.d
│   ├── vue-scroller.min.js.d
│   └── vue-scroller.min.js.map.d
├── frontend_7_06_js_test_0303.7z
└── jy
└── ggqq
├── function
│   └── function.js.d
├── ggqq_index.html.d
├── image
│   ├── add.png.d
│   ├── delete.png.d
│   ├── drcj.png.d
│   ├── fall.png.d
│   ├── ggqq_chicang.png.d
│   ├── ggqq_cljy.png.d
│   ├── ggqq_index_b.png.d
│   ├── ggqq_index.png.d
│   ├── ggqq_kcwt.png.d
│   ├── ggqq_ksxd.png.d
│   ├── ggqq_minus.png.d
│   ├── ggqq_nologin.png.d
│   ├── ggqq_plus.png.d
│   ├── ggqq_sybg.png.d
│   ├── ggqq_xq.png.d
│   ├── gou.png.d
│   ├── kcwt.png.d
│   ├── loading.gif.d
│   ├── minus.png.d
│   ├── mymoney.png.d
│   ├── notselect.png.d
│   ├── others@2x.png.d
│   ├── reset.png.d
│   ├── right2.png.d
│   ├── right.png.d
│   ├── rise.png.d
│   ├── select.png.d
│   ├── shoppingcar.png.d
│   └── zhcl@2x.png.d
├── js
│   └── ggqq_index.js.d
├── skin
│   ├── bdq.css.d
│   ├── chedan.css.d
│   ├── ggqq_bdsx.css.d
│   ├── ggqq_chicang.css.d
│   ├── ggqq_edcx.css.d
│   ├── ggqq_hylist.css.d
│   ├── ggqq_hysx.css.d
│   ├── ggqq_index.css.d
│   ├── ggqq_notice.css.d
│   ├── ggqq_qindex.css.d
│   ├── ggqq_xq.css.d
│   ├── ggqq_xycx.css.d
│   ├── ggqq_xyxq.css.d
│   ├── jsd.css.d
│   ├── login_mesg.css.d
│   ├── qchicang.css.d
│   ├── revisepassword.css.d
│   └── zjcc.css.d
└── skin01
├── bdq.css.d
├── ggqq_bankcr.css.d
├── ggqq_bdsx.css.d
├── ggqq_chicang.css.d
├── ggqq_edcx.css.d
├── ggqq_hylist.css.d
├── ggqq_hysx.css.d
├── ggqq_index.css.d
├── ggqq_notice.css.d
├── ggqq_qindex.css.d
├── ggqq_xq.css.d
├── ggqq_xycx.css.d
├── ggqq_xyxq.css.d
├── jsd.css.d
├── login_mesg.css.d
├── revisepassword.css.d
└── zjcc.css.d

二、思路

让前端开发把需要打入app的H5文件放入D:\scripts\src目录上  --->  读取该目录下文件的绝对路径  --->  在workspace中寻找对应的文件 ---> 并将这些文件复制到新的目录中,进行7za打包

python代码如下:

 #!/usr/bin/env python3
#-*- coding:GBK -*-
# author by Michael Ho
# contact:rui.he@geekthings.com.cn
import os, shutil def copy_app_H5_file(x_root, x_dir, src_dir, dst_dir):
if os.path.exists(dst_dir):
shutil.rmtree(dst_dir)
for root, dirs, files in os.walk(x_root):
for d_name in dirs:
d_name = os.path.join(root, d_name).rstrip()
d_dir = d_name.replace(x_dir, dst_dir)
if not os.path.exists(d_dir):
os.makedirs(d_dir)
for f_name in files:
f_name = os.path.join(root, f_name).rstrip()
if(os.path.splitext(f_name)[1] == ".d"):
s_file = f_name.replace(x_dir, src_dir)
d_file = f_name.replace(x_dir, dst_dir)
shutil.copyfile(s_file, d_file)
# 判断复制是否成功
if(s_file.replace(src_dir, "") == d_file.replace(dst_dir, "")):
print(d_file + "->" + "拷贝成功")
else:
print("拷贝过程发生错误,请检查...") if __name__ == '__main__':
# 上传的小包根目录,默认是d:\scripts,不要去动它!!!
x_root = "d:\\scripts" # 小包目录,默认是src,需要把小包的文件放在一个叫src目录里面!!!
x_dir = "d:\\scripts\\src" # 从gitlab上获取的目录
src_dir = "d:\\Jenkins\\frontend_encrypt" # 需要复制到目标目录,一般对其目录进行打包
dst_dir = "d:\\Jenkins\\H5_APP" copy_app_H5_file(x_root, x_dir, src_dir, dst_dir)

三、说明

1.在Jenkins客户端在Windows上,python编码格式要设定成    # -*- coding:GBK -*-  (笔者当时写的是 # -*- coding: utf-8 -*-)不然Jenkins会报以下错误

SyntaxError: (unicode error) 'utf-8' codec can't decode byte 0xbf in position 0: invalid start byte

2.python调系统级的接口不算太友好,个人认为没有批处理方便。如果整个文件夹复制的话,还是 xcopy  src_dir dst_dir /s /e /f方便,所以Jenkins上打包的脚本,笔者为了打包速度,Windows打包机采用batch+python【以上个人见解、水平有限】

python复制多层目录下的文件至其他盘符对应的目录中的更多相关文章

  1. rsync+inotify 实现资源服务器的同步目录下的文件变化时,备份服务器的同步目录更新,以资源服务器为准,去同步其他客户端

    测试环境: 资源服务器(主服务器):192.168.200.95 备份服务器(客户端):192.168.200.89 同步目录:/etc/test 同步时使用的用户名hadoop密码12345 实验目 ...

  2. PHP 批量获取指定目录下的文件列表(递归,穿透所有子目录)

    //调用 $dir = '/Users/xxx/www'; $exceptFolders = array('view','test'); $exceptFiles = array('BaseContr ...

  3. python统计目录和目录下的文件,并写入excel表

    运营那边提出需求,有些媒体文件需要统计下 目录结构大概是这样的 每个目录下面都有很多文件,目录下面没子目录 我这里是模拟下创建的目录和文件,和运营那边说的目录结构都是一致的 想最终统计结果如下格式 我 ...

  4. Python递归遍历目录下所有文件

    #自定义函数: import ospath="D:\\Temp_del\\a"def gci (path): """this is a stateme ...

  5. Python开发【笔记】:获取目录下所有文件

    获取文件 import os def sub_dirs(rdir): li = os.listdir(rdir) return li def main(rdir): content = sub_dir ...

  6. python实现指定目录下批量文件的单词计数:并发版本

    在 文章 <python实现指定目录下批量文件的单词计数:串行版本>中, 总体思路是: A. 一次性获取指定目录下的所有符合条件的文件 -> B. 一次性获取所有文件的所有文件行 - ...

  7. Python打开目录下所有文件

    用Python打开指定目录下所有文件,统计文件里特定的字段信息. 这里是先进入2017-02-25到2017-03-03目录,然后进入特定IP段目录下,最后打开文件进行统计 import os, gl ...

  8. Python遍历目录下所有文件的最后一行进行判断若错误及时邮件报警-案例

    遍历目录下所有文件的最后一行进行判断若错误及时邮件报警-案例: #-*- encoding: utf-8 -*- __author__ = 'liudong' import linecache,sys ...

  9. Python 读取某个目录下的文件

    读取某个目录下的文件,如'/Users/test/test_kmls'目录下有test1.txt.test2.txt. 第一种方法读出的all_files是test1.txt.test2.txt im ...

随机推荐

  1. [WC2015]未来程序(提交答案)

    sub1:ans=a*b%c,龟速乘即可. #include <stdio.h> #include <stdlib.h> unsigned long long a, b, c, ...

  2. Events|sample space|mutually exclusive events

    5.2Events The collection of all 52 cards—the possible outcomes—is called the sample space for this e ...

  3. A4988驱动42步进电机

    A4988步进电机驱动器驱动控制42步进电机速度,步进电机调速,调节驱动电流       1  A4988步进电机驱动器简介 方便使用,是我们这些用户最想要的,固有的名词和深入介绍在这就不多说了,您可 ...

  4. 单独安装jenkins-没有tomcat

    这里讲解war包的安装:windows的msi版安装很简单,双击即可,不用讲 1.官网下载 2. 3.把war包放到java目录下 4. 5.安装完成后打开:127.0.0.1:8080 输入密码后会 ...

  5. 吴裕雄--天生自然python学习笔记:python设置文档的格式

    Win32com 组件可为特定范围的内 容设置格式, 较常用的格式有标题格式.对齐 方式格式及字体格式 . 许多格式使用 常量表示 , 所 以 需先导入 constants常量模块 : 设置标题格式的 ...

  6. HTML配色表方案

  7. Differential Calculus

    Taylor's Formula Theorem 1.1. Let \(f\): \(I=(c,d)->\mathbb{R}\) be a n-times differentiable func ...

  8. linux下使用过的命令总结(未整理完)

    1.常用命令不需解释 ls\cd\cp\mv\pwd\file\vi\vim\cat 2.getconf LONG_BIT 终端返回32表示操作系统32位,返回64表示操作系统64位. 3.ifcon ...

  9. 求求你,下次面试别再问我什么是 Spring AOP 和代理了!

    https://mbd.baidu.com/newspage/data/landingsuper?context=%7B%22nid%22%3A%22news_9403056301388627935% ...

  10. 框架之MyBatis

    什么是框架,简单的来说框架就是一个程序的半成品,而我们就是的工作就是根据我们的工作需要将其完善.MyBatis框架的作用就是将我们使用JDBC操作数据库的过程移交给MyBatis,让它来帮我们完成这些 ...