Click here to Skip to main content
15,881,089 members
Articles / Programming Languages / C#
Tip/Trick

.NET Library for Accessing and Querying Google Analytics V3 via Service Account

Rate me:
Please Sign up or sign in to vote.
5.00/5 (5 votes)
30 Jan 2014CPOL4 min read 42.3K   12   6
Simple Google Analytics Access, with Service Account Credential OAuth2

Introduction

Simply .NET Project for Google Analytics API v3 Usage

Tired of looking for help, things isn't really working for you! get started to analytics quickly from here. Want to work with analytics the way it should be, that's the place for you.

This article will give you a quick jump to analytics and using it with Simple Analytics open source library that provides a framework on top of the .NET Google Analytics API, simplifying the access, retrieval and working with the Reporting API in a simple way

Background

First you need to configure your Google developer account and register a project in the cloud console.

ps. those instructions are using the new console for old console users you'll find same things but some are under different names in the menu

  1. create an account on Google Cloud Console
  2. Create a new Project from the console or select an existing project
  3. Under APIs and Auths : Enable the Analytics API
  4. Go to Credentials Page
    1. Create New Client ID
    2. Select Service Account
    3. you will be prompted to download a Key file store it some where safe as we'll be using it in authentication
    4. A table for service account credentials will be displayed to you containing Client ID, Email address, and public key finger prints
  5. Copy the email address then login in to your Analytics Web Account
    1. Go the Admin Section under it select the profile you want to access and under user accounts paste the email provided to you
    2. It's important to do that on both the profile (site) and the account
  6. Go to git hub and download the Simple Analytics Library
  7. In Visual Studio select the project you want to work on add the Analytics.dll library to it
  8. Run the following command under
Package Manager Console from : Tools -> Library Package Manager -> Package Manager Console

and execute the below command this will install and reference the google analytics API in your project

PM> Install-Package Google.Apis.Analytics.v3 -Pre [PROJECT_NAME] 

ok now you're ready to go

Using the code

Using the library is simple first you need to authenticate your self using the Google Service OAuth2 one line is required, please note that the email below is the email provided to you in the console service account section

C#
Analytics.AnalyticsManager manager = new Analytics.AnalyticsManager(Server.MapPath("~/bin/privatekey.p12"), "YOUR_EMAIL"); 

Now we need to query the profiles you have access to, and set a default Profile to work with you could get the number of the profile from the analytics url it's the number after p

https://www.google.com/analytics/web/?hl=en#management/Settings/a32880022w60002165p61347423/

C#
manager.LoadAnalyticsProfiles();<br />
manager.SetDefaultAnalyticProfile("80425770");

now we set the metrics and dimensions, a complete Reporting API Commands are placed in the project so you could easily set metrics and dimensions

C#
List<Analytics.Data.DataItem> metrics = new List<Analytics.Data.DataItem>();
metrics.Add(Analytics.Data.Visitor.Metrics.visitors);
metrics.Add(Analytics.Data.Session.Metrics.visits);
List<Analytics.Data.DataItem> dimensions = new List<Analytics.Data.DataItem>();
dimensions.Add(Analytics.Data.GeoNetwork.Dimensions.country);
dimensions.Add(Analytics.Data.GeoNetwork.Dimensions.city);  

now you're done get a table with analytics data

C++
System.Data.DataTable table = manager.GetGaDataTable(DateTime.Today.AddDays(-3),DateTime.Today, metrics, dimensions, null, metrics); 

that's it you could easily get any API data by changing/adding the metrics and dimensions properties

Library Implementation

The library implementation has a major class player called AnalyticsManager the analytics manager allows to initiate an authentication through OAuth2 Service Account Authentication

C#
var certificate = new X509Certificate2(certificateKeyPath, "notasecret", X509KeyStorageFlags.Exportable);
string x = certificate.IssuerName.Name;
credential = new ServiceAccountCredential(new ServiceAccountCredential.Initializer(apiEmail)
    {
                    Scopes = new[] { Google.Apis.Analytics.v3.AnalyticsService.Scope.Analytics }
        }.FromCertificate(certificate));

analyticsService = new Google.Apis.Analytics.v3.AnalyticsService(new BaseClientService.Initializer()
    {
            HttpClientInitializer = credential,
                    ApplicationName = "Jareeda"
        });

a certificate is created from the passed key the notasecret password is the same password for all key files from google nevertheless the signature differs in the key it self so the combination gives different signatures.

a service account credential is initialized and passed to it the Scope "Analytics" that's an enumerator where you could change it to AnalyticsReadOnly, AnalyticsEdit, AnalyticsManageUsers each will give you different access rights, depending on the permissions given from the Analytics Web.

an AnalyticsService object is then created this object will be used for querying along the class afterword

analytics methods require you pass a table_id or a profile_id

C#
public System.Data.DataTable GetGaDataTable(DateTime startDate, DateTime endDate, List<Data.DataItem> metricsList, List<Data.DataItem> dimensionsList, List<Data.DataItem> filtersList, int? maxResults, Google.Apis.Analytics.v3.DataResource.GaResource.GetRequest.OutputEnum? output, Google.Apis.Analytics.v3.DataResource.GaResource.GetRequest.SamplingLevelEnum? samplingLevel, List<Data.DataItem> segmentList, List<Data.DataItem> sortList, int? startIndex, List<Data.DataItem> fieldsList)
        {
            if (DefaultProfile == null)
                throw new Exception("Please set a default profile first using SetDefaultAnalyticProfile method");
            
            Google.Apis.Analytics.v3.Data.GaData gaData = GetGaData("ga:" + DefaultProfile.Id, startDate, endDate, Data.DataItem.GetString(metricsList), Data.DataItem.GetString(dimensionsList), Data.DataItem.GetString(filtersList), maxResults, output, samplingLevel, Data.DataItem.GetString(segmentList), Data.DataItem.GetString(sortList), startIndex, Data.DataItem.GetString(fieldsList));
            System.Data.DataTable table = BuildTableColumns(metricsList, dimensionsList);
            if(gaData != null)
                table = BuildTableRows(gaData, table);
            return table;
        }

in this method I pass the max parameters needed for retrieving GaData from analytics API where the passed DataItem List are the complete Dimensions and Metrics List Organized By Category based on the Google Complete Metrics and Dimensions Reference Guide I have parsed the complete guide and created an XML File for all the dimensions and metrics

XML
<DataCategory Name="Session">
    <ItemType Name="Dimensions" Value="1">
      <DataItem Name="visitLength" APICommand="ga:visitLength" WebViewName="Visit Duration" AppViewName="Session Duration" DataType="STRING" AllowedInSegments="True">The length of a visit to your property measured in seconds and reported in second increments. The value returned is a string.</DataItem>
    </ItemType>
    <ItemType Name="Metrics" Value="2">
      <DataItem Name="visits" APICommand="ga:visits" WebViewName="Visits" AppViewName="Sessions" DataType="INTEGER" AllowedInSegments="True">Counts the total number of sessions.</DataItem>
      <DataItem Name="bounces" APICommand="ga:bounces" WebViewName="Bounces" AppViewName="" DataType="INTEGER" AllowedInSegments="True">The total number of single page (or single engagement hit) sessions for your property.</DataItem>
      <DataItem Name="timeOnSite" APICommand="ga:timeOnSite" WebViewName="Visit Duration" AppViewName="Session Duration" DataType="TIME" AllowedInSegments="True">The total duration of visitor sessions represented in total seconds.</DataItem>
    </ItemType>
    <ItemType Name="Calculated" Value="3">
      <DataItem Name="visitBounceRate" APICommand="ga:visitBounceRate" WebViewName="Bounce Rate" AppViewName="" DataType="PERCENT" AllowedInSegments="False">The percentage of single-page visits (i.e., visits in which the person left your property from the first page). (ga:bounces / ga:visits ) </DataItem>
      <DataItem Name="avgTimeOnSite" APICommand="ga:avgTimeOnSite" WebViewName="Avg. Visit Duration" AppViewName="Avg. Session Duration" DataType="TIME" AllowedInSegments="False">The average duration visitor sessions represented in total seconds. (ga:timeOnSite / ga:visits ) </DataItem>
    </ItemType>
  </DataCategory> 

placed the XML file in the resources of the project and generated Classes based on the Categories containing Classes of Metrics and Dimensions for each that contains DataItem Objects which contains the definition of the complete properties for the commands retrieving properties from the XML Resource File - giving it the flavor of .NET Resources usage.

 

all public variables placed in classes like Session, Visitor etc. have description attribute above taken from the analytics API so you could reference features easily without having to get back to the API that allows you to reference features in code with intellisense on it

C#
List<Analytics.Data.DataItem> metrics = new List<Analytics.Data.DataItem>();
metrics.Add(Analytics.Data.Session.Metrics.visits);
metrics.Add(Analytics.Data.Visitor.Metrics.newVisits);

List<Analytics.Data.DataItem> dimensions = new List<Analytics.Data.DataItem>();
dimensions.Add(Analytics.Data.GeoNetwork.Dimensions.country);
dimensions.Add(Analytics.Data.Time.Dimensions.month); 

there are other 4 Overloaded Methods for the GetGaDataTable mentioned above

C#
 DataTable GetGaDataTable(DateTime startDate, DateTime endDate, List<Data.DataItem> metricsList)
C#
DataTable GetGaDataTable(DateTime startDate, DateTime endDate, List<Data.DataItem> metricsList, List<Data.DataItem> sortList)
C#
DataTable GetGaDataTable(DateTime startDate, DateTime endDate, List<Data.DataItem> metricsList, List<Data.DataItem> dimensionsList, List<Data.DataItem> filtersList, List<Data.DataItem> sortList)
C#
DataTable GetGaDataTable(DateTime startDate, DateTime endDate, List<Data.DataItem> metricsList, List<Data.DataItem> dimensionsList, List<Data.DataItem> filtersList, Google.Apis.Analytics.v3.DataResource.GaResource.GetRequest.SamplingLevelEnum? samplingLevel, List<Data.DataItem> sortList, List<Data.DataItem> fields) 

the returned data table contains not just the data but preserving the data types of the data too, so string, integer, float etc. types are assigned to the columns this means you could bind it directly and start operating on the tables  

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Software Developer (Senior)
Egypt Egypt
Fun Coder Smile | :) My Job is my Hobby Smile | :)

Comments and Discussions

 
QuestionReporting v4 Pin
gadhiya xxxxxz23-Aug-17 9:18
gadhiya xxxxxz23-Aug-17 9:18 
QuestionPlease set a default profile first using SetDefaultAnalyticProfile method Pin
nurullahd20-Sep-14 11:12
nurullahd20-Sep-14 11:12 
GeneralThanks a lot for help. Pin
om104224-Jun-14 1:32
om104224-Jun-14 1:32 
QuestionError on profile Pin
Member 1015321911-Jun-14 5:02
Member 1015321911-Jun-14 5:02 
QuestionUse filters Pin
driperdk_fr28-Feb-14 1:01
driperdk_fr28-Feb-14 1:01 
AnswerRe: Use filters Pin
rmostafa12-Mar-14 0:55
rmostafa12-Mar-14 0:55 
hey thanks for passing by please check this issue on the github hope it helps
https://github.com/rmostafa/DotNetAnalyticsAPI/issues/1[^]

let me know if I could by of any support

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.