Help us improve
Share bugs, ideas, or general feedback.
From arcgis-maps-sdk-js-ai-context
Implements ArcGIS authentication using OAuth 2.0, API keys, and IdentityManager for accessing secured services, portal items, and user content in JavaScript apps.
npx claudepluginhub saschabrunnerch/arcgis-maps-sdk-js-ai-context --plugin arcgis-maps-sdk-js-ai-contextHow this skill is triggered — by the user, by Claude, or both
Slash command
/arcgis-maps-sdk-js-ai-context:arcgis-authenticationThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Use this skill for implementing authentication, OAuth, API keys, and identity management.
Manages ArcGIS portal content: saves WebMaps/WebScenes, bookmarks, slides, portal items. For content persistence, map/scene configuration, navigation presets.
Provides Python patterns for external service authentication using API keys, OAuth, tokens. Includes verification flows, smoke tests, env checks, and error handling with leyline.
Detects app framework (React, Next.js, Vue, Angular, Express, Fastify, React Native), sets up Auth0 account/CLI/app, and routes to SDK-specific authentication workflows.
Share bugs, ideas, or general feedback.
Use this skill for implementing authentication, OAuth, API keys, and identity management.
import OAuthInfo from "@arcgis/core/identity/OAuthInfo.js";
import esriId from "@arcgis/core/identity/IdentityManager.js";
import Portal from "@arcgis/core/portal/Portal.js";
import esriConfig from "@arcgis/core/config.js";
const OAuthInfo = await $arcgis.import("@arcgis/core/identity/OAuthInfo.js");
const esriId = await $arcgis.import("@arcgis/core/identity/IdentityManager.js");
const Portal = await $arcgis.import("@arcgis/core/portal/Portal.js");
The simplest authentication method. Set once for all SDK requests.
import esriConfig from "@arcgis/core/config.js";
esriConfig.apiKey = "YOUR_API_KEY";
// Now basemaps and services will use the API key
const map = new Map({
basemap: "arcgis/streets",
});
<script src="https://js.arcgis.com/5.0/"></script>
<script>
$arcgis.config.apiKey = "YOUR_API_KEY";
</script>
import OAuthInfo from "@arcgis/core/identity/OAuthInfo.js";
import esriId from "@arcgis/core/identity/IdentityManager.js";
const oauthInfo = new OAuthInfo({
appId: "YOUR_APP_ID",
popup: false, // false = redirect, true = popup window
});
esriId.registerOAuthInfos([oauthInfo]);
async function checkSignIn() {
try {
await esriId.checkSignInStatus(oauthInfo.portalUrl + "/sharing");
const portal = new Portal({ authMode: "immediate" });
await portal.load();
console.log("Signed in as:", portal.user.username);
return portal;
} catch {
console.log("Not signed in");
return null;
}
}
async function signIn() {
try {
const credential = await esriId.getCredential(
oauthInfo.portalUrl + "/sharing",
);
return credential;
} catch (error) {
console.error("Sign in failed:", error);
}
}
function signOut() {
esriId.destroyCredentials();
window.location.reload();
}
import OAuthInfo from "@arcgis/core/identity/OAuthInfo.js";
import esriId from "@arcgis/core/identity/IdentityManager.js";
import Portal from "@arcgis/core/portal/Portal.js";
const oauthInfo = new OAuthInfo({ appId: "YOUR_APP_ID" });
esriId.registerOAuthInfos([oauthInfo]);
// Check if already signed in
esriId
.checkSignInStatus(oauthInfo.portalUrl + "/sharing")
.then(() => {
const portal = new Portal({ authMode: "immediate" });
return portal.load();
})
.then((portal) => {
console.log("Welcome,", portal.user.fullName);
displayUserContent(portal);
})
.catch(() => {
showSignInButton();
});
function showSignInButton() {
document.getElementById("signInBtn").onclick = () => {
esriId
.getCredential(oauthInfo.portalUrl + "/sharing")
.then(() => window.location.reload());
};
}
const oauthInfo = new OAuthInfo({
appId: "YOUR_APP_ID",
portalUrl: "https://your-portal.com/portal",
popup: true,
});
esriId.registerOAuthInfos([oauthInfo]);
import esriConfig from "@arcgis/core/config.js";
esriConfig.portalUrl = "https://your-portal.com/portal";
esriId.registerToken({
server: "https://services.arcgis.com/",
token: "YOUR_TOKEN",
});
const credential = await esriId.getCredential(
"https://services.arcgis.com/...",
);
console.log("Token:", credential.token);
console.log("Expires:", new Date(credential.expires));
import Portal from "@arcgis/core/portal/Portal.js";
const portal = new Portal({ authMode: "immediate" });
await portal.load();
console.log("Username:", portal.user.username);
console.log("Full name:", portal.user.fullName);
console.log("Email:", portal.user.email);
console.log("Role:", portal.user.role);
console.log("Thumbnail:", portal.user.thumbnailUrl);
console.log("Org name:", portal.name);
console.log("Org ID:", portal.id);
import PortalQueryParams from "@arcgis/core/portal/PortalQueryParams.js";
const queryParams = new PortalQueryParams({
query: `owner:${portal.user.username}`,
sortField: "modified",
sortOrder: "desc",
num: 20,
});
const result = await portal.queryItems(queryParams);
result.results.forEach((item) => {
console.log(item.title, item.type, item.id);
});
// Clear all stored credentials
esriId.destroyCredentials();
// Find a specific credential
const credential = esriId.findCredential("https://services.arcgis.com/...");
import esriConfig from "@arcgis/core/config.js";
esriConfig.request.trustedServers.push("https://services.arcgis.com");
esriConfig.request.trustedServers.push("https://your-server.com");
import esriConfig from "@arcgis/core/config.js";
// Configure proxy for cross-origin requests
esriConfig.request.proxyUrl = "/proxy/";
// Configure proxy rules
esriConfig.request.proxyRules.push({
urlPrefix: "https://services.arcgis.com",
proxyUrl: "/proxy/",
});
App ID redirect URIs: The App ID must be registered with correct redirect URIs at developers.arcgis.com. Mismatched URIs cause silent authentication failures.
Popup blockers on mobile: OAuth popup-based sign-in fails on mobile browsers.
// Anti-pattern: using popup flow on mobile
const oauthInfo = new OAuthInfo({
appId: "YOUR_APP_ID",
popup: true, // Blocked by most mobile browsers
});
// Correct: use redirect flow for mobile
const isMobile = /iPhone|iPad|Android/i.test(navigator.userAgent);
const oauthInfo = new OAuthInfo({
appId: "YOUR_APP_ID",
popup: !isMobile,
});
Portal URL trailing slash: A trailing slash in the portal URL causes token validation to fail.
// Anti-pattern
const portal = new Portal({
url: "https://myorg.maps.arcgis.com/", // Trailing slash
});
// Correct
const portal = new Portal({
url: "https://myorg.maps.arcgis.com", // No trailing slash
});
Token expiration: Tokens expire. Handle refresh or re-authentication gracefully.
CORS errors: Configure trusted servers or use a proxy for cross-origin requests to non-ArcGIS servers.
import esriConfig from "@arcgis/core/config.js";
esriConfig.request.interceptors.push({
urls: "https://services.arcgis.com",
before: (params) => {
// Add custom headers or modify request
console.log("Request to:", params.url);
},
after: (response) => {
// Process response
console.log("Response status:", response.httpStatus);
},
error: (error) => {
console.error("Request failed:", error);
},
});
identity-oauth-basic - Basic OAuth 2.0 authentication setupidentity-oauth-component - OAuth component-based authenticationbasemaps-portal - Authenticated portal access for basemapswebmap-save - Saving maps (requires authentication)arcgis-portal-content for managing portal items, WebMaps, and WebScenes.arcgis-rest-services for premium REST services requiring authentication.arcgis-starter-app for app scaffolding with authentication setup.