Новый скрипт от Brainlabs поможет отследить эффективность разных стратегий биддинга в Google Ads

Новый скрипт от Brainlabs поможет отследить эффективность разных стратегий биддинга в Google Ads

Глава Brainlabs Дэниел Гилберт (Daniel Gilbert) поделился скриптом, который позволяет отслеживать и сравнивать эффективность ряда стратегий назначения ставок в Google Ads.

Это решение призвано помочь рекламодателям понять, какие стратегии для них наиболее эффективны, и дать им возможность принимать более взвешенные решения.

Возможности скрипта

Скрипт может анализировать ряд автоматических и ручных стратегий
назначения ставок, включая Manual CPC,
Manual
CPV, Manual CPM, Target search page location, Maximize clicks, Maximize Conversion Value, Target Outranking Share, Target CPA, Target ROAS и
Maximize Conversions.

После того, как рекламодатель обозначит, какие кампании и стратегии назначения ставок нужно анализировать, скрипт предоставит отчёт по следующим метрикам: CPC, клики, конверсии, стоимость, CTR, показы и доля показов вверху страницы. При желании можно будет добавить другие показатели. Как это сделать, описано в руководстве Google.

Как использовать
скрипт

Сначала создайте копию следующей таблицы.

Затем введите ID аккаунта, кампании, которые нужно отслеживать, стратегии
назначения ставок, которые нужно анализировать, и метрики, которые должны быть
в отчёте.

После этого скопируйте приведённый ниже скрипт в новый
скрипт в Google Ads
(Инструменты > Массовые изменения > Скрипты).

Перед запуском скрипта отредактируйте переменную SPREADSHEET_URL,
добавив URL-адрес своей
таблицы.

Вы также можете указать, как скрипт должен обходиться с приостановленными кампаниями — игнорировать или учитывать их. Для этого нужно установить значение переменной ignorePausedCampaigns как «true» или «false» соответственно.

/**
*
* Bid Strategy Performance Monitor
*
* This script allows Google Ads MCC Accounts to monitor the performance
* of various bidding strategies on child accounts based on defined
* metrics.
*
* Version: 1.0
* Google Ads Script maintained on brainlabsdigital.com
*
**/

//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////

//Options

//Spreadsheet URL

var SPREADSHEET_URL = ‘https://docs.google.com/YOUR-SPREADSHEET-URL-HERE’;

//Ignore Paused Campaigns

// Set to ‘false’ to include paused campaigns in data.

var ignorePausedCampaigns = true;

//////////////////////////////////////////////////////////////////////////////
//////////////////////////////////////////////////////////////////////////////

// Metrics

// Metrics are written onto output sheet in order stated below. Read the ‘Metric’
// column of the Google Ads documentation to find other metrics to include:
// https://developers.google.com/adwords/api/docs/appendix/reports/campaign-performance-report

var METRICS = [
  ‘AverageCpc’,
  ‘Clicks’,
  ‘Conversions’,
  ‘Cost’,
  ‘Ctr’,
  ‘Impressions’,
  ‘TopImpressionPercentage’
];

//Bidding Strategies

var BIDDING_STRATEGIES =
{
  ‘Manual CPC’: ‘MANUAL_CPC’,
  ‘Manual CPV’: ‘MANUAL_CPV’,
  ‘Manual CPM’: ‘MANUAL_CPM’,
  ‘Target search page location’: ‘PAGE_ONE_PROMOTED’,
  ‘Maximize clicks’: ‘TARGET_SPEND’,
  ‘Target CPA’: ‘TARGET_CPA’,
  ‘Target ROAS’: ‘TARGET_ROAS’,
  ‘Maximize Conversions’: ‘MAXIMIZE_CONVERSIONS’,
  ‘Maximize Conversion Value’: ‘MAXIMIZE_CONVERSION_VALUE’,
  ‘Target Outranking Share’: ‘TARGET_OUTRANK_SHARE’
};

// Indices

var INPUT_HEADER_ROW = 1;
var INPUT_DATA_ROW = 3;
var OUTPUT_HEADER_ROW = 2;

//////////////////////////////////////////////////////////////////////////////

// Functions

function main() {

  var spreadsheet = getSpreadsheet(SPREADSHEET_URL);
  var inputSheet = spreadsheet.getSheetByName(«Input»);
  var outputSheet = spreadsheet.getSheetByName(«Output»);

  var tz = AdsApp.currentAccount().getTimeZone();

  //Store Sheet Headers and Indices

  var inputHeaders = inputSheet.getRange(INPUT_HEADER_ROW + «:» + INPUT_HEADER_ROW).getValues()[0];
  var statusColumnIndex = inputHeaders.indexOf(«Status»);
  var accountIDColumnIndex = inputHeaders.indexOf(«Account ID»);
  var accountNameColumnIndex = inputHeaders.indexOf(«Account Name»)
  var biddingStrategyColumnIndex = inputHeaders.indexOf(«Bidding Strategy»);
  var campaignNameContainsIndex = inputHeaders.indexOf(«Campaign Name Contains»);
  var campaignNameDoesNotContainIndex = inputHeaders.indexOf(«Campaign Name Doesn’t Contain»);
  var contactEmailsColumnIndex = inputHeaders.indexOf(«Contact email(s)»)
  var startDateColumnIndex = inputHeaders.indexOf(«Start Date»);
  var endDateColumnIndex = inputHeaders.indexOf(«End Date»);
  var outputHeaders = outputSheet.getRange(OUTPUT_HEADER_ROW + «:» + OUTPUT_HEADER_ROW).getValues()[0];
  var timeRunIndex = outputHeaders.indexOf(«Time Run»);

  //Get all rows of data.

  var allData = inputSheet.getRange(INPUT_DATA_ROW, 1, inputSheet.getLastRow() — (INPUT_HEADER_ROW + 1), inputSheet.getLastColumn()).getValues();

  //For each row of data:
  Logger.log(«Verifying each row of data…»)
  for (var i = 0; i < allData.length; i++) {
    var row = allData[i];
    if (row[statusColumnIndex] == «Paused») {
      continue;
    };
    var accountName = row[accountNameColumnIndex];
    var contacts = (row[contactEmailsColumnIndex]).split(‘,’).map(function(item){
      return item.trim();
    });
    var childAccount = getAccountId(row[accountIDColumnIndex], contacts, accountName);
    AdsManagerApp.select(childAccount);
    var dates = getDates([row[startDateColumnIndex], row[endDateColumnIndex]], tz, contacts, accountName);
    var combinedQueries = makeQueries(dates, row[campaignNameContainsIndex], row[campaignNameDoesNotContainIndex],row[biddingStrategyColumnIndex])
    var dataRow = getMetricsforRow(combinedQueries, contacts, accountName);
    var outputRows = [];
    outputRows = [row[accountNameColumnIndex], row[accountIDColumnIndex], row[biddingStrategyColumnIndex]]
      .concat(dataRow.map(function (data) {
        return data.value
      }));
    writeRows(outputSheet, outputRows);
    setDate(outputSheet, timeRunIndex);
  }
  Logger.log(«Success.»)
}

function getSpreadsheet(spreadsheetUrl) {
  Logger.log(‘Checking spreadsheet: ‘ + SPREADSHEET_URL + ‘ is valid.’);
  if (spreadsheetUrl.replace(/[AEIOU]/g, «X») == «https://docs.google.com/YXXR-SPRXXDSHXXT-XRL-HXRX») {
    throw («Problem with » + SPREADSHEET_URL +
    » URL: make sure you’ve replaced the default with a valid spreadsheet URL.»
    );
  }
  try {
    var spreadsheet = SpreadsheetApp.openByUrl(spreadsheetUrl);

    var sheet = spreadsheet.getSheets()[0];
    var sheetName = sheet.getName();
    sheet.setName(sheetName);

    return spreadsheet;
  } catch (e) {
    throw («Problem with » + SPREADSHEET_URL + » URL: ‘» + e + «‘. You may not have edit access»);
  }
}

function getAccountId(accountId, contacts, accountName) {
  var childAccount = AdsManagerApp.accounts().withIds([accountId]).get();
  if (childAccount.hasNext()) {
    return childAccount.next();
  }
  else {
    MailApp.sendEmail({
      to: contacts.join(),
      subject: «Bid Strategy Performance Monitor: error with account » + accountName,
      htmlBody: «Could not find account with ID: » + accountId + «.»
    });
  throw («Could not find account with ID: » + accountId);
  }
 
}

function getDates(dates, tz, contacts, accountName) {
  var validatedDates = dates.map(function (date) {
    if (date.length === 0) {
      var today = new Date()
      return Utilities.formatDate(today, tz, ‘yyyyMMdd’);
    }
    else {
      return Utilities.formatDate(new Date(date), tz, ‘yyyyMMdd’);
    }
  })
  if (validatedDates[0] <= validatedDates[1]) {
    return validatedDates;
  }
  else {
    MailApp.sendEmail({
      to: contacts.join(),
      subject: «Bid Strategy Performance Monitor: error with account » + accountName,
      htmlBody: («Invalid date ranges (yyyMMdd): End Date: «
        + validatedDates[1] + » precedes Start date: » + validatedDates[0])
    })
    throw («Invalid date ranges: End Date precedes Start Date.»);
  }
}

function makeQueries(dates, campaignNameContains, campaignNameDoesNotContain, biddingStrategy){
  var campaignNameContains = campaignNameContains.split(‘,’).map(function(item){
    return item.trim();
  });
  var campaignNameDoesNotContain = campaignNameDoesNotContain.split(‘,’).map(function(item){
    return item.trim();
  });
  var campaignFilterQueries = makeCampaignFilterStatements(campaignNameContains, campaignNameDoesNotContain, ignorePausedCampaigns);
  var biddingStrategyQuery = makeBiddingStrategyStatement((biddingStrategy));
  var combinedQueries = combineQueries(dates, biddingStrategyQuery, campaignFilterQueries);
  return combinedQueries;
}
function makeCampaignFilterStatements(campaignNameContains, campaignNameDoesNotContain, ignorePausedCampaigns) {
  var whereStatement = «WHERE «;
  var whereStatementsArray = [];

  if (ignorePausedCampaigns) {
    whereStatement += «CampaignStatus = ENABLED «;
  } else {
    whereStatement += «CampaignStatus IN [‘ENABLED’,’PAUSED’] «;
  }

  for (var i = 0; i < campaignNameDoesNotContain.length; i++) {
    if (campaignNameDoesNotContain  == «»){
      break;;
    }
    else {
    whereStatement += «AND CampaignName DOES_NOT_CONTAIN_IGNORE_CASE ‘»
      + campaignNameDoesNotContain[i].replace(/»/g, ‘\»‘) + «‘ «;
  }
  }

  if (campaignNameContains.length == 0) {
    whereStatementsArray = [whereStatement];

  }
  else {
    for (var i = 0; i < campaignNameContains.length; i++) {
      whereStatementsArray.push(whereStatement + ‘AND CampaignName CONTAINS_IGNORE_CASE «‘
        + campaignNameContains[i].replace(/»/g, ‘\»‘) + ‘» ‘
        );
    }
  }
  return whereStatementsArray;
}

function makeBiddingStrategyStatement(biddingStrategy) {
  if (biddingStrategy.length === 0) {
    return «»
  }
  else {
    for (var key in BIDDING_STRATEGIES) {
      if (key == biddingStrategy) {
        return » AND BiddingStrategyType = » + BIDDING_STRATEGIES[key]
      };
    }
  }
};

function combineQueries(dates, biddingStrategyQuery, campaignFilterQueries) {
  var combinedQueries = []
  for (var i = 0; i < campaignFilterQueries.length; i++) {
    combinedQueries.push(campaignFilterQueries[i]
      .concat(biddingStrategyQuery + » DURING » + dates[0] + «,» + dates[1]
      ));
  }
  return combinedQueries;
}

function getMetricsforRow(queries, contacts, accountName) {

  var ReportRows = getMetricsforSettings(queries, contacts, accountName);
  return ReportRows;
}

function getMetricsforSettings(queries, contacts, accountName) {
  var metricCounter = makeEmptyMetricCounter()
  var totalCampaignCount = 0;
  for (var i = 0; i < queries.length; i++) {
    var singleCampaignCount = 0
    var report = AdsApp.report(
      «SELECT » + METRICS.map(function (field) {
        return field;
      }).join(‘,’) + » FROM CAMPAIGN_PERFORMANCE_REPORT » + queries[i]
    );
    var rows = report.rows();
    if (rows.hasNext() === false) {
      MailApp.sendEmail({
        to: contacts.join(),
        subject: «Bid Strategy Performance Monitor: error with account » + accountName,
        htmlBody: «No campaigns found with the given settings: » + queries[i]
      });
    }
    while (rows.hasNext()) {
      singleCampaignCount = parseInt(singleCampaignCount) + parseInt(1);
      var row = rows.next();
      metricCounter.map(function (metric) {
        metric.value = parseFloat(metric.value) + (parseFloat((row[metric.name])) || 0);
        return metric.value;
      });
    }
    totalCampaignCount = parseInt(totalCampaignCount) + parseInt(singleCampaignCount)
  }
  metricCounter.push({name: «Total Campaigns», value: totalCampaignCount})
  return metricCounter;

  function makeEmptyMetricCounter() {
    var metricCounter = [];
    metricCounter.push(METRICS.map(function (field) {
      return {name: field, value: 0};
    }));
    return metricCounter[0];
  }
}

function writeRows(sheet, rows) {
  for (var i = 0; i < 5; i++) {
    try {
      sheet.getRange((sheet.getLastRow() + 1), 1, 1, rows.length).setValues([rows]);

      break;

    } catch (e) {
      if (e == «Exception: This action would increase the number of cells in the worksheet above the limit of 2000000 cells.») {
        Logger.log(«Could not write to spreadsheet: ‘» + e + «‘»);
        try {
          sheet.getRange(«R» + (sheet.getLastRow() + 2) + «C1»)
            .setValue(«Not enough space to write the data — try again in an empty spreadsheet»);
        } catch (e2) {
          Logger.log(«Error writing ‘not enough space’ message: » + e2);
        }
        break;
      }
      if (i == 4) {
        Logger.log(«Could not write to spreadsheet: ‘» + e + «‘»);
      }
    }
  }
}

function setDate(sheet, columnIndex) {
  var now = new Date();
  sheet.getRange((sheet.getLastRow()), columnIndex + 1).setValue(now);

Материалы по теме:

  • Четыре скрипта для управления ставками в Google Ads
  • Brainlabs разработал скрипт для заполнения пустых групп объявлений в AdWords
По материалам: www.searchengines.ru