Skip to content

Commit 20275f0

Browse files
author
Stanley Shi
committed
Initial commit of the scripts
0 parents  commit 20275f0

8 files changed

+608
-0
lines changed

CODE_OF_CONDUCT.md

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
# Code of Conduct
2+
3+
## Our Pledge
4+
5+
In the interest of fostering an open and welcoming environment, we as
6+
contributors and maintainers pledge to make participation in our project and
7+
our community a harassment-free experience for everyone, regardless of age, body
8+
size, disability, ethnicity, sex characteristics, gender identity and expression,
9+
level of experience, education, socio-economic status, nationality, personal
10+
appearance, race, religion, or sexual identity and orientation.
11+
12+
## Our Standards
13+
14+
Examples of behavior that contributes to creating a positive environment
15+
include:
16+
17+
* Using welcoming and inclusive language
18+
* Being respectful of differing viewpoints and experiences
19+
* Gracefully accepting constructive criticism
20+
* Focusing on what is best for the community
21+
* Showing empathy towards other community members
22+
23+
Examples of unacceptable behavior by participants include:
24+
25+
* The use of sexualized language or imagery and unwelcome sexual attention or
26+
advances
27+
* Trolling, insulting/derogatory comments, and personal or political attacks
28+
* Public or private harassment
29+
* Publishing others' private information, such as a physical or electronic
30+
address, without explicit permission
31+
* Other conduct which could reasonably be considered inappropriate in a
32+
professional setting
33+
34+
## Our Responsibilities
35+
36+
Project maintainers are responsible for clarifying the standards of acceptable
37+
behavior and are expected to take appropriate and fair corrective action in
38+
response to any instances of unacceptable behavior.
39+
40+
Project maintainers have the right and responsibility to remove, edit, or
41+
reject comments, commits, code, wiki edits, issues, and other contributions
42+
that are not aligned to this Code of Conduct, or to ban temporarily or
43+
permanently any contributor for other behaviors that they deem inappropriate,
44+
threatening, offensive, or harmful.
45+
46+
## Scope
47+
48+
This Code of Conduct applies within all project spaces, and it also applies when
49+
an individual is representing the project or its community in public spaces.
50+
Examples of representing a project or community include using an official
51+
project e-mail address, posting via an official social media account, or acting
52+
as an appointed representative at an online or offline event. Representation of
53+
a project may be further defined and clarified by project maintainers.
54+
55+
## Enforcement
56+
57+
Instances of abusive, harassing, or otherwise unacceptable behavior may be
58+
reported by contacting the project team at <[email protected]>. All
59+
complaints will be reviewed and investigated and will result in a response that
60+
is deemed necessary and appropriate to the circumstances. The project team is
61+
obligated to maintain confidentiality with regard to the reporter of an incident.
62+
Further details of specific enforcement policies may be posted separately.
63+
64+
Project maintainers who do not follow or enforce the Code of Conduct in good
65+
faith may face temporary or permanent repercussions as determined by other
66+
members of the project's leadership.
67+
68+
## Attribution
69+
70+
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 1.4,
71+
available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html
72+
73+
[homepage]: https://www.contributor-covenant.org
74+
75+
For answers to common questions about this code of conduct, see
76+
https://www.contributor-covenant.org/faq

CONTRIBUTING.md

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
# Contributing to Ads-Library-API-Script-Repository
2+
We want to make contributing to this project as easy and transparent as
3+
possible.
4+
5+
## Pull Requests
6+
We actively welcome your pull requests.
7+
8+
1. Fork the repo and create your branch from `master`.
9+
2. If you've added code that should be tested, add tests.
10+
3. If you've changed APIs, update the documentation.
11+
4. Ensure the test suite passes.
12+
5. Make sure your code lints.
13+
6. If you haven't already, complete the Contributor License Agreement ("CLA").
14+
15+
## Contributor License Agreement ("CLA")
16+
In order to accept your pull request, we need you to submit a CLA. You only need
17+
to do this once to work on any of Facebook's open source projects.
18+
19+
Complete your CLA here: <https://code.facebook.com/cla>
20+
21+
## Issues
22+
We use GitHub issues to track public bugs. Please ensure your description is
23+
clear and has sufficient instructions to be able to reproduce the issue.
24+
25+
Facebook has a [bounty program](https://www.facebook.com/whitehat/) for the safe
26+
disclosure of security bugs. In those cases, please go through the process
27+
outlined on that page and do not file a public issue.
28+
29+
## Coding Style
30+
* 4 spaces for indentation rather than tabs
31+
* 80 character line length
32+
* PEP8 formatting following [Black](https://black.readthedocs.io/en/stable/)
33+
34+
## License
35+
By contributing to Ads-Library-API-Script-Repository, you agree that your
36+
contributions will be licensed under the LICENSE file in the root directory
37+
of this source tree.

LICENSE

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
Copyright (c) Facebook, Inc. and its affiliates. All rights reserved.
2+
3+
You are hereby granted a non-exclusive, worldwide, royalty-free license to use,
4+
copy, modify, and distribute this software in source code or binary form for use
5+
in connection with the web services and APIs provided by Facebook.
6+
7+
As with any software that integrates with the Facebook platform, your use of
8+
this software is subject to the Facebook Platform Policy
9+
[http://developers.facebook.com/policy/]. This copyright notice shall be
10+
included in all copies or substantial portions of the software.
11+
12+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
13+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
14+
FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
15+
COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
16+
IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
17+
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

README.md

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
# Ads-Library-API-Script-Repository
2+
Ads-Library-API-Script-Repository is a set of code examples to help user/researchers understand how the Facebook Ads Library API works. It also provides a simple command-line interface(CLI) for users to easily use the Facebook Ads Library API.
3+
4+
## Examples
5+
Here's an example on how to use the CLI:
6+
7+
$ python fb_ads_library_api_cli.py -t {access_token} -f 'page_id,ad_snapshot_url,funding_entity,ad_delivery_start_time' -c 'CA' -s '.' -v count
8+
9+
It count the number of active polictical ads that are running in CA(Canada);
10+
11+
Note: please replace the '{access_token}' with your [Facebook Developer access token](https://developers.facebook.com/tools/accesstoken/).
12+
13+
## Requirements
14+
Ads-Library-API-Script-Repository requires or works with
15+
* Mac OS X or Linux or Window
16+
* Python 3.0+
17+
18+
19+
## How Ads-Library-API-Script-Repository works
20+
The script will query the [Facebook Ads library API](https://www.facebook.com/ads/library/api) to get all the Ads Library information on the Facebook platform;
21+
22+
## Full documentation
23+
You can find the full documentation here: (--to-be-added--)
24+
25+
## More about Facebook Ads Library
26+
* Website: https://www.facebook.com/ads/library
27+
* Report: https://www.facebook.com/ads/library/report
28+
* API: https://www.facebook.com/ads/library/api
29+
30+
See the [CONTRIBUTING](CONTRIBUTING.md) file for how to help out.
31+
32+
## License
33+
Ads-Library-API-Script-Repository is licensed under the Facebook Platform License, as found in the LICENSE file.

python/fb_ads_library_api.py

+126
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,126 @@
1+
#!/usr/bin/env python3
2+
3+
# Copyright (c) Facebook, Inc. and its affiliates.
4+
# All rights reserved.
5+
#
6+
# This source code is licensed under the license found in the
7+
# LICENSE file in the root directory of this source tree.
8+
9+
from datetime import datetime
10+
import json
11+
import re
12+
import requests
13+
14+
15+
def get_ad_archive_id(data):
16+
"""
17+
Extract ad_archive_id from ad_snapshot_url
18+
"""
19+
return re.search(r"/\?id=([0-9]+)", data["ad_snapshot_url"]).group(1)
20+
21+
22+
class FbAdsLibraryTraversal:
23+
default_url_pattern = (
24+
"https://graph.facebook.com/{}/ads_archive?access_token={}&"
25+
+ "fields={}&search_terms={}&ad_reached_countries={}&search_page_ids={}&"
26+
+ "ad_active_status={}&limit={}"
27+
)
28+
default_api_version = "v4.0"
29+
30+
def __init__(
31+
self,
32+
access_token,
33+
fields,
34+
search_term,
35+
country,
36+
search_page_ids="",
37+
ad_active_status="active",
38+
after_date="1970-01-01",
39+
page_limit=500,
40+
api_version=None,
41+
retry_limit=3,
42+
):
43+
self.page_count = 0
44+
self.access_token = access_token
45+
self.fields = fields
46+
self.search_term = search_term
47+
self.country = country
48+
self.after_date = after_date
49+
self.search_page_ids = search_page_ids
50+
self.ad_active_status = ad_active_status
51+
self.page_limit = page_limit
52+
self.retry_limit = retry_limit
53+
if api_version is None:
54+
self.api_version = self.default_api_version
55+
else:
56+
self.api_version = api_version
57+
58+
def generate_ad_archives(self):
59+
next_page_url = self.default_url_pattern.format(
60+
self.api_version,
61+
self.access_token,
62+
self.fields,
63+
self.search_term,
64+
self.country,
65+
self.search_page_ids,
66+
self.ad_active_status,
67+
self.page_limit,
68+
)
69+
return self.__class__._get_ad_archives_from_url(
70+
next_page_url, after_date=self.after_date, retry_limit=self.retry_limit
71+
)
72+
73+
@staticmethod
74+
def _get_ad_archives_from_url(
75+
next_page_url, after_date="1970-01-01", retry_limit=3
76+
):
77+
last_error_url = None
78+
last_retry_count = 0
79+
start_time_cutoff_after = datetime.strptime(
80+
after_date, "%Y-%m-%d"
81+
).timestamp()
82+
83+
while next_page_url is not None:
84+
response = requests.get(next_page_url)
85+
response_data = json.loads(response.text)
86+
if "error" in response_data:
87+
if next_page_url == last_error_url:
88+
# failed again
89+
if last_retry_count >= retry_limit:
90+
raise Exception(
91+
"Error message: [{}], failed on URL: [{}]".format(
92+
json.dumps(response_data["error"]), next_page_url
93+
)
94+
)
95+
else:
96+
last_error_url = next_page_url
97+
last_retry_count = 0
98+
last_retry_count += 1
99+
continue
100+
101+
filtered = list(
102+
filter(
103+
lambda ad_archive: datetime.strptime(
104+
ad_archive["ad_delivery_start_time"], "%Y-%m-%dT%H:%M:%S%z"
105+
).timestamp()
106+
>= start_time_cutoff_after,
107+
response_data["data"],
108+
)
109+
)
110+
if len(filtered) == 0:
111+
# if no data after the after_date, break
112+
next_page_url = None
113+
break
114+
yield filtered
115+
116+
if "paging" in response_data:
117+
next_page_url = response_data["paging"]["next"]
118+
else:
119+
next_page_url = None
120+
121+
@classmethod
122+
def generate_ad_archives_from_url(cls, failure_url, after_date="1970-01-01"):
123+
"""
124+
if we failed from error, later we can just continue from the last failure url
125+
"""
126+
return cls._get_ad_archives_from_url(failure_url, after_date=after_date)

0 commit comments

Comments
 (0)