Last active
June 15, 2024 15:35
-
-
Save mbarnes/a9559ae33f6d1f073b4ec16b45ed3789 to your computer and use it in GitHub Desktop.
BJs Wholesale Club Digital Coupon Clipper
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#!/usr/bin/python3 | |
# | |
# Clip all available digital coupons for a BJ's membership. | |
# | |
# To avoid interactive prompts, either set environment variables | |
# BJS_MEMBER_EMAIL and BJS_MEMBER_PASSWORD or add credentials to | |
# to your ~/.netrc file: | |
# | |
# machine bjs.com | |
# login <BJS_MEMBER_EMAIL> | |
# password <BJS_MEMBER_PASSWORD> | |
# | |
import base64 | |
import getpass | |
import http | |
import netrc | |
import os | |
import urllib | |
# 3rd-party modules | |
import requests | |
# Show HTTP requests and responses | |
http.client.HTTPConnection.debuglevel = 0 | |
# Minimum WebGL fingerprint | |
DEVICE_PROFILE=b''' | |
{ | |
"components": [ | |
{ | |
"key": "webgl", | |
"value": [ | |
"", | |
"webgl aliased line width range:[1, 7.375]", | |
"webgl aliased point size range:[1, 255]", | |
"webgl alpha bits:8", | |
"webgl antialiasing:yes", | |
"webgl blue bits:8", | |
"webgl depth bits:24", | |
"webgl green bits:8", | |
"webgl max anisotropy:16", | |
"webgl max combined texture image units:64", | |
"webgl max cube map texture size:16384", | |
"webgl max fragment uniform vectors:1024", | |
"webgl max render buffer size:16384", | |
"webgl max texture image units:32", | |
"webgl max texture size:16384", | |
"webgl max varying vectors:32", | |
"webgl max vertex attribs:16", | |
"webgl max vertex texture image units:32", | |
"webgl max vertex uniform vectors:1024", | |
"webgl max viewport dims:[16384, 16384]", | |
"webgl red bits:8", | |
"webgl shading language version:WebGL GLSL ES 1.0 (OpenGL ES GLSL ES 1.0 Chromium)", | |
"webgl vertex shader high float precision:23", | |
"webgl vertex shader high float precision rangeMin:127", | |
"webgl vertex shader high float precision rangeMax:127", | |
"webgl vertex shader medium float precision:23", | |
"webgl vertex shader medium float precision rangeMin:127", | |
"webgl vertex shader medium float precision rangeMax:127", | |
"webgl vertex shader low float precision:23", | |
"webgl vertex shader low float precision rangeMin:127", | |
"webgl vertex shader low float precision rangeMax:127", | |
] | |
}, | |
] | |
} | |
''' | |
USER_AGENT = 'Mozilla/5.0 (X11; Linux x86_64)' | |
def bjs_session_login(method): | |
def inner(session, *args, **kwargs): | |
"""Log in to BJs.com on first call""" | |
if not session.account: | |
pattern = "/digital/live/api/v1.4/storeId/{}/login" | |
url = pattern.format(session.store_id) | |
body = { | |
'deviceProfile': base64.b64encode(DEVICE_PROFILE), | |
'logonId': session.email, | |
'logonPassword': session.password | |
} | |
response = session.post(url, json=body) | |
response.raise_for_status() | |
pattern = "/digital/live/api/v1.0/storeId/{}/accountdetails" | |
url = pattern.format(session.store_id) | |
response = session.get(url) | |
response.raise_for_status() | |
session.account = response.json() | |
return method(session, *args, **kwargs) | |
return inner | |
class BJsSession(requests.Session): | |
"""BJs.com REST API session""" | |
base_url = "https://api.bjs.com" | |
# XXX This code appears hardcoded in the site's JavaScript, | |
# defined in the file "main-es2015.js". Unclear if the | |
# code is the same for everyone. If the site uses some | |
# 3rd party e-commerce software then the code may be an | |
# identifier for BJ's Wholesale Club, Inc. | |
store_id = 10201 | |
def __init__(self, base_url=None): | |
if base_url: | |
self.base_url = base_url | |
super().__init__() | |
self.headers['Referer'] = 'https://www.bjs.com/' | |
self.headers['User-Agent'] = USER_AGENT | |
self.__get_credentials() | |
self.account = None | |
def __get_credentials(self): | |
self.email = os.environ.get('BJS_MEMBER_EMAIL') | |
self.password = os.environ.get('BJS_MEMBER_PASSWORD') | |
if not (self.email and self.password): | |
try: | |
if auth := netrc.netrc().authenticators('bjs.com'): | |
self.email, _, self.password = auth | |
except FileNotFoundError: | |
pass | |
if not (self.email and self.password): | |
print('BJs.com Member Sign In') | |
self.email = input('Email Address: ').strip() | |
self.password = getpass.getpass('Password: ').strip() | |
def request(self, method, url, *args, **kwargs): | |
"""Send the request after generating the complete URL""" | |
url = self.create_url(url) | |
return super().request(method, url, *args, **kwargs) | |
def create_url(self, url): | |
"""Create the URL based off this partial path""" | |
return urllib.parse.urljoin(self.base_url, url) | |
@bjs_session_login | |
def activate_offer(self, offer): | |
"""Activate a discount offer""" | |
url = "/digital/live/api/v1.0/store/{}/coupons/activate" | |
params = { | |
'offerId': offer['offerId'], | |
'zip': self.account['zipCode'][:5] | |
} | |
response = self.get(url.format(self.store_id), params=params) | |
response.raise_for_status() | |
data = response.json() | |
try: | |
return data['success'] | |
except KeyError: | |
print('JSON:', data) | |
# If JSON response is empty, | |
# assume success and proceed. | |
if not data: | |
return True | |
raise | |
@bjs_session_login | |
def available_offers(self): | |
"""List available discount offers""" | |
url = "/digital/live/api/v1.0/member/available/offers" | |
offer_count = 0 | |
total_available = 1 | |
while offer_count < total_available: | |
body = { | |
'brand': '', | |
'category': '', | |
'indexForPagination': max(0, offer_count - 1), | |
'isNext': True, | |
'isPrev': False, | |
'membershipNumber': self.account['MembershipNumber'], | |
'pagesize': 100, | |
'searchString': '', | |
'zipcode': self.account['zipCode'][:5] | |
} | |
response = self.post(url, json=body) | |
response.raise_for_status() | |
data = response.json().pop() | |
offer_count += len(data['availableOffers']) | |
total_available = data['totalAvailable'] | |
yield from data['availableOffers'] | |
def main(): | |
with BJsSession() as session: | |
clipped = [] | |
for offer in session.available_offers(): | |
if session.activate_offer(offer): | |
print('CLIPPED:', offer['offerSummary'], offer['offerDescription']) | |
clipped.append(offer) | |
print(len(clipped), "coupons clipped") | |
if __name__ == '__main__': | |
main() |
This started working again for me with no changes.
Don't think it's working anymore.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
No, they've started using bot detection.
Will probably require a full rewrite using a web driver framework like Selenium to execute JavaScript.