<head><meta http-equiv="Content-Type" content="text/html; charset=utf-8"/><title>Finally Figured Out the Three GCP Credentials (Kinda)</title></head><body><article class="page sans"><header><h1 class="page-title">Finally Figured Out the Three GCP Credentials (Kinda)</h1><p class="page-description"></p><table class="properties"><tbody><tr class="property-row property-row-last_edited_time"><th><span class="icon property-icon"><div data-testid="/icons/clock_gray.svg" style="width:14px;height:14px;flex-shrink:0;transform:scale(1.2);mask:url(/icons/clock_gray.svg?mode=light) no-repeat center;-webkit-mask:url(/icons/clock_gray.svg?mode=light) no-repeat center;background-color:rgba(71, 70, 68, 0.6);fill:rgba(71, 70, 68, 0.6)"></div></span>Last edited</th><td><time>@2025年9月8日 21:28</time></td></tr><tr class="property-row property-row-multi_select"><th><span class="icon property-icon"><div data-testid="/icons/list_gray.svg" style="width:14px;height:14px;flex-shrink:0;transform:scale(1.2);mask:url(/icons/list_gray.svg?mode=light) no-repeat center;-webkit-mask:url(/icons/list_gray.svg?mode=light) no-repeat center;background-color:rgba(71, 70, 68, 0.6);fill:rgba(71, 70, 68, 0.6)"></div></span>Tags</th><td><span class="selected-value select-value-color-default">GCP</span><span class="selected-value select-value-color-orange">Post</span></td></tr></tbody></table></header><div class="page-body"><h1 class="">TL;DR</h1><p class="">OAuth 2.0 Client ID = Let users sign into your website</p><p class="">Service Account = Let your backend automatically access Google services</p><p class="">API Key = Permission slip for accessing public Google services<br/></p><p class="">Even simpler:</p><ul class="bulleted-list"><li style="list-style-type:disc">User login → OAuth 2.0</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Backend automation → Service Account</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Public APIs → API Key</li></ul><p class="">Memory tricks:</p><ul class="bulleted-list"><li style="list-style-type:disc">OAuth: Someone needs to be here (real person must authorize)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">Service Account: Agentless but also an agent (virtual worker)</li></ul><ul class="bulleted-list"><li style="list-style-type:disc">API Key: Just a simple neutral key</li></ul><p class="">
</p><p class="">
</p><h1 class="">OAuth 2.0 Client IDs - User Authorization for Login</h1><p class="">Recently I was working on GIS login (Google Identity Services - you know, that Google One Tap login popup like Medium has). Found out I needed OAuth 2.0 Client ID. At first I was confused because I was already using Firebase Auth for SSO. I thought, &quot;Is GIS a different system from Firebase Auth&#x27;s Google login?&quot;</p><p class="">
</p><p class="">Turns out both are OAuth 2.0 Client IDs. The key difference is just how they&#x27;re implemented, but <mark class="highlight-red">both need users to participate in the authorization</mark> flow. Actually, you can combine them - use GIS One Tap to get the credential, then pass it to Firebase Auth for unified management:</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// OAuth 2.0 example: user actively logs in
google.accounts.id.initialize({
  client_id: &quot;YOUR-FIREBASE-WEB-CLIENT-ID.apps.googleusercontent.com&quot;,
  callback: (response) =&gt; {
    // response.credential contains user&#x27;s JWT token
    // can be used directly with Firebase Auth
    firebase.auth().signInWithCredential(
      firebase.auth.GoogleAuthProvider.credential(response.credential)
    )
  }
});</code></pre><p class="">Steps to create OAuth 2.0 Client ID:</p><p class="">GCP Console → APIs &amp; Services → Credentials → Create Credentials → OAuth 2.0 Client ID → Choose application type → Set up authorized redirect URIs</p><p class="">
</p><p class="">
</p><h1 class="">Service Account - Backend Service-to-Service Automation</h1><p class="">Machine-to-machine communication, no user needed. Service Account is like a &quot;virtual employee&quot; role where you can define what permissions this role has.</p><p class="">I got stuck on a project before: tried using OAuth 2.0 to let my backend read Google Calendar. Didn&#x27;t work because OAuth needs a user to log in to access &quot;that user&#x27;s own&quot; Calendar. But what I wanted was for my system to automatically <mark class="highlight-red">read a &quot;specific&quot; Calendar without any user involvement</mark>. This is exactly what Service Account is for.</p><p class="">
</p><h3 class="">Real Example</h3><p class="">Let me use one of my previous projects. My backend needed to fetch Google Calendar events. The flow requires:</p><ol type="1" class="numbered-list" start="1"><li>Enable <mark class="highlight-red">Calendar API</mark> in GCP - opening the main door</li></ol><ol type="1" class="numbered-list" start="2"><li>Create Service Account - who can go through that door</li></ol><ol type="1" class="numbered-list" start="3"><li>Add the Service Account email to your Calendar&#x27;s share list (this step is crucial)</li></ol><ol type="1" class="numbered-list" start="4"><li>Download the Service Account&#x27;s JSON credentials file and extract the values to build auth before using the SDK</li></ol><p class="">Steps 2 and 3 are the key to making it work. I forgot step 3 back then and kept getting &quot;permission denied&quot; errors from the API.</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// Complete Service Account implementation example
function getGoogleAuth() {
  // Extract these three values from the downloaded JSON file into environment variables
  const clientEmail = process.env.GOOGLE_CLIENT_EMAIL      // SA&#x27;s email address
  const privateKey = process.env.GOOGLE_PRIVATE_KEY?.replace(/\\n/g, &#x27;\n&#x27;)  // SA&#x27;s private key (note the newline handling)
  const projectId = process.env.GOOGLE_PROJECT_ID          // GCP project ID

  return new google.auth.GoogleAuth({
    credentials: {
      client_email: clientEmail,
      private_key: privateKey,
      project_id: projectId,
    },
    scopes: [&#x27;https://www.googleapis.com/auth/calendar&#x27;],  // Define permission scope
  })
}

export function getCalendar() {
  return google.calendar({ version: &#x27;v3&#x27;, auth: getGoogleAuth() })
}

// Actual usage
const calendar = getCalendar()
const response = await calendar.events.list({
  calendarId: process.env.GOOGLE_CALENDAR_ID,   // Target calendar ID
  timeMin: formatDateTime(startDate),           // Query start time
  timeMax: formatDateTime(endDate),             // Query end time
  singleEvents: true,                           // Expand recurring events
  orderBy: &#x27;startTime&#x27;,                         // Sort by start time
  maxResults: 500,                              // Return max 500 entries
})
</code></pre><p class="">
</p><h3 class="">ADC Mechanism Supplement (Application Default Credentials)</h3><p class="">ADC helps simplify Service Account usage. When you deploy apps in GCP environment, the system automatically creates default Service Accounts. For example, Cloud Run creates:<br/><mark class="highlight-red">PROJECT_NUMBER-compute@developer.gserviceaccount.com</mark></p><p class="">These default SAs have basic permissions like writing logs, monitoring, etc. These aren&#x27;t sensitive operations anyway.</p><p class="">
</p><p class="">In the examples below, we don&#x27;t need to specify credentials - we let ADC find them automatically</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// Using ADC - works for GCP internal services
const authWithADC = new google.auth.GoogleAuth({
  scopes: [&#x27;https://www.googleapis.com/auth/cloud-platform&#x27;], // Most GCP APIs
});

// But this WONT WORK for Google Workspace APIs (like Calendar)
const impossibleAuth = new google.auth.GoogleAuth({
  scopes: [&#x27;https://www.googleapis.com/auth/calendar&#x27;] // Need to specify credentials explicitly
});</code></pre><p class="">ADC credential search order:</p><ol type="1" class="numbered-list" start="1"><li>Does your environment have the <mark class="highlight-red">GOOGLE_APPLICATION_CREDENTIALS</mark> variable? If yes, use that as credential</li></ol><ol type="1" class="numbered-list" start="2"><li><mark class="highlight-red">gcloud</mark> CLI login status, maybe you have a default auth? Oh yeah, there&#x27;s a default credential, let&#x27;s use that</li></ol><ol type="1" class="numbered-list" start="3"><li>GCP service&#x27;s default Service Account with some basic functions. Like Cloud Run has the basic <mark class="highlight-red">PROJECT_NUMBER-compute@developer.gserviceaccount.com</mark> that can access other basic services.</li></ol><ol type="1" class="numbered-list" start="4"><li>Cloud Shell user account - temporary terminal where you can log in with User or SA.</li></ol><p class="">
</p><p class="">When Service Account is between GCP services, it&#x27;s pretty convenient. Basic permissions rely on default SA, like Cloud Run can use log functionality through default SA. If you need advanced permissions, just grant additional permissions to this SA.</p><p class="">But Google Calendar belongs to Google Workspace API services, not GCP IAM scope. You can&#x27;t let it automatically fallback to find default SA. You must explicitly specify credentials plus the calendar must give SA permissions.</p><p class="">
</p><h3 class="">Compare with AWS</h3><p class="">AWS is actually similar - every AWS service has its own default Role with specific permissions.</p><p class="">For example: when you use Lambda, the system very clearly gives you an <mark class="highlight-red">AWSLambdaBasicExecutionRole-xxxxxxx </mark>when you create Lambda A resource, explicitly telling you it can CreateLogStream, PutLogEvents.<br/></p><p class="">But when you create another Lambda B, it gets a different <mark class="highlight-red">AWSLambdaBasicExecutionRole-yyyyyy</mark>. Even though the content is the same, they&#x27;re independent Roles.</p><p class="">By the way, when you delete Lambda, the Role doesn&#x27;t get deleted with it.</p><p class="">
</p><p class="">But in GCP, when you create two Cloud Runs, they share the same Service Account. In terms of user experience, AWS relies more on admins manually &quot;explicitly&quot; attaching permissions. Using GCP feels more seamless and painless, while AWS gives you very high control.</p><p class="">
</p><h1 class="">API Keys - Permission Slips for Public Services</h1><p class="">Like Google Maps uses this: accessing public services that don&#x27;t involve personal privacy</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// Map data anyone can see
fetch(`https://maps.googleapis.com/maps/api/geocode/json?address=Taipei&amp;key=${API_KEY}`)</code></pre><p class=""><strong>Why need API Key</strong>: Prevent abuse, limit QPS, track usage, billing management.</p><p class=""><strong>Creation process</strong>: GCP Console → APIs &amp; Services → Credentials → Create Credentials → API Key</p><p class="">
</p><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">// API Key usage example: accessing public map data

// No API Key = 403 Forbidden
fetch(`https://maps.googleapis.com/maps/api/geocode/json?address=Taipei`)

// With API Key = works fine (within free quota)
fetch(`https://maps.googleapis.com/maps/api/geocode/json?address=Taipei&amp;key=${API_KEY}`)
  .then(response =&gt; response.json())
  .then(data =&gt; {
    console.log(&#x27;Geocoding results:&#x27;, data.results);
  })
  .catch(error =&gt; {
    console.error(&#x27;API call failed:&#x27;, error);
  });</code></pre><p class="">
</p><p class="">Just like AWS API Gateway also needs API Key - you need identity to use services. Every API Key has free quota limits, QPS limits, domain whitelists and other security controls.</p><p class="">
</p><p class="">
</p><h1 class="">Putting It All Together</h1><p class="">Complete frontend + backend flow:</p><ol type="1" class="numbered-list" start="1"><li>Frontend uses OAuth 2.0 for user login</li></ol><ol type="1" class="numbered-list" start="2"><li>Frontend passes user token to backend for identity verification</li></ol><ol type="1" class="numbered-list" start="3"><li>Backend uses Service Account for Google service integration</li></ol><ol type="1" class="numbered-list" start="4"><li>If you need public services like Maps, add an API Key</li></ol><h1 class="">Decision Tree</h1><script src="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/prism.min.js" integrity="sha512-7Z9J3l1+EYfeaPKcGXu3MS/7T+w19WtKQY/n+xzmw4hZhJ9tyYmcUS+4QqAlzhicE5LAfMQSF3iFTK9bQdTxXg==" crossorigin="anonymous" referrerPolicy="no-referrer"></script><link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/prism/1.29.0/themes/prism.min.css" integrity="sha512-tN7Ec6zAFaVSG3TpNAKtk4DOHNpSwKHxxrsiw4GHKESGPs5njn/0sMCUMl2svV4wo4BK/rCP7juYz+zx+l6oeQ==" crossorigin="anonymous" referrerPolicy="no-referrer"/><pre class="code code-wrap"><code class="language-JavaScript" style="white-space:pre-wrap;word-break:break-all">if (need user) {
  if (need user involved actively) return &#x27;OAuth 2.0 Client ID&#x27;
  else return &#x27;Service Account&#x27;
} else {
  return &#x27;API KEY&#x27;
}</code></pre><p class="">
</p><p class="">
</p><p class="">And that&#x27;s it! Once you understand these three differences, you won&#x27;t be confused when you see GCP Credentials anymore. Don&#x27;t be like me who tried using OAuth 2.0 for everything at first, then wondered why the backend wouldn&#x27;t work...</p></div></article><span class="sans" style="font-size:14px;padding-top:2em"></span></body>
~/
about
posts
frontbacknetworkoscloud
readings
css
bookmarks
archives
And maybe its just slow involvement at first, but try to sort of creep your career in that direction, because if youre not being challenged, if youre not a little bit scared all the time, just a little bit, then youre not gonna improve. - The Myth of the Genius Programmer
© 2024 jialin00.com Original content since 2022