Skip to content

Water-Ghost/Program_Notes

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

51 Commits
 
 
 
 
 
 

Repository files navigation

1. Program Notes

2. Contents

3. Python

3.1. Self-define colormap

有时候,你需要自定义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)

3.2. Clip data with shp, and save it to npz

# -*- 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

3.3. Get all files of a folder with os.walk

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)

3.4. Tests with pkgutils.get_data()

根据官方文档、《python3标准库》、《python cookbook》和网上资料,测试发现均不能正确出结果,随自行测试pkgutils.get_data函数的使用方法。

3.4.1. Package

Package structures are as follows:

pk
├── __init__.py
├── data
│   ├── __init__.py
│   ├── base.html
│   └── t1
│       └── test.csv
├── spam.py
├── test.py
├── test2.csv
└── test_.py

3.4.2. Codes

  • 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)

3.4.3. Output

$ 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

3.4.4. Discussions

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.

3.5. Get abs path of package

import os
import numpy as np

abs_dir = os.path.dirname(np.__file__)

3.6. Dingding Robot with requests

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)

3.7. Decorator factory

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)

3.8. Built-in Exceptions

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

3.9. FastAPI 运行不报错,但是有问题解决方法

MissingBackendError: bcrypt: no backends available -- recommend you install one (e.g. 'pip install bcrypt')

3.10. Pandas sort_values with specified orders

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

3.11. pandas.read_csv with null byte

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')

3.12. oracle

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')

3.13. Multi log files with loguru

loguru documentation

#!/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')

3.14. pygrib显示grib中的变量名

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")

4. Linux

4.1. Create list with fixed digits

#!/bin/bash

for i in `seq 0 1 24`
do
    echo `printf "%03d" "$i"`
done

4.2. lrzsz install

# 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"

4.3. shebang usage in python

"shebang" has two kinds of usages.
First,
#!/usr/bin/python3 # use this interpreter
Second,
#!/usr/bin/env python3 # use the first found python3 interpreter

4.4. killall

killall - kill processes by name

4.5. screen

# 创建会话(-m 强制)
$ screen -dmS session_name

# 关闭会话
$ screen -X -S [session_name] quit

# 查看所有会话
$ screen -ls

# 进入会话
$ screen -r session_name

5. Meteorology

5.1. Turn grib2 to nc with wgrib2

#!/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

5.2. Get wind direction name in Chinese

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

6. DataBase

6.1. Sqlite

6.1.1. Create table unique in multuple columns

# 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

7. Docker

7.1. tldr docker

  • 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

7.2. Common cmd

# 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的容器

7.3. Working with docker file

  • 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

8. Airflow

8.1. CMD

  • 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

9. Javascript

9.1. Xpath

  • 根结点和非根结点
    • /div 选择div节点,只有当它是文档的根结点时
    • //div 选择文档中所有的div节点
  • 通过属性选择节点
    • //@href 选择带href属性的所有节点
    • //a[@href='http://google.com'] 选择页面中所有指向Google网站的链接
  • 通过位置选择节点
    • //a[3] 选择文档中的第三个链接
    • //table[last()] 选择文档中的最后一个表
    • //a[postion() < 3] 选择文档中的前三个链接
  • 星号(*)匹配任意字符或节点,可以在不同条件下使用
    • //table/tr/* 选择所有表格行tr标签的所有子节点
    • //div[@*] 选择带任意属性的所有div标签

10. Django

10.1. pymysql

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()

11. MySQL

mysql.md

About

Program Notes

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published