Skip to content

Instantly share code, notes, and snippets.

@mbarnes
Last active June 15, 2024 15:35
Show Gist options
  • Save mbarnes/a9559ae33f6d1f073b4ec16b45ed3789 to your computer and use it in GitHub Desktop.
Save mbarnes/a9559ae33f6d1f073b4ec16b45ed3789 to your computer and use it in GitHub Desktop.
BJs Wholesale Club Digital Coupon Clipper
#!/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()
@mbarnes
Copy link
Author

mbarnes commented Sep 27, 2023

No, they've started using bot detection.
Will probably require a full rewrite using a web driver framework like Selenium to execute JavaScript.

@mbarnes
Copy link
Author

mbarnes commented Feb 22, 2024

This started working again for me with no changes.

@LightningATCHub
Copy link

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