Flask 处理上传的文件非常简单,总结归纳可以分为三步:
<form>
标签被标记有 enctype=multipart/form-data
,并且在里面包含一个 <input type=file>
标签假设将上传的文件存放在 static/uploads
目录中。
werkzeug 库可以判断文件名是否安全,例如防止文件名是 /../test.png
, 安装
pip install werkzeug
具体代码:
from flask import Flask, request
from werkzeug.utils import secure_filename
import os
app = Flask(__name__)
app.config['UPLOAD_FOLDER'] = 'static/uploads/'
app.config['ALLOWED_EXTENSIONS'] = set(['png', 'jpg', 'jpeg', 'gif'])
# For a given file, return whether it's an allowed type or not
def allowed_file(filename):
return '.' in filename and \
filename.rsplit('.', 1)[1] in app.config['ALLOWED_EXTENSIONS']
@app.route('/')
def hello_world():
return 'hello world'
@app.route('/upload', methods=['POST'])
def upload():
upload_file = request.files['file']
if upload_file and allowed_file(upload_file.filename):
filename = secure_filename(upload_file.filename)
upload_file.save(os.path.join(app.root_path, app.config['UPLOAD_FOLDER'], filename))
return 'hello, '+request.form.get('name', 'little apple')+'. success'
else:
return 'hello, '+request.form.get('name', 'little apple')+'. failed'
if __name__ == '__main__':
app.run(debug=True)
app.config
中的 config 是字典的子类,可以用来设置自有的配置信息,也可以设置自己的配置信息。函数 allowed_file(filename)
用来判断 filename 是否有后缀以及后缀是否在app.config['ALLOWED_EXTENSIONS']
中。
这里使用了 werkzeug
自带的 secure_filename
方法,该方法会过滤所有非 ASCII 码,对于中文文件名处理就不怎么友好了,所以我们可以定义自己的 secure_filename
方法
def secure_filename(filename):
"""
确保文件名不包含 / -
:param filename:
:return:
"""
filename = re.sub('[" "\/\--]+', '-', filename)
filename = re.sub(r':-', ':', filename)
filename = re.sub(r'^-|-$', '', filename)
return filename
使用正则将文件名中的非法字符去除掉。
Flask 提供了 getlist
方法
for upload in request.files.getlist("file"):
filename = os.path.splitext(upload.filename)[0]
filename = secure_filename(filename)
save_name = hashlib.md5('video'+ str(time.time())).hexdigest()[:16]
dest = '/'.join([dest_folder, save_name])
print("Accept incoming file:", filename)
upload.save(dest)
模拟上传文件
import requests
files = {'file': open('01.jpg', 'rb')}
user_info = {'name': 'einverne'}
r = requests.post("http://127.0.0.1:5000/upload", data=user_info, files=files)
print r.text
要控制上产文件的大小,可以设置请求实体的大小,例如:
app.config['MAX_CONTENT_LENGTH'] = 16 * 1024 * 1024 #16MB
如果要获取上传文件的内容可以:
file_content = request.files['image01'].stream.read()
在 template 目录下新建 upload.html
,确保在 HTML 表单中设置 enctype=”multipart/form-data” 属性
<form action="\{\{ url_for('.upload_file') \}\}" method=post enctype=multipart/form-data>
<div class="form-group">
<label for="exampleInputFile">上传文件</label>
<input type="file" name="file">
<p class="help-block">上传数据</p>
<button type="submit" class="btn btn-default">Submit</button>
</div>
</form>
在 views.py 中写 upload_file
方法处理上传逻辑
@crawler_control.route('/upload', methods=['POST'])
def upload_file():
if request.method == 'POST':
f = request.files['file']
if f and '.' in f.filename and f.filename.rsplit('.', 1)[1] in app.config['ALLOWED_EXTENSIONS']:
filename = secure_filename(f.filename)
path = op.join('/tmp', filename)
f.save(path)
return "Success"
return render_template_string("only support POST")
处理 JSON 时,请求和响应头的 Content-Type
设置为 application/json
。
from flask import Flask, request, Response
import json
app = Flask(__name__)
@app.route('/json', methods=['POST'])
def my_json():
print request.headers
print request.json
rt = {'info':'hello '+request.json['name']}
# add custom headers
response.headers.add('Server', 'python flask')
return Response(json.dumps(rt), mimetype='application/json')
if __name__ == '__main__':
app.run(debug=True)
模拟客户端请求
import requests, json
user_info = {'name': 'einverne'}
headers = {'content-type': 'application/json'}
r = requests.post("http://127.0.0.1:5000/json", data=json.dumps(user_info), headers=headers)
print r.headers
print r.json()
模拟 RESTful 接口
from flask import Flask
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'hello world'
@app.route('/user/<username>')
def user(username):
print username
print type(username)
return 'hello world'
@app.route('/user/<username>/friends')
def user_friends(username):
print username
print type(username)
return 'hello world'
if __name__ == '__main__':
app.run(debug=True)
使用内置类型装换
@app.route('/page/<int:num>')
def page(num):
print num
print type(num)
return 'hello world'
自动将 num 转换成 int 类型,目前 Flask 支持的转换有:
类别 | 解释 |
---|---|
string | 任何不带 slash(/) 的字符串 |
int | 整数 |
float | float |
path | 接受 slash(/) |
any | 所有 |
uuid | uuid string |
复杂的用法,比如定义一个范围
@app.route('/page/<int:num1>-<int:num2>')
def page(num1, num2):
print num1
print num2
return 'hello world'
自定义转换器
自定义的转换器是一个继承 werkzeug.routing.BaseConverter
的类,修改 to_python
和 to_url
方法即可。to_python
方法用于将url中的变量转换后供被 @app.route
包装的函数使用,to_url
方法用于 flask.url_for
中的参数转换。
下面是一个示例,将 HelloWorld/index.py
修改如下:
from flask import Flask, url_for
from werkzeug.routing import BaseConverter
class MyIntConverter(BaseConverter):
def __init__(self, url_map):
super(MyIntConverter, self).__init__(url_map)
def to_python(self, value):
return int(value)
def to_url(self, value):
return 'hi'
app = Flask(__name__)
app.url_map.converters['my_int'] = MyIntConverter
@app.route('/page/<my_int:num>')
def page(num):
print num
print url_for('page', num='123')
return 'hello world'
if __name__ == '__main__':
app.run(debug=True)
浏览器访问 http://127.0.0.1:5000/page/123 后,HelloWorld/index.py 的输出信息是:
123
/page/hi
上面例子只能提到的 url_for
方法用于构建URL,他的使用方法如下
from flask import Flask, url_for
app = Flask(__name__)
@app.route('/')
def hello_world():
pass
@app.route('/user/<name>')
def user(name):
pass
@app.route('/page/<int:num>')
def page(num):
pass
@app.route('/test')
def test():
print url_for('hello_world')
print url_for('user', name='einverne')
print url_for('page', num=1, q='hadoop mapreduce 10%3')
print url_for('static', filename='uploads/01.jpg')
return ''
if __name__ == '__main__':
app.run(debug=True)
输出
/
/user/einverne
/page/1?q=hadoop+mapreduce+10%253
/static//uploads/01.jpg
url for 后接方法名
Flask 是一个轻量级的基于 python 的 web 框架。
一般情况下,只要通过 pip 安装 Flask 即可:
pip install Flask
进入 python shell
>>> import flask
>>> print flask.__doc__
flask
~~~~~
A microframework based on Werkzeug. It's extensively documented
and follows best practice patterns.
:copyright: (c) 2015 by Armin Ronacher.
:license: BSD, see LICENSE for more details.
>>> print flask.__version__
0.12.2
创建 index.py
文件,在文件中放入以下内容:
from flask import Flask
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'Hello World!'
if __name__ == '__main__':
app.run()
运行 python index.py
python index.py
* Running on http://127.0.0.1:5000/
变量 app 是一个 Flask 实例,当客户端访问根目录 /
时,将响应 hello_world()
函数返回的内容。这不是返回 Hello World! 这么简单,Hello World! 只是 HTTP 响应报文的实体部分,状态码等信息既可以由 Flask 自动处理,也可以通过编程来制定。
在创建 Flask 时使用额外的参数,具体可参考 __doc__
添加 run 函数参数:
app.run(debug=True)
默认情况下,Flask 使用 127.0.0.1
,端口为 5000
,通过如下方式更改:
app.run(host='0.0.0.0', port=80, debug=True)
绑定 80 端口需要 root 权限运行 index.py
。
from flask import Flask, request
app = Flask(__name__)
@app.route('/')
def hello_world():
params = request.args.items()
return params.__str__()
if __name__ == '__main__':
app.run()
另外可以通过 request.full_path
和 request.path
来获取客户端请求 url
from flask import Flask, request
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'hello world'
@app.route('/register', methods=['POST'])
def register():
print request.headers
print request.form
print request.form['name']
print request.form.get('name')
print request.form.getlist('name')
print request.form.get('nickname', default='little apple')
return 'welcome'
if __name__ == '__main__':
app.run(debug=True)
使用 requests
模拟客户端请求:
import requests
user_info = {'name': 'einverne', 'password': '123'}
r = requests.post("http://127.0.0.1:5000/register", data=user_info)
print r.text
构造访问 URL 时可能会遇到如下两种情况:
@app.route('/projects/')
def projects():
return 'The project page'
@app.route('/about')
def about():
return 'The about page'
解释:
Flask 下有 Flask-Restless 结合 SQLAlchemy 轻松实现 RESTful 。
因为 HTTP 协议是无状态的,服务器不知道用户上一次做了什么,这阻碍了交互式 Web 应用程序的实现,所以引入了 Cookie 和 Session,用来记录用户的状态。
要记住的一点是 Session 是服务端记录状态,而 Cookie 是客户端记录状态。
对于一个分布式应用来说服务端记录状态会涉及到大量的成本。
from flask import Flask, render_template_string, \
session, request, redirect, url_for
app = Flask(__name__)
app.secret_key = 'F12Zr47j\3yX R~X@H!jLwf/T'
@app.route('/')
def hello_world():
return 'hello world'
@app.route('/login')
def login():
page = '''
<form action="\{\{ url_for('do_login') \}\}" method="post">
<p>name: <input type="text" name="user_name" /></p>
<input type="submit" value="Submit" />
</form>
'''
return render_template_string(page)
@app.route('/do_login', methods=['POST'])
def do_login():
name = request.form.get('user_name')
session['user_name'] = name
return 'success'
@app.route('/show')
def show():
return session['user_name']
@app.route('/logout')
def logout():
session.pop('user_name', None)
return redirect(url_for('login'))
if __name__ == '__main__':
app.run(debug=True)
设置 session 过期时间
from datetime import timedelta
from flask import session, app
session.permanent = True
app.permanent_session_lifetime = timedelta(minutes=5)
from flask import Flask, request, Response, make_response
import time
app = Flask(__name__)
@app.route('/')
def hello_world():
return 'hello world'
@app.route('/add')
def login():
res = Response('add cookies')
res.set_cookie(key='name', value='letian', expires=time.time()+6*60)
return res
@app.route('/show')
def show():
return request.cookies.__str__()
@app.route('/del')
def del_cookie():
res = Response('delete cookies')
res.set_cookie('name', '', expires=0)
# print res.headers
# print res.data
return res
if __name__ == '__main__':
app.run(host='0.0.0.0', debug=True)
毕业面临人生第一次租房,而到现在从学校搬出来也已经有近一个月时间,期间工作有些繁忙,而有一些租房相关的笔记一直躺在WizNote中,直到今天回到家,才想到应该把我的经历记下来,提醒自己以后租房过程中避坑。
现在的社会是一个信息为主的社会,我们经常挂在嘴边说的一句话就是信息不对称能够产生利益,现在社会很多的金钱交易都是基于信息的不对称。而租房信息获取的途径也会直接导致以后一年的住房条件和心情。下面就讲讲我参考的几个主要站点和App。
首先就是自如,也是我周围同学使用最多的,大多数周围的同学都通过自如找到了房子,我也看过一些自如的房子,自如有如下特点
缺点:
在同学QQ群中第一次听说,立马下载安装注册之后,需要缴纳100元作为认证费用,只有缴纳费用之后才能看到发布者联系方式。可是缴纳认证费用之后连续联系的两家都已经将房屋出租出去。不及时更新房屋信息,并且在缴纳认证费用的界面没有任何提示。如果30天内没有签约,费用不退。
总结下:
强烈建议将该App 拉入黑名单。白白让我交了100块的认证。
房源多,整租单间价格和市场几乎就是市场均价,但是环境看机会。
类似链家。
同样也是一个找租房的应用,不过看官网和他应用的主要内容似乎是一家经营房产,买卖房屋为主业的公司,租房只是他们其中的一个副业,当然也可以在上面找找周围合适的房屋信息,个人感觉比那个鬼 107 间要靠谱一些,和自如相同也是有“房管”来管理租赁一个范围内的房屋,和自如类似。可以参考,毕竟最后没有在安居客租房因此也不能下结论。
如果你公司内网有租房信息,请一定要利用好,租房那天在外面跑了很多地方,顶着大太阳看了一个有一个房子,却只有在公司内网同事分享的一些房子靠谱一些。个人最后也在同学的推荐下找到了相对比较合适的房子。
其他 58,链家 ,豆瓣 之类,只是寥寥看了一下,并没有真正却联系,在看房的时候也了解到不乏有之前的房东在 58 上找到很好的房子,进而想要帮房东更快的租出房子而内网发布信息的。当然如果个人贸然出租请还是注意防骗。
这个社会的复杂性都在一直合同中变的扁平和简洁,而着又同样带来了合同的复杂性,合同越复杂带来的后续处理细节就越方便。我们找到房东直租,但房东是一位上了年龄的阿姨,合同也让我们自己带,网上,同学问了一圈,最后在住房和城市建设委员会的网站上找了一份租房合同模板 。但网站上给出的版本还是 08 年的版本,我们又找了一份 14 年的修订版 —- 北京市房屋租赁合同(自行成交版2014)
有了合同模板就好很多,按照上面的填空,填完也能够整个合同流程有一个大概的印象,应该注意的地方合同中大多都有提到,当然除了合同中提到的内容,涉及到金钱的地方要额外的注意。
对于标题中提出的问题,可以拆分来一步步解决。先来区分 MySQL 中的 KEY
和 INDEX
,之后问题就可以简化为 PRIMARY KEY,UNIQUE KEY 和 INDEX 的区别。而这三者也正好是索引的划分,主键索引,唯一索引和普通索引(INDEX)。
INDEX 可以用来提高数据库中查询数据的速度。INDEX 通常加在那些 JOIN, WHERE,和 ORDER BY 子句的列上。创建索引时,需要确保该索引是应用在 SQL 查询语句的条件(一般作为 WHERE 子句的条件)。实际上,索引也是一张表,该表保存了主键与索引字段,并指向实体表的记录。
索引也有它的缺点:虽然索引提高了查询速度,却会降低更新表的速度,如对表进行 INSERT、UPDATE 和 DELETE。因为更新表时,MySQL 不仅要保存数据,还要保存一下索引文件。
KEY 通常是 INDEX 同义词,但实际上还是有区别的。如果关键字属性 PRIMARY KEY 在列定义中已给定,则 PRIMARY KEY 也可以只指定为 KEY。这么做的目的是与其它数据库系统兼容。PRIMARY KEY 是一个唯一 KEY,此时,所有的关键字列必须定义为 NOT NULL。如果这些列没有被明确地定义为 NOT NULL,MySQL 应隐含地定义这些列。
KEY 即键值,是关系模型理论中的一部份,比如有主键(PRIMARY KEY),外键(Foreign KEY)等,用于数据完整性检验和唯一性约束等。而 INDEX 则处于实现层面,可以对表的任意列建立索引,那么当建立索引的列处于 SQL 语句中的 Where 条件中时,就可以得到快速的数据定位,从而提高检索效率。至于 UNIQUE INDEX,则只是属于 INDEX 中的一种而已,建立了 UNIQUE INDEX 表示此列数据不可重复。
于是,在设计表的时候,KEY 只是要处于模型层面的,而当需要进行查询优化,则对相关列建立索引即可。
KEY 是数据库的物理结构,一般理解为键,包含两层含义,一是约束,偏重于约束和规范数据库的结构完整性,二是索引,辅助查询。
可见,key 是同时具有 constraint 和 index 的意义。
INDEX 是数据库的物理结构,一般翻译为索引,但只辅助查询,会在创建时占用另外的空间。索引分为前缀索引、全文索引等。索引只是索引,不会去约束索引字段的行为。
PRIMARY KEYs(主键) 和 UNIQUE KEYs(唯一键约束) 是类似的, PRIMARY KEY 通常是一列,也有可能多列,通常由他来决定一行数据 (row)。 一张表只能有一个 PRIMARY KEY,但可以有很多 UNIQUE KEY。 当给一列设置为 UNIQUE KEY 之后,不能有两行在该列上有相同的数据。 PRIMARY KEY 不允许有 NULL 值,但是 UNIQUE KEY 可以。
修改表 ALTER TABLE table_name ADD PRIMARY KEY(column_name, ...)
总结,相同点:
差异点:
PRIMARY KEY 不能有空值, UNIQUE KEY 可以有。如果 PRIMARY KEY 的 1 个或多个列为 NULL,在增加 PRIMARY KEY 时,列自动更改为 NOT NULL 。而 UNIQUE KEY 对列没有要求是通过参考索引实施的,如果插入的值均为 NULL,则根据索引的原理,全 NULL 值不被记录在索引上,所以插入全 NULL 值时,可以有重复的,而其他的则不能插入重复值。
alter table t add constraint uk_t_1 UNIQUE (a,b); insert into t (a ,b ) values (null,1); # 不能重复 insert into t (a ,b ) values (null,null);#可以重复
在 MySQL 中,对于一个 PRIMARY KEY 的列,MySQL 已经自动对其建立了 UNIQUE INDEX,无需重复再在上面建立索引了。
网上关于 PRIMARY KEY 和 UNIQUE INDEX 的一段解释:
Note that “PRIMARY” is called PRIMARY KEY not INDEX.
KEY is something on the logical level, describes your table and database design (i.e. enforces referential integrity …)
INDEX is something on the physical level, helps improve access time for table operations.
Behind every PK there is (usually) UNIQUE INDEX created (automatically).
建立索引会占用磁盘空间的索引文件。
CREATE INDEX IndexName ON mytable(username(length));
如果是 CHAR,VARCHAR 类型,length 可以小于字段实际长度;如果是 BLOB 和 TEXT 类型,必须指定 length。
在创建表时创建索引:
CREATE TABLE mytable(
ID INT NOT NULL,
username VARCHAR(15) NOT NULL,
INDEX [INDEXName] (username(length))
);
删除索引
DROP INDEX [INDEXName] ON mytable;
最近用到 MySQL 5.7 把所有设备上的 MySQL 版本都升级到了最新,在 Ubuntu/Debian 上升级MySQL 5.7 的内容可以在之前的文章看到。现在记录一下树莓派中升级 MySQL 的步骤。使用到 MySQL 5.7 主要也是因为其支持的新数据类型,之前项目用到了,迁移的时候会遇到问题。在网上寻找解决方案的时候遇到了一个和我遭遇差不多的,需要使用到 MySQL 5.7+ 才支持的 JSON data-type。
在 respberry pi 官方的源中,只有稳定版的 5.5 MySQL,如果要用到最新的版本只能够自己手动编译更新安装,幸而 Debian 提供了编译好的安装包 ,根据以下步骤就能够安装上。
apt-get
安装
sudo apt-get install libaio1 libaio-dev libhtml-template-perl libevent-core-2.0-5
需要安装新版本的 gcc,所以要从 stretch 源中拉取
sudo vim /etc/apt/sources.list
# change jessie to stretch
sudo apt-get update
sudo apt-get install gcc-6 g++-6
# change stretch back to jessie
sudo apt-get update
然后安装两个依赖
wget http://ftp.debian.org/debian/pool/main/m/mecab/libmecab2_0.996-3.1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/l/lz4/liblz4-1_0.0~r131-2+b1_armhf.deb
sudo dpkg -i libmecab2_0.996-3.1_armhf.deb
sudo dpkg -i liblz4-1_0.0~r131-2+b1_armhf.deb
sudo apt-get --purge remove mysql-server # and a lot of like client, common etc
下载 deb 并安装
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/libmysqlclient-dev_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/libmysqlclient20_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/libmysqld-dev_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/mysql-client-5.7_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/mysql-client-core-5.7_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/mysql-server-5.7_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-5.7/mysql-server-core-5.7_5.7.18-1_armhf.deb
wget http://ftp.debian.org/debian/pool/main/m/mysql-defaults/mysql-common_5.8+1.0.2_all.deb
sudo dpkg -i mysql-common_5.8+1.0.2_all.deb
sudo dpkg -i mysql-client-core-5.7_5.7.18-1_armhf.deb
sudo dpkg -i mysql-client-5.7_5.7.18-1_armhf.deb
sudo dpkg -i mysql-server-core-5.7_5.7.18-1_armhf.deb
sudo dpkg -i mysql-server-5.7_5.7.18-1_armhf.deb
注意这里的版本号,可能之后会更新,去 ftp 中获取最新的版本在这里替换即可。
安装完成之后 sudo reboot
在启动之后
mysql_upgrade -u root -p --force
# 然后输入密码
sudo service mysql restart
安装完毕。
rsync 全名 Remote Sync,是类 UNIX 系统下的数据镜像备份工具。可以方便的实现本地,远程备份,rsync 提供了丰富的选项来控制其行为。rsync 优于其他工具的重要一点就是支持增量备份。
rsync - a fast, versatile, remote (and local) file-copying tool
rsync 是一个功能非常强大的工具,其命令也有很多功能选项,它的特性如下:
rsync 的官方网站:http://rsync.samba.org/,可以从上面得到最新的版本。
Rsync 的命令格式可以为以下六种:
# 本地模式
rsync [OPTION...] SRC DEST
# 远程 Push
rsync [OPTION...] SRC [USER@]HOST:DEST
# 远程 Pull
rsync [OPTION...] [USER@]HOST:SRC DEST
# 通过 Rsync daemon Pull
rsync [OPTION...] [USER@]HOST::SRC DEST
rsync [OPTION...] rsync://[USER@]HOST[:PORT]/SRC... [DEST]
# 通过 Rsync daemon Push
rsync [OPTION...] SRC [USER@]HOST::DEST
rsync [OPTION...] SRC... rsync://[USER@]HOST[:PORT]/DEST
上述命令中,SRC 表示源地址,而 DEST 表示目标地址,这二者可以是本地目录,也可以是远程服务器地址。当只有 SRC 地址没有 DEST 时会列出所有的文件列表,而不会执行拷贝。
rsync 有两种方式来连接远程服务器
这两种方式的直接区别体现在路径中的冒号 (:) ,当只有一个冒号时使用 remote shell,当有两个冒号时使用 daemon 连接。
rsync 有六种不同的工作模式:
这 6 种方式看似复杂,其实只要记住一些常用参数,然后记住一些常用方法就能够将 rsync
利用起来。
可以 man rsync
参考 rsync 文档,了解详细的使用方法,下面解析一些参数的使用
常用的几个参数
-v verbose 详细输出
-a 归档模式,递归方式传输文件,并保持连接,权限,用户和组,时间信息
-z 压缩文件传输
-h human-readable, 输出友好
-u 跳过已经存在的文件,备份更新
rsync 参数的具体解释如下:
-v, --verbose 详细模式输出
-q, --quiet 精简输出模式
-c, --checksum 打开校验开关,强制对文件传输进行校验
-a, --archive 归档模式,表示以递归方式传输文件,并保持所有文件属性,等于 -rlptgoD
-r, --recursive 对子目录以递归模式处理
-R, --relative 使用相对路径信息
-b, --backup 创建备份,也就是对于目的已经存在有同样的文件名时,将老的文件重新命名为~filename。可以使用 --suffix 选项来指定不同的备份文件前缀。
--backup-dir 将备份文件(如~filename) 存放在在目录下。
-suffix=SUFFIX 定义备份文件前缀
-u, --update 仅仅进行更新,也就是跳过所有已经存在于 DST,并且文件时间晚于要备份的文件。(不覆盖更新的文件)
-l, --links 保留软链结
-L, --copy-links 想对待常规文件一样处理软链结
--copy-unsafe-links 仅仅拷贝指向 SRC 路径目录树以外的链结
--safe-links 忽略指向 SRC 路径目录树以外的链结
-H, --hard-links 保留硬链结
-p, --perms 保持文件权限
-o, --owner 保持文件属主信息
-g, --group 保持文件属组信息
-D, --devices 保持设备文件信息
-t, --times 保持文件时间信息
-S, --sparse 对稀疏文件进行特殊处理以节省 DST 的空间
-n, --dry-run 现实哪些文件将被传输
-W, --whole-file 拷贝文件,不进行增量检测
-x, --one-file-system 不要跨越文件系统边界
-B, --block-size=SIZE 检验算法使用的块尺寸,默认是 700 字节
-e, --rsh=COMMAND 指定使用 rsh、ssh 方式进行数据同步
--rsync-path=PATH 指定远程服务器上的 rsync 命令所在路径信息
-C, --cvs-exclude 使用和 CVS 一样的方法自动忽略文件,用来排除那些不希望传输的文件
--existing 仅仅更新那些已经存在于 DST 的文件,而不备份那些新创建的文件
--delete 删除那些 DST 中 SRC 没有的文件
--delete-excluded 同样删除接收端那些被该选项指定排除的文件
--delete-after 传输结束以后再删除
--ignore-errors 及时出现 IO 错误也进行删除
--max-delete=NUM 最多删除 NUM 个文件
--partial 保留那些因故没有完全传输的文件,以是加快随后的再次传输
--force 强制删除目录,即使不为空
--numeric-ids 不将数字的用户和组 ID 匹配为用户名和组名
--timeout=TIME IP 超时时间,单位为秒
-I, --ignore-times 不跳过那些有同样的时间和长度的文件
--size-only 当决定是否要备份文件时,仅仅察看文件大小而不考虑文件时间
--modify-window=NUM 决定文件是否时间相同时使用的时间戳窗口,默认为 0
-T --temp-dir=DIR 在 DIR 中创建临时文件
--compare-dest=DIR 同样比较 DIR 中的文件来决定是否需要备份
-P 等同于 --partial
--progress 显示备份过程
-z, --compress 对备份的文件在传输时进行压缩处理
--exclude=PATTERN 指定排除不需要传输的文件模式
--include=PATTERN 指定不排除而需要传输的文件模式
--exclude-from=FILE 排除 FILE 中指定模式的文件
--include-from=FILE 不排除 FILE 指定模式匹配的文件
--version 打印版本信息
下面举例说明 rsync 的六种不同工作模式:
当 SRC 和 DES 路径信息都不包含有单个冒号 “:” 分隔符时就启动这种工作模式。
同步文件
rsync -ahvz backup.tar.gz /backups/ # DESC 不存在时自动创建
将备份文件同步到 /backups/
目录下。
同步目录
rsync -avzh /home/src /backups/files/
将 /home/src
目录下的文件同步发送到 /backups/files
目录下。记住如果目标地址没有 src
目录,rsync 会自动创建该文件夹。
rsync -avz /home/src/ /backups/files/
SRC 路径末尾的 /
表示不自动创建 DEST 文件夹,在 man rsync
中的解释就是末尾的 /
表示”拷贝当前目录下的文件” ,而不是”拷贝当前的目录”.
使用一个远程 shell 程序(如 rsh、ssh) 来实现将本地机器的内容拷贝到远程机器。当 DES 路径地址包含单个冒号”:”分隔符时启动该模式。
rsync -avz /local/path/ user@remoteip:/path/to/files/
将本地 /local/path/
中的文件同步备份到远程 /path/to/files/
目录。
使用一个远程 shell 程序(如 rsh、ssh) 来实现将远程机器的内容拷贝到本地机器。当 SRC 地址路径包含单个冒号”:”分隔符时启动该模式。
rsync -avz user@remoteip:/home/user/src ./src
从远程 rsync 服务器中拷贝文件到本地机。当 SRC 路径信息包含”::”分隔符时启动该模式。
rsync -av user@remoteip::www /databack
从本地机器拷贝文件到远程 rsync 服务器中。当 DES 路径信息包含”::”分隔符时启动该模式。
rsync -av /databack user@remoteip::www
列远程机的文件列表。这类似于 rsync 传输,不过只要在命令中省略掉本地机信息即可。
rsync -v rsync://remoteip /www
经常遇见的一种情况就是 ssh 更改了默认 22 端口,这个时候就需要使用 -e
参数。
rsync 有两种常用的认证方式,一种为 rsync-daemon 方式,另外一种则是 ssh。
ssh 一般为首选,但当远端服务器的 ssh 默认端口被修改后,rsync 找不到一个合适的方法来输入对方 ssh 服务端口号。
比如现在向机器 remoteip 传送文件,但此时 remoteip 的 ssh 端口已经不是默认的 22 端口。
键入命令
rsync /local/path user@remoteip:/path/to/files/ # 出现错误
rsync 中的命令 参数 -e, --rsh=COMMAND
指定使用 rsh、ssh 方式进行数据同步。
参数的作用是可以使用户自由选择想要使用的 shell 程序来连接远端服务器,当然也可以设置成使用默认的 ssh 来连接,但是这样我们就可以加入 ssh 的参数了。
现在命令可以这样写了:
rsync -avz -e "ssh -p $port" /local/path/ user@remoteip:/path/to/files/
可以使用 --progress
选项来显示进度
rsync -avzhe ssh --progress /home/files/ root@remoteip:/path/to/files/
设置 Max size 备份文件
rsync -avzhe ssh --max-size='2000k' /var/lib/rpm/ root@remoteip:/root/tmprpm
rsync --remove-source-files -zvh backup.tar /tmp/backups/
需要注意的是 rsync
使用 --remove-source-files
之后源文件同步之后会被删除,但是源文件所在的文件夹是不会被删除的,可以通过如下命令删除空文件夹:
find . -depth -type d -empty -delete
使用 --delete
选项。
rsync -avh --delete /path/to/local root@remote:/path/to/remote
--bwlimit=RATE
选项允许用户指定最大传输速率,RATE 值可以是字符串也可以是数值,如果是字符串,比如 --bwlimit=1.5m
表示每秒最高传输速率 1.5m,如果没有后缀那么单位是 1024 bytes。
rsync --bwlimit=100 -avzhe ssh /var/lib/rpm/ root@remoteip:/root/tmprpm/
Tmux 是一个很棒的终端复用工具,和 screen 命令类似,但是 Tmux 终极的分屏功能要比 screen 强大很多,当然入门也比 screen 要高很多。如果你长时间在终端进行编程或者操作,或者你陷入无数的 Tab 而无法自拔,那么你应该需要开始了解一些 Tmux 的基本使用。
本文会从如下几个方面对 Tmux 进行介绍:
Tmux 主要包括以下几个模块:
Ubuntu/Debian 下:
sudo apt-get install tmux
如果是在 macOS 下:
brew install tmux
Tmux 和其他系统的命令一样拥有很多的启动选项,在 man tmux
里面能看到很多。比如 -2
就是启动 256 colours 支持。
默认情况下,tmux 会加载系统配置 /etc/tmux.conf
然后是用户配置 ~/.tmux.conf
。如果配置了该选项,tmux 会在启动时加载,如果配置发生错误,那么会在第一个 session 创建时报错,然后继续处理下面的配置文件。
如果不想要加载自己的配置文件可以在启动的时候使用 -f file
来指定。
Tmux is a tool that allows running multiple terminal sessions through a single terminal window. It allows you to have terminal sessions running in the background and attach and detach from them as needed, which is very useful.
Tmux 的快捷键前缀(Prefix), 为了使自身的快捷键和其他软件的快捷键互不干扰,Tmux 提供了一个快捷键前缀,和 screen 默认激活控制台的 Ctrl+a 不同,Tmux 默认的是 Ctrl+b。当想要使用 Tmux 的快捷键时,需要先按下快捷键前缀,然后再按下快捷键。Tmux 所使用的快捷键前缀默认是组合键 Ctrl-b(同时按下 Ctrl 键和 b 键)。假如你想通过快捷键显示当前 Tmux 中的 session 列表(对应的快捷键是 s),那么你只需要做以下几步:
Ctrl-b
(Tmux 快捷键前缀)Ctrl-b
s
键使用快捷键之后就可以执行一些相应的指令了。当然如果你不习惯使用 Ctrl+b
,也可以在 ~/.tmux.conf
文件中加入以下内容把快捷键变为 Ctrl+a, 或者其他快捷键:
# Set prefix key to Ctrl-a
unbind-key C-b
set-option -g prefix C-a
在下文中就使用 <prefix>
来代替 Tmux 的前缀快捷键了。
每当开启一个新的会话 session 时,Tmux 都会先读取 ~/.tmux.conf
这个文件。该文件中存放的就是对 Tmux 的配置。
如果你希望新的配置项能够立即生效,那么你可以将下面这一行配置加入到文件 ~/.tmux.conf
中。
bind r source-file ~/.tmux.conf \; display-message "tmux config reloaded" # create new short cut to reload tmux.conf
这样配置了之后,每当向 ~/.tmux.conf 文件中添加了新的配置,只需要按下 <prefix> r
就可以重新加载配置并使新的配置生效,从而免去了开启一个新的会话使之生效的步骤。
以下所有的操作都是激活控制台之后,即键入 <prefix>
前提下才可以使用的命令
基本操作
<prefix> ? 列出所有快捷键;按 q 返回
<prefix> d Detach当前会话,可暂时返回 Shell 界面,输入`tmux attach`能够重新进入之前会话
<prefix> s 切换会话 session;在同时开启了多个会话时使用
<prefix> D 选择要脱离的会话;在同时开启了多个会话时使用
<prefix> : 进入命令行模式;此时可输入支持的命令,例如 kill-server 所有 Tmux 会话
<prefix> [ 复制模式,光标移动到复制内容位置,空格键开始,方向键选择复制,回车确认,q/Esc 退出
<prefix> ] 进入粘贴模式,粘贴之前复制的内容,按 q/Esc 退出
<prefix> = 选择性粘贴缓冲区中内容
<prefix> ~ 列出提示信息缓存;其中包含了之前 Tmux 返回的各种提示信息
<prefix> t 显示当前的时间
<prefix> Ctrl+z 挂起当前会话
如果要查看当前 Tmux 的配置,可以通过 tmux show -g
来查看。
Tmux 的一个 Session 可以包含多个 Windows.
在 Tmux 外部 Shell 中可以使用如下方式来管理 Tmux 的 Session:
tmux 创建 session
tmux new -s $session_name 创建并指定 session 名字
tmux ls 列出存在的 session,包括 session 中存在的 windows
tmux attach -t session_name 进入指定会话 session_name
tmux a -t $session_name 上面的缩写形式,进入已存在的 session
tmux kill-session -t $session_name 删除指定 session
在 Tmux 内:
<prefix> :kill-session 删除退出当前 session
<prefix> d 临时退出 session,会话在后台运行,可以通过 attach 进入指定的会话
<prefix> :kill-server 删除所有活跃 session
<prefix> :new -s session_name 在 Tmux 中新建 session
<prefix> s 列出所有活跃 session,并可以从列表中选择 session
<prefix> $ 重命名 session
window(窗口)在 session 里,一个 session 可以有 N 个 window,并且 window 可以在不同的 session 里移动。 window 可以看成是一个 session 的不同 tab。
<prefix> c 创建 window
<prefix> & 删除或者关闭 window
<prefix> n 下一个 window
<prefix> p 上一个 window
<prefix> w 列出现在开启的 window
<prefix> , 重命名 window
<prefix> f 在多个 window 里搜索关键字
<prefix> l last window 在相邻的两个 window 里切换
<prefix> 0,1,2 在 window 之间切换,如果窗口数超过 10 个,可以使用 `<prefix> 'num` 来切换
pane 在 window 里,可以有 N 个 pane,并且 pane 可以在不同的 window 里移动、合并、拆分。
创建 pane:
<prefix> " 横切 split pane horizontal,后面会 remap 这个键
<prefix> % 竖切 split pane vertical,后面 remap 这个键
<prefix> o 按顺序在 pane 之间移动
<prefix> x 关闭 pane
<prefix> z 最大化 pane 和恢复原状 toggle pane zoom
<prefix> ! 移动 pane 至 window
<prefix> "空格" 更换 pane 排版
<prefix> { 移动 pane 往左边,往上面
<prefix> } 移动 pane 往右边,往下面
<prefix> q 显示 pane 编号,在显示编号的时候按对应数字可以切换到该 pane,这个操作太复杂,后面 remap
<prefix> 方向键上下左右 上下左右选择 pane
调整 pane 的大小
<prefix> :resize-pane -U #向上
<prefix> :resize-pane -D #向下
<prefix> :resize-pane -L #向左
<prefix> :resize-pane -R #向右
<prefix> :resize-pane -D 20 (Resizes the current pane down by 20 cells)
<prefix> :resize-pane -U 20 (Resizes the current pane upward by 20 cells)
<prefix> :resize-pane -L 20 (Resizes the current pane left by 20 cells)
<prefix> :resize-pane -R 20 (Resizes the current pane right by 20 cells)
<prefix> :resize-pane -t 2 20 (Resizes the pane with the id of 2 down by 20 cells)
<prefix> :resize-pane -t -L 20 (Resizes the pane with the id of 2 left by 20 cells)
在上下左右的调整里,最后的参数可以加数字 用以控制移动的大小,例如:
<prefix> :resize-pane -D 50
移动 pane 合并至某个 window
<prefix> :join-pane -t $window_name
列出缓冲区目标
<prefix> :list-buffer
查看缓冲区内容
<prefix> :show-buffer
vi 模式
<prefix> :set mode-keys vi
快捷键帮助
<prefix> ? (<prefix> :list-keys)
Tmux 内置命令帮助
<prefix> :list-commands
让你的 Tmux 更加高效,一下内容都可以编辑进 ~/.tmux.conf
用来进一步自定义 Tmux 的行为。 默认的 Tmux 有很多操作方式可能比较 awkward, 只有自己配置让自己熟悉他的行为之后才能让 Tmux 展现出最高效的部分。
在前面也说过, Tmux 默认的 Prefix 是 Ctrl + b,可以按照自己的习惯设定 Prefix 快捷键,很多人将 Caps 和 Ctrl 互换,并且将 Prefix 定义为 Ctrl + a, 我自己使用了以下还是不怎么习惯,所以我将 Prefix 定义成了 Ctrl + \
默认情况下 Tmux 使用 "
来垂直分割成上下两部分,使用 %
来水平分割成左右两部分,但是这两个键需要 Shift+',以及很难按到的%,不如直观上使用 |
和 -
来分割 pane。
# split panes using | and -
bind | split-window -h
bind - split-window -v
unbind '"'
unbind %
在 pane 中移动是最高频的操作了,默认 Tmux 的行为需要每一次都按 <prefix>
, 这样导致每一次都非常麻烦,所以在配置中定义 M-<direction>
(M 代表 Meta,也就是键盘上的 Alt 键), 这样每一次只需要按 Alt+h
, 就能够移动到左边的 pane。
# switch panes using Alt-arrow without prefix
bind -n M-h select-pane -L
bind -n M-l select-pane -R
bind -n M-k select-pane -U
bind -n M-j select-pane -D
这里另外推荐一个 Plugin,如果不想自己配置,可以使用这个插件 vim-tmux-navigator ,这个插件做到了在 pane 中移动就像在 vim 中一样,并且可以和 vim 无缝衔接。使用 Ctrl + hjkl 来在不同的 Pane 中移动。
在设置完 Tmux 在 Tmux 中使用 Vim 的时候会发现,Vim 的 colorscheme 变的有些奇怪,需要在 .bashrc
或者 .zshrc
中设置:
export TERM="xterm-256color"
如果要在 Tmux 中进行复制可以使用 Tmux 的复制模式。
Tmux 中的复制需要使用 <prefix> [
来进入,具体分为如下几步:
<prefix> [
进入复制模式<prefix> ]
进行粘贴注意这种方式只能在 Tmux 中粘贴,而不会拷贝到系统粘贴板。所以我习惯重新 map 几个快捷键,沿用 Vim 中的方式
# in version 2.3 and below https://github.com/tmux/tmux/commit/76d6d3641f271be1756e41494960d96714e7ee58
setw -g mode-keys vi
bind-key -T copy-mode-vi 'v' send -X begin-selection # Begin selection in copy mode.
bind-key -T copy-mode-vi 'C-v' send -X rectangle-toggle # Begin selection in copy mode.
bind-key -T copy-mode-vi 'y' send -X copy-selection # Yank selection in copy mode.
# https://superuser.com/a/693990/298782
这样以后我就能够在复制模式中按下 y
来将内容拷贝到系统粘贴板。
X Windows 系统中常用的 Xterm,GNU Screen,SSH, GNome 中的 Terminal,KDE 中的 Konsole,Mac 下常用的 iTerm2 等,这些软件都属于 Terminal Emulator。 iTerm 等只是一个 GUI 软件,它的窗格只是窗格。而 Tmux 是终端复用,在一个命令行窗口中不仅可以显示多个 Shell 的内容,而且可以保持多个会话。 最重要的是:Tmux 和 Vim 一样属于字符终端软件,不需要任何 GUI 的支持,在远程登录时尤其有用。
升级了 tmux 之后,在启动 tmux 时如果遇到如下问题:
lost server
no server running on /tmp/tmux-1000/default
no server running on /tmp/tmux-1000/default
no sessions
检查一下 ~/.tmux.conf
配置,删除掉
set -g status-utf8 on
该配置已经被废弃。
最后我的 dotfiles 配置中有我过去多年使用的配置。
这些天折腾 Django 的时候用到了 MySQL,然而本地和VPS 上使用的版本不一致,本地使用了 5.7 版本,而 VPS 上使用了 5.5 的老版本,在数据迁移的时候遇到了 5.5 版本下不支持 DATETIME(6) 这样的数据类型。 DATETIME(6) 用来保存精确到微秒的时间。
环境:
系统:Debian 7, 按道理 Ubuntu/Debian 系应该都可以
无奈只能升级 MySQL 到 5.7 , 结果也比较顺利,官方有很详细 的升级说明:
wget http://dev.mysql.com/get/mysql-apt-config_x.y.z-1_all.deb # 从官网找到最新的版本,上面的链接中有
sudo dpkg -i mysql-apt-config_x.y.z-1_all.deb
sudo apt-get update
sudo apt-get install mysql-server
在中间弹出配置框时选择 5.7 ,应用,即可
mysql --version
最后记得运行
sudo mysql_upgrade -u root -p