Skip to content

python_setup

Guo Ling edited this page Sep 1, 2020 · 13 revisions

MMKV for Python (on POSIX)

MMKV is an efficient, small, easy-to-use mobile key-value storage framework used in the WeChat application. It's currently available on both Android, iOS/macOS, Win32 and Python (POSIX).

Getting Started

Prerequisites

  • Python 2.7 and above.
  • CMake 3.8.0 and above.
  • C++ compiler that supports C++ 17 standard.
  • Linux(Ubuntu, Arch Linux, CentOS, Gentoo), Unix(macOS, FreeBSD, OpenBSD) are supported.

Installation Via Source

  1. Getting source code from the git repository:
git clone https://github.com/Tencent/MMKV.git
  1. Prepare pybind11.
    2.1 Get pybind11 (v2.5.0) source code:
cd MMKV
git submodule update --init --recursive

2.2 On Linux, install python-dev for Python 2, or python3-dev for Pythond 3. On macOS, you don't need to install those things. See also pybind11 docs.
For example, let's say we are using Python 3 on Ubuntu:

sudo apt-get install python3-dev
  1. Build the MMKV native library for Python:
cd POSIX/Python
mkdir build
cd build
cmake ..
make -j8

You can find out which Python is used to build MMKV from the logs of cmake. It's something like:

-- Found PythonInterp: /usr/bin/python3.6 (found version "3.6.9") .

  1. Copy the generated libmmkv.so (the exact name is vary on different platforms, for example mmkv.cpython-36m-x86_64-linux-gnu.so) to your Python library directory PYTHONPATH. If you are using Python 3, copying the so file to your working directory is just fine.

  2. Test MMKV:
    Make sure we are using the same Python that builds MMKV in step 3:

python3 --version

Run the test:

cp ../demo.py ./
python3 demo.py
cp ../unit_test.py ./
python3 unit_test.py

Tutorial

You can use MMKV as you go. All changes are saved immediately, no save, no sync calls needed.

Configuration

  • Setup MMKV on App startup, say in your main() function, add these code:

    import mmkv
    
    if __name__ == '__main__':
        mmkv.MMKV.initializeMMKV('/tmp/mmkv')

CRUD Operations

  • MMKV has a global instance, that can be used directly:

    kv = mmkv.MMKV.defaultMMKV()
    
    kv.set(True, 'bool')
    print('bool = ', kv.getBool('bool'))
    
    kv.set(-1 * (2 ** 31), 'int32')
    print('int32 = ', kv.getInt('int32'))
    
    kv.set((2 ** 32) - 1, 'uint32')
    print('uint32 = ', kv.getUInt('uint32'))
    
    kv.set(2 ** 63, 'int64')
    print('int64 = ', kv.getLongInt('int64'))
    
    kv.set((2 ** 64) - 1, 'uint64')
    print('uint64 = ', kv.getLongUInt('uint64'))
    
    kv.set(3.1415926, 'float')
    print('float = ', kv.getFloat('float'))
    
    kv.set('Hello world, MMKV for Python!', 'string')
    print('string = ', kv.getString('string'))
    
    lst = range(0, 10)
    kv.set(bytes(lst), 'bytes')
    bt = kv.getBytes('bytes')
    print('raw bytes = ', bt, ', decoded bytes = ', list(bt))

    As you can see, MMKV is quite easy to use.

  • Deleting & Querying:

    kv = mmkv.MMKV.defaultMMKV()
    print('keys before remove:', sorted(kv.keys()))
    
    kv.remove('bool')
    print('"bool" exist after remove: ', ('bool' in kv))
    
    kv.remove(['int32', 'float'])
    print('keys after remove:', sorted(kv.keys()))
  • If different modules/logics need isolated storage, you can also create your own MMKV instance separately:

    kv = mmkv.MMKV('test_python')
    kv.set(True, 'bool')
  • If multi-process accessing is needed,you can set MMKV_MULTI_PROCESS on MMKV initialization:

    kv = mmkv.MMKV('test_python', mmkv.MMKVMode.MultiProcess)
    kv.set(True, 'bool')

Supported Types

  • Primitive Types:

    • Boolean, int, float

    Note: Due to the limitation of protobuf, int32, uint32, int64, uint64 has different encoding & decoding protocol. You should call different getXXXInt() method according to the value's possible range. For each range of those integer types, checkout MMKV/POSIX/Python/unit_test.py.

  • Classes:

    • str, bytes

Logs

  • By default, MMKV doesn't print any logs. You can turn on logging in initialization:

    mmkv.MMKV.initializeMMKV('/tmp/mmkv', mmkv.MMKVLogLevel.Info)
  • MMKV prints log to stdout, which is not convenient for the interpreter or diagnosing online issues. You can setup MMKV log redirecting on App startup. Implement a callback function with signature (logLevel: mmkv.MMKVLogLevel, file: str, line: int, function: str, message: str) -> None, register it as log handler. Remember to unregister it before exit, otherwise your script won't exit properly.

def logger(log_level, file, line, function, message):
    level = {
        mmkv.MMKVLogLevel.NoLog : 'N',
        mmkv.MMKVLogLevel.Debug : 'D',
        mmkv.MMKVLogLevel.Info : 'I',
        mmkv.MMKVLogLevel.Warning : 'W',
        mmkv.MMKVLogLevel.Error : 'E'
    }
    # use your logging tools instead of print()
    print('[{0}] <{1}:{2}:{3}> {4}'.format(level[log_level], file, line, function, message))


if __name__ == '__main__':
    # enable logging
    mmkv.MMKV.initializeMMKV('/tmp/mmkv', mmkv.MMKVLogLevel.Info)

    # redirect logging
    mmkv.MMKV.registerLogHandler(logger)
    
    # some logic
    ...
    
    # unregister before exit, otherwise the script won't exit properly
    mmkv.MMKV.unRegisterLogHandler()

    # or just call onExit() will do the job
    # mmkv.MMKV.onExit()
Clone this wiki locally