Facebook广告投放数据API对接流程

news2025/1/12 20:58:49

说明:仅供学习使用,请勿用于非法用途,若有侵权,请联系博主删除

作者:zhu6201976

一、需求背景

App在Facebook、Google等巨头进行广告投放,想要拿到实时广告投放效果数据,如曝光、点击、花费、触达等核心指标并进行分析,可对接官方API实现。因对接过程十分复杂,此处以Facebook为例,进行简单记录。

二、对接流程

1.文档地址

https://developers.facebook.com/docs/marketing-api/insights

2. access_token获取

a. 创建App https://developers.facebook.com/apps

b. 添加产品

  • 点击进入某个App --> Add products to your app --> Marketing API --> Set up

  • 点击左下角Marketing API --> Tools --> 勾选token权限 --> Get Token 即可生成一个长效token,有效期2个月,过期后需要重新生成 步骤相同

c. 查看access_token有效期(可选)

复制保存步骤2生成的token,查看有效期。

地址:https://developers.facebook.com/tools/debug/accesstoken

如图所示,通过此种方式,可以获取到一个有效期2个月的长效access_token。

三、示例代码

获取到长效access_token后,我们可以开始请求广告数据了。

    def __init__(self, **kwargs):
        super().__init__(**kwargs)
        self.api_url = 'https://graph.facebook.com'
        self.api_version = 'v19.0'

1. 获取所有广告账号adaccount

    def start_requests(self):
        page = 1
        url = f'{self.api_url}/{self.api_version}/me/adaccounts?'
        params = {
            'fields': 'id,name',
            'access_token': self.access_token
        }
        url = f'{url}{urlencode(params)}'
        yield scrapy.Request(url, meta={'page': page}, priority=1, callback=self.parse_adaccounts)

通过该API,可以获取到所有广告账号,分页返回,每页25条。fields字段控制返回的字段,此处为account_id、account_name。

后续翻页控制代码:

resp_str = resp.body.decode('utf-8', 'ignore')
resp_dict = json.loads(resp_str)
next_url = resp_dict.get('paging', {}).get('next')
if next_url:
    page += 1
    yield scrapy.Request(next_url, meta={'page': page}, priority=1, callback=self.parse_adaccounts)

2. 获取广告账号广告量

为什么要获取广告账号广告量?因为如果某个广告账号下根本没有投放广告,则无需后续请求,提高效率,减少请求次数。

    def get_adaccount_ads_volume_req(self, ad_account_id, ad_account_name):
        """
        构造账号广告量查询请求
        https://developers.facebook.com/docs/marketing-api/insights-api/ads-volume
        :param ad_account_id:
        :return:
        """
        url = f'{self.api_url}/{self.api_version}/{ad_account_id}/ads_volume?'
        params = {
            'access_token': self.access_token
        }
        url = f'{url}{urlencode(params)}'
        return scrapy.Request(url,
                              meta={'ad_account_id': ad_account_id,
                                    'ad_account_name': ad_account_name},
                              priority=2,
                              callback=self.parse_adaccount_ads_volume)

解析账号广告量响应代码:

        resp_str = resp.body.decode('utf-8', 'ignore')
        resp_dict = json.loads(resp_str)
        # self.logger.info(f'{method_name} {ad_account_id} {resp_dict}')
        ads_volume = resp_dict.get('data', [])
        self.logger.info(f'{method_name} {ad_account_id} 提取到 {len(ads_volume)} ads_volume')
        for volume in ads_volume:
            ads_running_or_in_review_count = volume.get('ads_running_or_in_review_count', 0)
            if ads_running_or_in_review_count > 0:
                yield self.get_campaigns_req(ad_account_id, ad_account_name, page=1)

3. 获取广告账号所有campaign

    def get_campaigns_req(self, ad_account_id, ad_account_name, page):
        """
        构造启动campaigns请求
        :param ad_account_id:
        :param ad_account_name:
        :param page:
        :return:
        """
        url = f'{self.api_url}/{self.api_version}/{ad_account_id}/campaigns?'
        params = {
            'effective_status': "['ACTIVE', 'PAUSED']",
            'fields': 'id,name',
            'access_token': self.access_token,
        }
        url = f'{url}{urlencode(params)}'
        return scrapy.Request(url,
                              meta={'ad_account_id': ad_account_id,
                                    'ad_account_name': ad_account_name,
                                    'page': page},
                              priority=2,
                              callback=self.parse_get_campaigns)

4. 对campaign聚合获取所有广告

该步骤非常重要,我们无需通过完整路径:adaccount --> campaign --> adset --> ads 获取到广告数据,可以利用API的level进行聚合,直接在campaign层级获取到所有广告投放数据。如下所示:

    def ad_insights_req(self, ad_account_id, ad_account_name, ad_campaign_id, ad_campaign_name, d_str, page):
        url = f'{self.api_url}/{self.api_version}/{ad_campaign_id}/insights?'
        params = {
            'level': 'ad',
            'fields': 'adset_id,adset_name,ad_id,ad_name,impressions,clicks,unique_clicks,spend,reach,cpc,cpm,cpp,ctr,' +
                      'date_start,date_stop,actions,action_values',
            # 'date_preset': 'today',  # today,yesterday,last_7d,last_30d,...
            'time_range': '{"since":' + f'"{d_str}"' + "," + '"until":' + f'"{d_str}"' + "}",
            'access_token': self.access_token,
        }
        url = f'{url}{urlencode(params)}'
        return scrapy.Request(url,
                              meta={'ad_account_id': ad_account_id,
                                    'ad_account_name': ad_account_name,
                                    'ad_campaign_id': ad_campaign_id,
                                    'ad_campaign_name': ad_campaign_name,
                                    'd_str': d_str,
                                    'page': page},
                              priority=3,
                              callback=self.parse_ad_insights)

 5. 解析广告数据

        for ad in ads:
            ad_insights_item = FbAdInsightsItem()
            ad_insights_item['ad_account_id'] = ad_account_id
            ad_insights_item['ad_account_name'] = ad_account_name
            ad_insights_item['ad_campaign_id'] = ad_campaign_id
            ad_insights_item['ad_campaign_name'] = ad_campaign_name
            ad_insights_item['ad_set_id'] = ad.get('adset_id', '')
            ad_insights_item['ad_set_name'] = ad.get('adset_name', '')
            ad_insights_item['ad_id'] = ad.get('ad_id', '')
            ad_insights_item['ad_name'] = ad.get('ad_name', '')
            ad_insights_item['impressions'] = ad.get('impressions', '')
            ad_insights_item['clicks'] = ad.get('clicks', '')
            ad_insights_item['unique_clicks'] = ad.get('unique_clicks', '')
            ad_insights_item['spend'] = ad.get('spend', '')
            ad_insights_item['reach'] = ad.get('reach', '')
            ad_insights_item['cpc'] = ad.get('cpc', '')
            ad_insights_item['cpm'] = ad.get('cpm', '')
            ad_insights_item['cpp'] = ad.get('cpp', '')
            ad_insights_item['ctr'] = ad.get('ctr', '')
            ad_insights_item['date_start'] = ad.get('date_start', '')
            ad_insights_item['date_stop'] = ad.get('date_stop', '')

一键三连,有需要的请私聊获取详细源码。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1579833.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

最少按键次数

题目描述 给你一个字符串 s,由小写英文字母组成。 电话键盘上的按键与 不同 小写英文字母集合相映射,可以通过按压按键来组成单词。例如,按键 2 对应 ["a","b","c"],我们需要按一次键来输入 &quo…

C语言从入门到实战————文件操作

目录 前言 1. 为什么使用文件? 2. 什么是文件? 2.1 程序文件 2.2 数据文件 2.3 文件名 3. ⼆进制文件和文本文件? 4. 文件的打开和关闭 4.1 流和标准流 4.1.1 流 4.1.2 标准流 4.2 文件指针 4.3 文件的打开和关闭 5. 文…

公寓水电表预付费控制系统

随着现代化管理理念的不断发展,许多公寓管理者开始寻求更加高效、透明的水电计费方式。在此趋势下,公寓水电表预付费控制系统应运而生,成为了一种创新的能源管理解决方案。它通过预付费方式,帮助住户和管理者实现了更加精确和便捷…

Jmeter针对多种响应断言的判断

有时候response返回的结果并非一种,有多种,需要对这几种进行判断的时候需要使用Bean Shell。 (1)首先获取响应数据 String response prev.getResponseDataAsString(); ResponseCode 响应状态码 responseHeaders 响应头信息 res…

Cyber Weekly #1

赛博新闻 1、弱智吧竟成最佳中文AI训练数据?!中科院等:8项测试第一,远超知乎豆瓣小红书 使用弱智吧数据训练的大模型,跑分超过百科、知乎、豆瓣、小红书等平台,甚至是研究团队精心挑选的数据集。弱智吧数…

第十三章 OpenGL ES-RGB、HSV、HSL模型介绍

第十三章 OpenGL ES-RGB、HSV、HSL模型详细介绍 第一章 OpenGL ES 基础-屏幕、纹理、顶点坐标 第二章 OpenGL ES 基础-GLSL语法简单总结 第三章 OpenGL ES 基础-GLSL渲染纹理 第四章 OpenGL ES 基础-位移、缩放、旋转原理 第五章 OpenGL ES 基础-透视投影矩阵与正交投影矩阵…

vue+springboot实现JWT登录验证

目录 前言概念实际演示路由信息初始访问登录界面登录验证验证过期 vue实现依赖引入main.js获取和设置token工具类登录方法实体登录方法axios请求 router配置 springboot实现依赖引入JWT工具类忽视jwt验证注解拦截器逻辑跨域&调用拦截器配置登录接口&验证token接口 结语…

算法打卡day36|动态规划篇04| 01背包理论基础、416. 分割等和子集

目录 01背包理论基础 01背包问题描述 01背包解法 二维数组 一维数组 算法题 Leetcode 416. 分割等和子集 个人思路 解法 动态规划 01背包理论基础 不同的背包种类,虽然有那么多中南背包,但其中01背包和完全背包是重中之重; 01背包问…

flutter多入口点entrypoint

native中引擎对象本身消耗内存(每个引擎对象约莫消耗42MB内存) 多引擎:native多引擎>启动>flutter多入口点entrypoint>多main函数>多子包元素集>多(子)程序 单引擎(复用):native单引擎>复用启动>flutter多入口点entrypoint>多m…

六、企业级架构缓存篇之memcached

一、memcached概述 1、网站架构优化流程: LNMP架构中网站应用访问流程: 浏览器 (app) → web 服务器 → 后端服务 (php) → 数据库 (mysql) 访问流程越多,访问速度越慢,出现问题的几率也越大。 网站访问流程优化思路&#xff1…

HarmonyOS 开发-PixelMap深拷贝案例

介绍 在图片开发过程中经常会涉及到PixelMap的深拷贝,本例通过使用PixelMap的readPixelsToBuffer方法来实现深拷贝。在创建源PixelMap的时候,需要将解码参数设置为BGRA_8888,而在深拷贝创建目标PixelMap的时候需要将解码参数设置为RGBA_8888…

首发!澳门地区OSGB数据V0.2版免费分享

之前分享了澳门OSGB V0.1版数据(独家,澳门地区OSGB数据免费分享!),得到了众多读者的认同,也有读者在数据使用过程中发现的问题,向我进行了反馈与交流,基于此,我在V0.1版本的基础上进行了修正与更…

vue3+Ts+Ant Design Vue +天地图组件封装

​&#x1f308;个人主页&#xff1a;前端青山 &#x1f516;人终将被年少不可得之物困其一生 依旧青山,本期给大家带来Vue3TsAntDesign-Vue组件天地图组件的封装 示例图 首先,在index.html引入天地图资源,vue3选择v4版本 <script src"http://api.tianditu.gov.cn/api?…

JVM 内存溢出排查

说明&#xff1a;记录一次JVM内存溢出的排查过程&#xff1b; 场景 项目开发完成后&#xff0c;首次提交到测试环境。测试、产品同事反馈页面先是操作响应慢&#xff0c;抛出超时异常&#xff0c;最后直接无法使用。查看日志后得知是内存溢出。 重启服务后&#xff0c;我对前…

2024,还没掌握 JS 发布者、订阅者模式吗

面试中经常出现问到如何实现JS 发布者、订阅者模式。下面是ES5实现发布订阅模式。 1、直接上代码。 function EventEmitter() {this.events {}; }; // 订阅者 EventEmitter.prototype.on function(ename, callback) {if (!this.events[ename]) {// 初始化创建订阅&#xff…

算法训练营第二十一天(二叉树part7)

算法训练营第二十一天&#xff08;二叉树part7&#xff09; 530.二叉搜索树的最小绝对差 力扣题目链接(opens new window) 题目 给你一棵所有节点为非负值的二叉搜索树&#xff0c;请你计算树中任意两节点的差的绝对值的最小值。 示例&#xff1a; 提示&#xff1a;树中至…

9.枚举类与注解

文章目录 1. 枚举类1.1 自定义枚举类1.2 enum关键字定义枚举类1.3 Enum类主要方法 2. 注解2.1 注解作用2.1.1 生成文档相关注解2.1.2 JDK内置基本注解-在编译时进行格式检查2.1.3 跟踪代码依赖性&#xff0c;实现替代配置文件功能 2.2 自定义Annotation2.3 JDK中的元注解2.3.1 …

Python零基础从小白打怪升级中~~~~~~~流程控制语句

第三节&#xff1a;Python的流程控制语法 一、Python条件语句的语法 if 条件1:条件1成立执⾏的代码一条件1成⽴执⾏的代码二...... elif 条件2&#xff1a;条件2成立执⾏的代码三条件2成立执⾏的代码四...... ...... else:以上条件都不成⽴&#xff0c;执行的代码五以上条件都…

海外盲盒系统开发,盲盒出口成为企业新机遇

随着盲盒的兴起&#xff0c;国内消费市场形成了万物皆可盲盒的态势。并且&#xff0c;盲盒自带社交属性&#xff0c;也成为了年轻人的社交神器。 除了在国内&#xff0c;盲盒在海外也掀起了一股热潮&#xff0c;呈现出了爆发式增长形势&#xff0c;国内热门盲盒企业也出口到了…

Redis的三种部署方案

文章目录 单机模式主从复制哨兵模式分片集群 在Redis中提供的集群方案总共有三种&#xff1a;单机模式&#xff0c;主从复制集群、哨兵模式&#xff0c;Redis分片集群 单机模式 Redis 只运行在一台服务器上&#xff0c;并且所有的数据都存储在这一台服务器的内存中。 主从复制…