Skip to content
This repository has been archived by the owner on Feb 8, 2018. It is now read-only.

Commit

Permalink
Updated to use current_payment_instructions view
Browse files Browse the repository at this point in the history
current_payment_instructions view used in place of the new
payment_instructions_due table which has now been removed
  • Loading branch information
rorepo committed Sep 3, 2015
1 parent 710399e commit feeec33
Show file tree
Hide file tree
Showing 2 changed files with 201 additions and 3 deletions.
203 changes: 201 additions & 2 deletions gratipay/models/participant.py
Original file line number Diff line number Diff line change
Expand Up @@ -809,8 +809,207 @@ def update_avatar(self):
self.set_attributes(avatar_url=avatar_url)


# Giving and Taking
# =================
# Random Junk
# ===========

@property
def profile_url(self):
base_url = gratipay.base_url
username = self.username
return '{base_url}/{username}/'.format(**locals())


def get_teams(self, only_approved=False, cursor=None):
"""Return a list of teams this user is the owner of.
"""
teams = (cursor or self.db).all( "SELECT teams.*::teams FROM teams WHERE owner=%s"
, (self.username,)
)
if only_approved:
teams = [t for t in teams if t.is_approved]
return teams


def get_old_teams(self):
"""Return a list of old-style teams this user was a member of.
"""
return self.db.all("""
SELECT team AS name
, ( SELECT count(*)
FROM current_takes
WHERE team=x.team
) AS nmembers
FROM current_takes x
WHERE member=%s;
""", (self.username,))


def insert_into_communities(self, is_member, name, slug):
participant_id = self.id
self.db.run("""
INSERT INTO community_members
(ctime, name, slug, participant, is_member)
VALUES ( COALESCE (( SELECT ctime
FROM community_members
WHERE participant=%(participant_id)s
AND slug=%(slug)s
LIMIT 1
), CURRENT_TIMESTAMP)
, %(name)s, %(slug)s, %(participant_id)s, %(is_member)s
)
""", locals())


def change_username(self, suggested):
"""Raise Response or return None.
Usernames are limited to alphanumeric characters, plus ".,-_:@ ",
and can only be 32 characters long.
"""
# TODO: reconsider allowing unicode usernames
suggested = suggested and suggested.strip()

if not suggested:
raise UsernameIsEmpty(suggested)

if len(suggested) > USERNAME_MAX_SIZE:
raise UsernameTooLong(suggested)

if set(suggested) - ASCII_ALLOWED_IN_USERNAME:
raise UsernameContainsInvalidCharacters(suggested)

lowercased = suggested.lower()

if lowercased in gratipay.RESTRICTED_USERNAMES:
raise UsernameIsRestricted(suggested)

if suggested != self.username:
try:
# Will raise IntegrityError if the desired username is taken.
with self.db.get_cursor(back_as=tuple) as c:
add_event(c, 'participant', dict(id=self.id, action='set', values=dict(username=suggested)))
actual = c.one( "UPDATE participants "
"SET username=%s, username_lower=%s "
"WHERE username=%s "
"RETURNING username, username_lower"
, (suggested, lowercased, self.username)
)
except IntegrityError:
raise UsernameAlreadyTaken(suggested)

assert (suggested, lowercased) == actual # sanity check
self.set_attributes(username=suggested, username_lower=lowercased)

return suggested

def update_giving_and_tippees(self):
with self.db.get_cursor() as cursor:
updated_tips = self.update_giving(cursor)
for tip in updated_tips:
Participant.from_username(tip.tippee).update_receiving(cursor)

def update_giving(self, cursor=None):
updated = []
# Update is_funded on tips
if self.get_credit_card_error() == '':
updated = (cursor or self.db).all("""
UPDATE current_payment_instructions
SET is_funded = true
WHERE participant = %s
AND is_funded IS NOT true
RETURNING *
""", (self.username,))

giving = (cursor or self.db).one("""
UPDATE participants p
SET giving = COALESCE((
SELECT sum(amount)
FROM current_payment_instructions s
JOIN teams t ON t.slug=s.team
WHERE participant=%(username)s
AND amount > 0
AND is_funded
AND t.is_approved
), 0)
WHERE p.username=%(username)s
RETURNING giving
""", dict(username=self.username))
self.set_attributes(giving=giving)

return updated

def update_due(self, team, id, cursor=None):
"""Transfer existing due value to newly inserted record
"""
# Copy due to new record
(cursor or self.db).run("""
UPDATE payment_instructions p
SET due = COALESCE((
SELECT due
FROM payment_instructions s
WHERE participant=%(username)s
AND team = %(team)s
AND due > 0
), 0)
WHERE p.id = %(id)s
""", dict(username=self.username,team=team,id=id))

# Reset older due values to 0
(cursor or self.db).run("""
UPDATE payment_instructions p
SET due = 0
WHERE participant = %(username)s
AND team = %(team)s
AND due > 0
AND p.id != %(id)s
""", dict(username=self.username,team=team,id=id))

def update_receiving(self, cursor=None):
if self.IS_PLURAL:
old_takes = self.compute_actual_takes(cursor=cursor)
r = (cursor or self.db).one("""
WITH our_tips AS (
SELECT amount
FROM current_tips
JOIN participants p2 ON p2.username = tipper
WHERE tippee = %(username)s
AND p2.is_suspicious IS NOT true
AND amount > 0
AND is_funded
)
UPDATE participants p
SET receiving = (COALESCE((
SELECT sum(amount)
FROM our_tips
), 0) + taking)
, npatrons = COALESCE((SELECT count(*) FROM our_tips), 0)
WHERE p.username = %(username)s
RETURNING receiving, npatrons
""", dict(username=self.username))
self.set_attributes(receiving=r.receiving, npatrons=r.npatrons)
if self.IS_PLURAL:
new_takes = self.compute_actual_takes(cursor=cursor)
self.update_taking(old_takes, new_takes, cursor=cursor)

def update_is_free_rider(self, is_free_rider, cursor=None):
with self.db.get_cursor(cursor) as cursor:
cursor.run( "UPDATE participants SET is_free_rider=%(is_free_rider)s "
"WHERE username=%(username)s"
, dict(username=self.username, is_free_rider=is_free_rider)
)
add_event( cursor
, 'participant'
, dict(id=self.id, action='set', values=dict(is_free_rider=is_free_rider))
)
self.set_attributes(is_free_rider=is_free_rider)


# New payday system

def set_payment_instruction(self, team, amount, update_self=True, update_team=True,
cursor=None):
Expand Down
1 change: 0 additions & 1 deletion tests/py/test_billing_payday.py
Original file line number Diff line number Diff line change
Expand Up @@ -92,7 +92,6 @@ def test_payday_preserves_due_until_charged(self, fch):
fch.return_value = {}
Payday.start().run() # payday 4


obama = Participant.from_username('obama')
picard = Participant.from_username('picard')

Expand Down

0 comments on commit feeec33

Please sign in to comment.