- 1. Program Notes
- 2. Contents
- 3. Python
- 3.1. Self-define colormap
- 3.2. Clip data with shp, and save it to npz
- 3.3. Get all files of a folder with os.walk
- 3.4. Tests with pkgutils.get_data()
- 3.5. Get abs path of package
- 3.6. Dingding Robot with requests
- 3.7. Decorator factory
- 3.8. Built-in Exceptions
- 3.9. FastAPI 运行不报错,但是有问题解决方法
- 3.10. Pandas sort_values with specified orders
- 3.11. pandas.read_csv with null byte
- 3.12. oracle
- 3.13. Multi log files with loguru
- 3.14. pygrib显示grib中的变量名
- 4. Linux
- 5. Meteorology
- 6. DataBase
- 7. Docker
- 8. Airflow
- 9. Javascript
- 10. Django
- 11. MySQL
有时候,你需要自定义colormap,自定义代码如下:
# -*- coding: utf-8 -*-
"""
@author: Ghost
"""
import matplotlib.colors as col
import matplotlib.cm as cm
from matplotlib.colors import ListedColormap
from matplotlib.ticker import MaxNLocator
norm = col.Normalize(0, 1)
colors_ = [[norm(0), "#FFFFFF"], # 白色
[norm(0.02), "#E0FFFF"], # 淡青色
[norm(0.04), "#98FB98"], # 弱绿色
[norm(0.08), "#008000"], # 纯绿色
[norm(0.12), "#00FF00"], # 闪光绿
[norm(0.16), "#00BFFF"], # 深天蓝
[norm(0.2), "#000080"], # 海军蓝
[norm(0.4), "#FF1493"], # 深粉红
[norm(1), "#FF8C00"]] # 深橙色
# 创建名称为“rain_fall”的colormap
_cmap = col.LinearSegmentedColormap.from_list("rain_fall", colors_)
_cmap.set_over('#8B0000') # 深红色
_cmap.set_under("#FFFFFF") # 白色
# 注册“rain_fall”到matplotlib.cm,然后绘图就可以指定cmap=“rain_fall”
cm.register_cmap(cmap=_cmap)
#levels = MaxNLocator(nbins=50).tick_values(0, 50)
# -*- coding: utf-8 -*-
"""
@author: Ghost
"""
import geopandas as gpd
import netCDF4 as nc
import numpy as np
import sys
from datetime import datetime, timedelta
def get_clip_data(boundary, variables):
"""
Get clip area data(limit by boundary) from variables.
Args:
boundary: (minlon, minlat, maxlon, maxlat),
tyepes:(float, float, float, float);
clip boundary of data.
variables: list of np.array, [lon, lat, var1, var2, ...].
Return:
variables_cut: list of np.array. [lon_cut, lat_cut, var1_cut, ...]
"""
lon, lat, *_ = variables
minlon, minlat, maxlon, maxlat = boundary
lonmin = minlon // 1 - 1
lonmax = maxlon // 1 + 1
latmin = minlat // 1 - 1
latmax = maxlat // 1 + 1
limit = np.where((lon > lonmin) &
(lon < lonmax) &
(lat > latmin) &
(lat < latmax))
len0 = len(set(limit[0]))
len1 = len(set(limit[1]))
print(len0, len1)
variables_cut = [var[limit].reshape(len0, len1) for var in variables]
return variables_cut
def get_boundary_from_shp(shp_path):
"""Get (minlon, minlat, maxlon, maxlat) from shp_file.
Parameters:
shp_path.
Return:
tuple: (minlon, minlat, maxlon, maxlat)
"""
shp_file = '{}.shp'.format(shp_path)
_df = gpd.GeoDataFrame.from_file(shp_file)
bounds = _df.bounds
minlon = bounds.minx.min()
minlat = bounds.miny.min()
maxlon = bounds.maxx.max()
maxlat = bounds.maxy.max()
return minlon, minlat, maxlon, maxlat
def main(time):
'''
Extract met variables, turn them to npy, pkl and csv.
Args:
time: datetime object.
Return:
No return, files will be saved.
'''
# 0 Generate input and output path
time_str = time.strftime('%Y%m%d%H')
nc_dir = '/path/of/ncfile_dir'
nc_path = f'{nc_dir}/{time_str}.nc'
out_dir = '/path/of/outdir'
out_path = f'{out_dir}/{time_str}.npy'
shp_path = './shp/china'
# 1 Get data
nc_data = nc.Dataset(nc_path)
var_names = ['longitude', 'latitude',
'UGRD_10maboveground', 'VGRD_10maboveground']
# you can also add other names.
var_units = ['°', '°',
'm/s', 'm/s',]
# Corresponding units.
variables_ori = [nc_data.variables[var_name][:] for var_name in var_names]
lon, lat, *met_vars = variables_ori
lont, latt = np.meshgrid(lon, lat)
variables = [lont, latt]
for _ in met_vars:
variables.append(_[0])
# clip data.
boundary = get_boundary_from_shp(shp_path)
variables_cut = get_clip_data(boundary, variables)
# lon_cut, lat_cut, *met_vars_cut = variables_cut
np.savez(out_path,
data=variables_cut,
var_names=var_names,
var_units=var_units
)
if __name__ == "__main__":
try:
if len(sys.argv) == 1:
print("No args received, current time will be used.")
time = datetime.now()
main(time)
elif len(sys.argv) != 3:
print("Timestr and num needed, format '%Y%m%d%H' and 'int'.")
else:
print(len(sys.argv))
time = datetime.strptime(sys.argv[1], '%Y%m%d%H')
num = int(sys.argv[2])
for i in range(num):
main(time)
time += timedelta(hours=+1)
except (IOError, KeyboardInterrupt):
raise
import os
def get_files_of_dir(folder):
'''
Get all files of specified folder.
Parameters
----------
folder : str.
Returns
-------
file_list : list of str, list of filepath.
'''
file_list = []
for root, _dir, files in os.walk(folder):
for file in files:
file_list.append(os.path.join(root, file))
return file_list
folder = '/your/folder'
file_list = _files_of_dir(folder)
根据官方文档、《python3标准库》、《python cookbook》和网上资料,测试发现均不能正确出结果,随自行测试pkgutils.get_data
函数的使用方法。
Package structures are as follows:
pk
├── __init__.py
├── data
│ ├── __init__.py
│ ├── base.html
│ └── t1
│ └── test.csv
├── spam.py
├── test.py
├── test2.csv
└── test_.py
- spam.py
import pandas as pd
import pkgutil
import io
def test1():
# 正确返回结果
print('-' * 20, 'Test 1', '-' * 20)
base = pkgutil.get_data('data', 'base.html')
print(base.decode('utf-8'))
def test2():
# 正确返回结果
print('-' * 20, 'Test 2', '-' * 20)
csv = pkgutil.get_data('data', 't1/test.csv')
print(csv.decode('utf-8'))
print("\nRead data with pandas >>>")
df = pd.read_csv(io.BytesIO(csv))
print(df)
def test3():
# 返回None
print('-' * 20, 'Test 3', '-' * 20)
csv = pkgutil.get_data('data.t1', 'test.csv')
print(csv)
def test4():
# 返回None
print('-' * 20, 'Test 4', '-' * 20)
csv2 = pkgutil.get_data('pk', 'test2.csv')
print(csv2)
def test5():
# 正确返回结果
print('-' * 20, 'Test 5', '-' * 20)
csv3 = pkgutil.get_data('spam', 'test2.csv')
print(csv3.decode('utf-8'))
def test6():
# 正确返回结果
print('-' * 20, 'Test 6', '-' * 20)
csv4 = pkgutil.get_data('test', 'test2.csv')
print(csv4.decode('utf-8'))
def test7():
# 正确返回结果
print('-' * 20, 'Test 7', '-' * 20)
csv5 = pkgutil.get_data('test_', 'test2.csv')
print(csv5.decode('utf-8'))
test1()
test2()
test3()
test4()
test5()
test6()
test7()
- test.py
print("This is test")
- test_.py (emtpy)
$ python spam.py
-------------------- Test 1 --------------------
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
</body>
</html>
-------------------- Test 2 --------------------
name, age
jack, 1444
kim, 20
Read data with pandas >>>
name age
0 jack 1444
1 kim 20
-------------------- Test 3 --------------------
None
-------------------- Test 4 --------------------
None
-------------------- Test 5 --------------------
-------------------- Test 1 --------------------
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
</body>
</html>
-------------------- Test 2 --------------------
name, age
jack, 1444
kim, 20
Read data with pandas >>>
name age
0 jack 1444
1 kim 20
-------------------- Test 3 --------------------
None
-------------------- Test 4 --------------------
None
-------------------- Test 5 --------------------
key, value
name, jacks
age, 23
-------------------- Test 6 --------------------
This is test
key, value
name, jacks
age, 23
-------------------- Test 7 --------------------
key, value
name, jacks
age, 23
key, value
name, jacks
age, 23
-------------------- Test 6 --------------------
key, value
name, jacks
age, 23
-------------------- Test 7 --------------------
key, value
name, jacks
age, 23
From the output, when run to test5()
, the code restart run from test1()
to test7()
. And "This is test" should not be shown here, it was the result of test.py
. Finally, the results of test5()
, test6()
and test7()
was shown last.
Because, in test5()
and test6()
, when using pkgutil.get_data()
they refered to spam.py
and test.py
. They are files not dirctory, and are not empty.
The best solution is put your data to a sub dirctory, with a __init__.py
in it. And we can see, the outer directory name, pk
, is irrelevant.
import os
import numpy as np
abs_dir = os.path.dirname(np.__file__)
Auto sent message to Dingding Group, robot are secured with signature.
import time
import hmac
import hashlib
import base64
import urllib.parse
import requests
import json
# Generate timestamp and sign
timestamp = str(round(time.time() * 1000))
secret = 'Your secret key'
secret_enc = secret.encode('utf-8')
string_to_sign = '{}\n{}'.format(timestamp, secret)
string_to_sign_enc = string_to_sign.encode('utf-8')
hmac_code = hmac.new(secret_enc, string_to_sign_enc, digestmod=hashlib.sha256).digest()
sign = urllib.parse.quote_plus(base64.b64encode(hmac_code))
# Define headers
headers = {
'Content-Type': 'application/json',
}
# Define parameters
params = (
('access_token', 'your token'),
('timestamp', f'{timestamp}'),
('sign', sign),
)
# Define msg and jsonify the msg
msg = "我就是我, 是不一样的烟火123"
data = {"msgtype": "text", "text": {'content': msg}}
data = json.dumps(data)
# Send msg
response = requests.post('https://oapi.dingtalk.com/robot/send',
headers=headers,
params=params,
data=data)
print(response.text)
The way to build a decorator factory
import time
DEFAULT_FMT = '[{elapsed:0.8f}s] {name}({args}) -> {result}'
def clock(fmt=DEFAULT_FMT):
def decorate(func):
def clocked(*_args):
t0 = time.time()
_result = func(*_args)
elapsed = time.time() - t0
name = func.__name__
args = ', '.join(repr(arg) for arg in _args)
result = repr(_result)
print(fmt.format(**locals()))
return _result
return clocked
return decorate
@clock()
def snoonze(seconds):
time.sleep(seconds)
for i in range(3):
snoonze(.123)
BaseException
+-- SystemExit
+-- KeyboardInterrupt
+-- GeneratorExit
+-- Exception
+-- StopIteration
+-- StopAsyncIteration
+-- ArithmeticError
| +-- FloatingPointError
| +-- OverflowError
| +-- ZeroDivisionError
+-- AssertionError
+-- AttributeError
+-- BufferError
+-- EOFError
+-- ImportError
| +-- ModuleNotFoundError
+-- LookupError
| +-- IndexError
| +-- KeyError
+-- MemoryError
+-- NameError
| +-- UnboundLocalError
+-- OSError
| +-- BlockingIOError
| +-- ChildProcessError
| +-- ConnectionError
| | +-- BrokenPipeError
| | +-- ConnectionAbortedError
| | +-- ConnectionRefusedError
| | +-- ConnectionResetError
| +-- FileExistsError
| +-- FileNotFoundError
| +-- InterruptedError
| +-- IsADirectoryError
| +-- NotADirectoryError
| +-- PermissionError
| +-- ProcessLookupError
| +-- TimeoutError
+-- ReferenceError
+-- RuntimeError
| +-- NotImplementedError
| +-- RecursionError
+-- SyntaxError
| +-- IndentationError
| +-- TabError
+-- SystemError
+-- TypeError
+-- ValueError
| +-- UnicodeError
| +-- UnicodeDecodeError
| +-- UnicodeEncodeError
| +-- UnicodeTranslateError
+-- Warning
+-- DeprecationWarning
+-- PendingDeprecationWarning
+-- RuntimeWarning
+-- SyntaxWarning
+-- UserWarning
+-- FutureWarning
+-- ImportWarning
+-- UnicodeWarning
+-- BytesWarning
+-- ResourceWarning
MissingBackendError: bcrypt: no backends available -- recommend you install one (e.g. 'pip install bcrypt')
from pandas.api.types import CategoricalDtype
df = pd.DataFrame({
'cloth_id': [1001, 1002, 1003, 1004, 1005, 1006],
'size': ['S', 'XL', 'M', 'XS', 'L', 'S'],
})
cat_size_order = CategoricalDtype(
['XS', 'S', 'M', 'L', 'XL'],
ordered=True
)
df['size'] = df['size'].astype(cat_size_order)
df
>>>
cloth_id size
0 1001 S
1 1002 XL
2 1003 M
3 1004 XS
4 1005 L
5 1006 S
What you need is to replace b'\x00' with b''.
import pandas as pd
csv_file = '/you/file/folder/test.csv'
new_file = '/you/file/folder/new.csv'
with open(csv_file, 'rb') as f:
csv = f.read().replace('\x00'.encode(), ''.encode()).decode() # .decode('gb18030')
with open(new_file, 'w') as f:
f.write(csv)
# If you don't want to save file, you can simply use io.BytesIO.
with open(path, 'rb') as f:
csv_byte = f.read().replace('\x00'.encode(), ''.encode())
df = pd.read_csv(BytesIO(csv_byte), skiprows=4, encoding='gb18030')
import cx_Oracle
cx_Oracle.init_oracle_client(lib_dir='/clint/path/instantclient_19_8')
engine = create_engine('oracle+cx_oracle://user:pasword@ip:port/instancename')
#!/usr/bin/env python
# -*- coding: utf-8 -*-
import os
import sys
from pathlib import Path
from loguru import logger
here = Path(__file__).parent
# 1 Write log to file based on content in message
logger.add('normal.log', filter=lambda x: '[normal]' in x['message'],
retention="10 days", rotation="12:00", level="DEBUG")
logger.add('error.log', filter=lambda x: '[error]' in x['message'],
retention="10 days", rotation="12:00", level="DEBUG")
# 2 Write all log to file
logger.add('all.log', retention="10 days", rotation="12:00", level="DEBUG")
# 3 Write to file, based on level
logger.add('warning.log', filter=lambda x: 'WARNING' in x['level'].name)
if __name__ == '__main__':
logger.info('[normal] this is a normal')
logger.error('[error] this is an error')
logger.info('[all] this is')
logger.warning('[warning] this is')
logger.warning('test')
import pygrib
file = '<your/grib/file>'
grbs = pygrib.open(file)
# 写入变量到1.txt
with open('1.txt', 'w') as f:
grbs.seek(0)
for grb in grbs:
print(grb.__repr__())
f.write(f"{grb.__repr__()}\n")
#!/bin/bash
for i in `seq 0 1 24`
do
echo `printf "%03d" "$i"`
done
# If you don't have sudo permission, you cann't "yum -y install epel-release".
# epel include tools: screen, lrzsz, tree, locate, and htop.
# Then, this is what you need.
# 1 get the lrzsz.taz.gz file.
# from "http://freshmeat.sourceforge.net/projects/lrzsz/" or other sources.
tar -zxvf lrzsz-0.12.20.tar.gz
cd lrzsz-0.12.20
./configure --prefix=/your/path/lrzsz
make && make install
# 2 Don't forget to add lrzsz path to PATH.
vi ~/.bashrc
# add the following code to it.
export PATH="/your/path/lrzsz/bin":$PATH
alias rz="lrz"
alias sz="lsz"
"shebang" has two kinds of usages.
First,
#!/usr/bin/python3
# use this interpreter
Second,
#!/usr/bin/env python3
# use the first found python3 interpreter
killall - kill processes by name
# 创建会话(-m 强制)
$ screen -dmS session_name
# 关闭会话
$ screen -X -S [session_name] quit
# 查看所有会话
$ screen -ls
# 进入会话
$ screen -r session_name
#!/bin/bash
# Assumed that wgrib2 had been installed.
# 1 Generate vars
vars="(:RH:2 m ab|:TMP:2 m ab|:APCP:surface:|:DSWRF:surface|:PRES:surface|:DLWRF:surface|:PEVPR:surface|:HPBL:surface|:UGRD:10 m above|:VGRD:10 m above|:TMP:850 mb|:HGT:850 mb|:UGRD:850 mb|:VGRD:850 mb|:TMP:500 mb|:HGT:500 mb|:RH:850 mb:|:VVEL:0.995 sigma level:)"
# 2 Extract vars and turn to nc
# input
gfs_dir="/gfs/dir"
gfs_name="gfs.t00z.pgrb2.0p25.f040"
# output
out_dir="/out/dir"
nc_name="2019010100"
wgrib2 ${gfs_dir}/${gfs_name}.grb -s | egrep "`echo $vars`" | wgrib2 -i ${gfs_dir}/${gfs_name}.grb -netcdf ${out_dir}/${nc_name}.nc
def get_wind_direction(degree):
'''
Get wind direction name in Chinese of degree in wind.
Parameters
----------
degree : int or float.
Wind direction, [0, 360].
Returns
-------
wd_CN : str
Wind direction name in Chinese.
'''
wds = ['北风', '东北风', '东风', '东南风',
'南风', '西南风', '西风', '西北风', '北风']
degrees = np.arange(22.5, 390, 45)
index = np.argwhere(degrees > degree)[0, 0]
# print(index)
wd_CN = wds[index]
return wd_CN
# Open sqlite database
$ .sqlite3 <filename.db>
# check tables
$ .table
# create table, unique in TIME and SITE
$ CREATE TABLE <table_name> (ID interger primary key autoincrement,
TIME date NOT NULL,
SITE text NOT NULL, VALUE real,
UNIQUE(TIME, SITE) ON CONFILICT REPLACE);
# headers on
$ .headers on
-
List currently running docker containers:
docker ps
-
List all docker containers (running and stopped):
docker ps -a
-
Start a container from an image, with a custom name:
docker run --name container_name image
-
Start or stop an existing container:
docker start|stop container_name
-
Pull an image from a docker registry:
docker pull image
-
Open a shell inside of an already running container:
docker exec -it container_name sh
-
Remove a stopped container:
docker rm container_name
-
Fetch and follow the logs of a container:
docker logs -f container_name
# Create a container
$ docker run -i -t ubuntu /bin/bash
# -i 保证容器的STDIN是开启的,-t 告诉Docker为容器分配一个伪tty终端
# Name a container
$ docker run --name container_name -i -t ubuntu /bin/bash
# update and install vim
$ apt-get update && apt-get install vim
# Restart a stoped container
$ docker start container_name/id
# Delete a container
$ docker rm --force contain_name/id
# Delete a contianer and its volume
$ docker rm -v contain_name
# Attach to a container
$ docker attach container_name/id
# Create a daemonized container
$ docker run --name daemon_dave -d ubuntu /bin/sh -c "while true; do echo hello world; sleep 1; done"
# -d Docker会将容器放到后台运行
# Check the log
$ docker logs daemon_dave
$ docker logs -f daemon_dave # 跟踪日志
$ docker logs -ft daemon_dave # 跟踪日志,并添加时间戳
# Statistics
$ docker top daemon_dave
$ docker stats daemon_dave <others> ...
# Execute tasks
$ docker exec -d daemon_dave touch /etc/new_config_file
# -d 表明需要运行一个后台进程
# Execute a interactive cmd in a container
$ docker exec -i -t daemon_dave /bin/bash
# Stop a daemonized container
$ docker stop daemon_dave/id
# Check the last x container
$ docker ps -n x
# Create a auto start container
$ docker run --restart=always --name daemon_dave -d ubuntu /bin/sh -c "while true; do echo hello world; sleep 1; done"
# Inspect
$ docker inspect daemon_dave
# List the images
$ docker images
# Pull a image
$ docker pull ubuntu:14.04
# Create a container with a tag
$ docker run -t -i --name new_container ubuntu:14:04 /bin/bash
# Delete images, you can delete it if only the correspond container not exists.
$ docker rmi container_name/id
# Search images
$ docker search puppet
# Pull a image
$ docker pull ansible/centos7-ansible
# Create a container
$ docker run -i -t ansible/centos7-ansible /bin/bash
# Create a container with nginx
$ docker run -d --name web container_nginx nginx -g "daemon off;"
# container_nginx 是带有nginx的容器
- Dockerfile
# Version: 0.0.1
FROM ubuntu:14.04
LABEL maintainer="zyz" mail='[email protected]'
VOLUME [ "/data" ]
RUN mkdir /work
WORKDIR /work
ENV DEBIAN_FRONTEND noninteractive
RUN apt-get update && apt-get install -y wget
# Download miniconda.sh and install
RUN wget --quiet https://repo.anaconda.com/miniconda/Miniconda3-latest-Linux-x86_64.sh \
-O /work/miniconda.sh
RUN /bin/bash /work/miniconda.sh -b -p /work/miniconda && rm /work/miniconda.sh
RUN echo "export PATH=/work/miniconda/bin:$PATH" >> ~/.bashrc
RUN ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime
- Command Line Metadata Validation
# print the list of active DAGs
airflow list_dags
# prints the list of tasks the "tutorial" dag_id
airflow list_tasks tutorial
# prints the hierarchy of tasks in the tutorial DAG
airflow list_tasks tutorial --tree
- Testing
# command layout: command subcommand dag_id task_id date
# testing print_date
airflow test tutorial print_date 2015-06-01
# testing sleep
airflow test tutorial sleep 2015-06-01
- Backfill
# optional, start a web server in debug mode in the background
# airflow webserver --debug &
# start your backfill on a date range
airflow backfill tutorial -s 2015-06-01 -e 2015-06-07
- 根结点和非根结点
- /div 选择div节点,只有当它是文档的根结点时
- //div 选择文档中所有的div节点
- 通过属性选择节点
- //@href 选择带href属性的所有节点
- //a[@href='http://google.com'] 选择页面中所有指向Google网站的链接
- 通过位置选择节点
- //a[3] 选择文档中的第三个链接
- //table[last()] 选择文档中的最后一个表
- //a[postion() < 3] 选择文档中的前三个链接
- 星号(*)匹配任意字符或节点,可以在不同条件下使用
- //table/tr/* 选择所有表格行tr标签的所有子节点
- //div[@*] 选择带任意属性的所有div标签
Error:
django.core.exceptions.ImproperlyConfigured: mysqlclient 1.4.0 or newer is required; you have 0.10.1.
import pymysql
pymysql.version_info = (1, 4, 6, 'final', 0)
pymysql.install_as_MySQLdb()