Skip to content

Latest commit

 

History

History
1018 lines (746 loc) · 15.1 KB

README.md

File metadata and controls

1018 lines (746 loc) · 15.1 KB

Python Quickstart

Premise: you know programming but know no (or only a little of) Python.

Note: This is for Python 3, NOT legacy Python (2.7)

Introduction

  • Python: a slow, statically typed (but checked at runtime) scripting language.
  • Sorry for choosing it, here are some steps to make it less painful
  • YMHH (You Might Have Heard):
    • whitespace sensitive
    • slow because of the GIL
  • Our friend Google DuckDuckGo

Virtual environments, packages

mkdir python-quickstart
cd python-quickstart
pyvenv env # Not relocable !!!
source env/bin/activate
# our just
./env/bin/python
./env/bin/pip

import this

print("Hello World")

Packages are pulled from PyPi (not to be confused with PyPy an alternative to CPython)

pip install pudb
pip install pudb==2016.2
pip install pudb>=2016.0
# requirements.txt
requests==2.13.0
# requirements_dev.txt
-r requirements.txt
pudb==2016.2
# can also contain URLs
# command line:
pip install -r requirements_dev.txt
pip freeze > requirements_frozen.txt
pip list --outdated --format=columns
__pycache__/
*.py[cod]
...
env/
...
htmlcov/
.tox/
.coverage
.coverage.*
...
  • Let's talk about native packages. Some strategies:

    • Hope that the packager included wheels
    • Install from package manager and afterwards do pyvenv --system-site-packages env
    • Have a complete build-chain ready (build-essential, python3-dev, libpq-dev, ...)
  • pipdeptree

Booleans, numbers, strings, lists, tuples, dicts, sets, range

None

def noReturn():
  pass

print(noReturn())
True
False
0o777
0b101
1234
0xABC
10**123
10/3
10//3
type(1234)
str(1234)
repr(1234)
hash(1234)
id(1234)
'foo\tbar'
"foo\x09bar"
"""foo"""
'''foo'''
r'foo\tbar'
'foo%d: %.2f (%r)' % (1, 2, 3)
'foo{0}'.format(1)
b'foo'.decode('utf-8')
[]
[1, "abc", 2]
[1, "abc", 2] + [3]
list((1, 2, 3))
l = [1, "abc", 2]
l[0]
len(l)
l[-1]
l[::2]
l[::-1]
del l[1]
a, *b = [1, 2, 3]
(1, 2)
(1,)
(1, 2)[::-1]
{1: 2, "foo": "bar"}
dict(fiz=2, foo="bar")
d = {1: 2, "foo": "bar"}
d[1]
d[2]
d.keys()
d.values()
d.items()
1 in d
2 not in d
d.update({1: 2, 3: 5})
d.get(2)
d.pop(2)
d.pop(2, None)
d.pop(1)
d.pop(1)
del d['foo']
{1, 2, 3.1415}
set((1, 2, 3.1415))
1 in {1, 2}
{}
range(0, 10)
list(range(0, 10))
list(range(0, 10, 3))

More: https://docs.python.org/3.5/library/stdtypes.html

(1, 2) < (1, 2, 3)
sorted((dict(a=1, b=2), dict(a=2, b=3)), key=lambda e: (e['a'], e['b']))

Comments

# this is a comment
'''
This is a multi-line comment - almost
'''

Ordered dicts, frozensets, named touple, counter

import collections

counter = collections.Counter()
counter.update((1, 1, 1, 1))
counter.update({1: 4})
counter[1] = 0
counter[42]
collections.OrderedDict
dd = collections.defaultdict(list)
dd[1].append(5)
# from https://docs.python.org/3.5/library/collections.html#collections.namedtuple
Point = collections.namedtuple('Point', ['x', 'y'])

Control structures

if a:
  pass
elif b:
  pass
else:
  pass


if 1 < 3 <= 3:
  pass

if all/any(...):
  ...

if a is (not) b:
  ...

if a in (not) b: # including strings
  ...

a = 0 if True else False
for i in {1: 2, 3: 4}: print(i)
for i in {1: 2, 3: 4}.items(): print(i)
for k, v in {1: 2, 3: 4}.items(): print(k + v)
for k, v in {1: 2, 3: 4}.items(): print('%s -> %s' % (k, v))
for k, _ in {1: 2, 3: 4}.items(): print(k)
for _, _ in {1: 2, 3: 4}.items(): print("Hello")
while True:
  break / continue
for i in (1, 2, 3):
  if i == 2:
    print("Found!")
    break;
else:
  print("Not found!")
for i in enumerate((1, 2, 3)):
  print(i)

https://docs.python.org/3.5/library/functions.html

Modules

dir/__init__.py
import dir
import dir.foo
from dir import foo
import sys
sys.path
import foo
foo.__file__
# config_local.py - added to .gitignore
CONFIG1 = 5

# config.py
from config_local import *

CONFIG1

# server.py
import config

print(config.CONFIG1)

!!! Modules with the same name on the import path are not merged

Circular imports:

# fizz.py
import buzz

def doFizz():
  pass

# buzz.py
import fizz

fizz.doFizz()

def doBuzz():
  pass

# somewhere else
import fizz, buzz
# or
import buzz, fizz
  • Also: zipimport

Testing, TDD

import unittest


class TestTest(unittest.TestCase):
    def testMe(self):
        self.assertEqual(1, 2)

# assertRaises
# assertRaisesRegex
# assertLogs
# assertIs
# assertIsNot
# assertIsNone
# assertIsNotNone
# assertIn
# assertNotIn
# assertIsInstance
# assertNotIsInstance

# self.maxDiff = None

if __name__ == '__main__':
    unittest.main()
python -m unittest discover tests
python -m unittest test
python -m unittest -f test
pip install pytest
def testMe():
    assert 1 == 2
def testMe():
    assert 1 == 2
def inc(x):
    return x + 1

def testAnswer():
    assert inc(3) == 5

Function, decorators

  • Named parameters
def sum(a, b):
    return a + b


print(sum(b=1, a=2))
  • "varargs"
def dump(*args, **kwargs):
    print(args)
    print(kwargs)


dump(1, 2, 3, 4, foo='bar')
  • default parameters
def foo(a=1, b=2):
    return a+b

print(foo(2))
print(foo(b=4))

Don't use mutable default parameters - they are evaluated only once per module loading

# !!! WRONG !!!
def appendTo(a=[], b=[]):
    a += b
    return a

print(appendTo(b=[1]))
print(appendTo(b=[2]))
# BETTER
def appendTo(a=[], b=[]):
    return a + b

print(appendTo(b=[1]))
print(appendTo(b=[2]))
# BESTEST
def appendTo(a=None, b=None):
    a = a or []
    b = b or []
    return a + b

print(appendTo(b=[1]))
print(appendTo(b=[2]))
  • Closures
def incrementFactory(w):
    def inc(x):
        return x + w
    return inc


inc = incrementFactory(5)
print(inc(1))
  • side-effecting closures
  1. Don't do it
  2. See rule 1
# !!! WON'T WORK !!!
def foo():
    called = False
    def do():
        called = True
    do()
    assert called

foo()
def foo():
    called = [False]
    def do():
        called[0] = True
    do()
    assert called[0]

foo()
  • Lambdas / anonymous functions
def incrementFactory(w):
    return lambda x: x + w


inc = incrementFactory(5)
print(inc(1))
import functools


def sum(a, b):
    return a + b


def incrementFactory(w):
    return functools.partial(sum, w)

inc = incrementFactory(5)
print(inc(1))
  • decorators
def decorate(f):
    def decorated(*args, **kwargs):
        print('Before!')
        f(*args, **kwargs)
        print('After!')
    return decorated


@decorate
def test(*args):
    return sum(args)


print(test(1, 2, 3))
import functools


def decorate(f):
    @functools.wraps(f)
    def decorated(*args, **kwargs):
        print('Before!')
        try:
            return f(*args, **kwargs)
        finally:
            print('After!')
    return decorated


@decorate
def test(*args):
    return sum(args)


print(test(1, 2, 3))

Exceptions

try:
    ...
except:
    ...


try:
    ...
except ValueError:
    ...


try:
    ...
except (ValueError, TypeError) as e:
    ...


try:
    ...
except ...:
    ...
finally:
    ...
    

raise ExceptionClass()

Classes

  • Don't overuse them
class Foo(object):
    def foo(self):
        print(1)


def Fizz(object):
    def foo(self):
        print(2)

    def bar(self):
        print(3)


class Bar(Foo, Fizz):
    def foo(self, param):
        super().foo()
        print(4)

bar = Bar()
bar.foo(1)
bar.bar()
print(Bar.__mro__)

See: https://en.wikipedia.org/wiki/C3_linearization

class Foo(object):
    def __init__(self, a, b):
        self._a = a
        self.__b = b


class Bar(Foo):
    def __init__(self, a):
        self.__b = a
        super().__init__(a, a)


b = Bar(2)
print(b.__dict__)
class Foo(object):
    def do(self):
        print(self.PARAM)


class Bar(Foo):
    PARAM = 5


Bar().do()
class Foo(object):
    def __init__(self, a):
        self.dodo()
        self.a = a


class Bar(Foo):
    def dodo(self):
        print(self.a)


Bar(1)
def getValue():
    print('Getting value')
    return []


class Foo(object):
    a1 = getValue()

    def __init__(self):
        self.a2 = getValue()


f1 = Foo()
f2 = Foo()

assert f1.a1 is f2.a1
assert f1.a2 is not f2.a2
class Foo(object):
    def iMethod(self):
        print(1)

    @staticmethod
    def sMethod():
        print(2)

    @classmethod
    def cMethod(cls):
        print(cls)


class Bar(Foo):
    def iMethod(self):
        super().iMethod()

    @staticmethod
    def sMethod():
        Foo.sMethod()

    @classmethod
    def cMethod(cls):
        super().cMethod()


b = Bar()
# b.iMethod()
# b.sMethod()
# b.cMethod()

# Bar.sMethod()
# Foo.sMethod()
# Bar.cMethod()
# Foo.cMethod()

# Foo.iMethod(None)
class AttrDict(object):
    def __init__(self, **kwargs):
        for k, v in kwargs.items():
            setattr(self, k, v)


d = AttrDict(a=1, b=2)
print(d.a)
class AttrDict(dict):
    __getattribute__ = dict.__getitem__


d = AttrDict(a=1, b=2)
print(d.a)
class Klazz(object):
    def __call__(self, *args):
        return args


def fun(*args):
    return args


bubu = Klazz()
print(bubu(1, 2, 3))

bubu = fun
print(bubu(1, 2, 3))
class Props(objects):
    x = property(getx, setx, delx, "I'm the 'x' property.")

class Props(objects):
    def __init__(self):
        self._x = None

    @property
    def x(self):
        """I'm the 'x' property."""
        return self._x

    @x.setter
    def x(self, value): # <-- duplicate names ???
        self._x = value

    @x.deleter
    def x(self):
        del self._x

Scoping

  1. Only functions introduce new scope
  2. *Weird exception for classes
  3. after the outermost scope the builtins are searched
u = 1

class Foo(object):
    u = 2

    class Bar(object):
        u = 3

        def do(self):
            u = 4
            print(u)

Foo.Bar().do()

* Using nested classes to hide abstract superclass from unittest * The "Meta" class for Django

Mocking

import random


def randomPlusOne():
    return random.randint(0, 1000000) + 1
from unittest import mock

import src


@mock.patch('src.random')
def testRandomPlusOne(random_mock):
    random_mock.randint.return_value = 1
    assert 2 == src.randomPlusOne()

mock.Mock / mock.MagicMock

List/Dict/Set comprehensions

[i**2 for i in range(0, 10) if i % 2 == 0]

{i**2 for i in range(0, 10) if i % 2 == 0}

{i: i**2 for i in range(0, 10) if i % 2 == 0}

Generators

def gen():
    for i in range(0, 100):
        yield i


print(sum(gen()))

Itertools and more itertools

Context managers

with open(__file__, 'r') as f:
    for l in f:
        print(l)
import contextlib


@contextlib.contextmanager
def greet(name):
    print('Hello %s' % name)
    yield
    print('Goodbye %s' % name)


with greet('world'):
    print(42)
with foo() as f_in, \
      bar() as f_out:
    ...

Serialization

  • JuJ: Just Use Json
  • Pickle and YAML are both security risks!

Static analyzers, coverage

import random


def randomPlusOne():
    return random.randint(0, 1000000) + 1
import unittest
from unittest import mock

import src


class TestTest(unittest.TestCase):
    @mock.patch('src.random')
    def testRandomPlusOne(self, random_mock):
        random_mock.randint.return_value = 1
        self.assertEqual(2, src.randomPlusOne())
pip install flake8
pip install pylint
flake8 src.py
pylint src.py
coverage run --branch --module unittest test
coverage report

Debugging

  • pdb / pudb
  • set_trace()
  • no cross-thread BPs
  • python -m p[u]db

Logging

import logging

logging.error('Huston: %s', 'We have a problem')

try:
    raise ValueError()
except ValueError:
    logging.exception('Error!')

WSGI, Flask

import flask
app = flask.Flask(__name__)


@app.route("/")
def hello():
    return 1 + "Hello World!"


if __name__ == "__main__":
    app.run(debug=True)

Some well-known modules:

Podcasts

Other tidbits:

Good luck!