{ "cells": [ { "attachments": {}, "cell_type": "markdown", "metadata": {}, "source": [ "## XBRL US API - FERC schedule by entity \n", "\n", "### Authenticate for access token \n", "Click in the gray code cell below, then click the Run button above to execute the cell. Type your XBRL US Web account email, account password, Client ID, and secret as noted, pressing the Enter key on the keyboard after each entry.\n", "\n", "XBRL US limits records returned for a query to improve efficiency; this script loops to collect all data from the Public Filings Database for a query. **Non-members might not be able to return all data for a query** - join XBRL US for comprehensive access - https://xbrl.us/join." ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [], "source": [ "import os, re, sys, json\n", "import requests\n", "import pandas as pd\n", "from IPython.display import display, HTML\n", "import numpy as np\n", "import getpass\n", "from datetime import datetime\n", "import urllib\n", "from urllib.parse import urlencode\n", "\n", "\n", "class tokenInfoClass:\n", " access_token = None\n", " refresh_token = None\n", " email = None\n", " username = None\n", " client_id = None\n", " client_secret = None\n", " url = 'https://api.xbrl.us/oauth2/token'\n", " headers = {\"Content-Type\": \"application/x-www-form-urlencoded\"}\n", "\n", "def refresh(info):\n", " refresh_auth = {\n", " 'client_id': info.client_id,\n", " 'client_secret' : info.client_secret,\n", " 'grant_type' : 'refresh_token',\n", " 'platform' : 'ipynb',\n", " 'refresh_token' : info.refresh_token\n", " }\n", " refreshres = requests.post(info.url, data=refresh_auth, headers=info.headers)\n", " refresh_json = refreshres.json()\n", " info.access_token = refresh_json.get('access_token')\n", " info.refresh_token = refresh_json.get('refresh_token')\n", " print('Your access token(%s) is refreshed for 60 minutes. If it expires again, run this cell to generate a new token and continue to use the query cells below.' % (info.access_token))\n", " return info\n", "\n", "tokenInfo = tokenInfoClass()\n", "\n", "# Helper to prompt only if value is missing\n", "def prompt_if_missing(value, prompt_text, secret=False):\n", " if value:\n", " return value\n", " if secret:\n", " return getpass.getpass(prompt=prompt_text)\n", " return input(prompt_text)\n", "\n", "# Load credentials (if .json exists)\n", "creds = {}\n", "if os.path.exists('creds.json'):\n", " try:\n", " with open('creds.json', 'r') as f:\n", " creds = json.load(f)\n", " print(\"Loaded .json\")\n", " except Exception as e:\n", " print(\"Warning: failed to read from .json:\", e)\n", "\n", "if creds:\n", " # Try nested prod object first\n", " selected = None\n", " if isinstance(creds.get('prod'), dict):\n", " selected = creds['prod']\n", " \n", " # Next, try prod-prefixed keys\n", " if not selected:\n", " selected = {}\n", " keys = ['email', 'password', 'client_id', 'client_secret']\n", " for k in keys:\n", " prefixed_key = 'prod' + k\n", " if creds.get(prefixed_key):\n", " selected[k] = creds.get(prefixed_key)\n", " # fall back to top-level key if prod variant not found\n", " elif creds.get(k):\n", " selected[k] = creds.get(k)\n", " \n", " # Verify we have all required keys\n", " if not all(selected.get(k) for k in ('email', 'password', 'client_id', 'client_secret')):\n", " # Fill in missing values from prompts\n", " selected = {\n", " 'email': selected.get('email'),\n", " 'password': selected.get('password'),\n", " 'client_id': selected.get('client_id'),\n", " 'client_secret': selected.get('client_secret')\n", " }\n", " \n", " # Assign values, prompting for any missing ones\n", " tokenInfo.email = prompt_if_missing(selected.get('email'), 'Enter your XBRL US Web account email: ')\n", " tokenInfo.password = prompt_if_missing(selected.get('password'), 'Password: ', secret=True)\n", " tokenInfo.client_id = prompt_if_missing(selected.get('client_id'), 'Client ID: ', secret=True)\n", " tokenInfo.client_secret = prompt_if_missing(selected.get('client_secret'), 'Secret: ', secret=True)\n", "\n", " print('Using credentials from .json as available.')\n", "else:\n", " # No creds.json — prompt the user\n", " tokenInfo.email = input('Enter your XBRL US Web account email: ')\n", " tokenInfo.password = getpass.getpass(prompt='Password: ')\n", " tokenInfo.client_id = getpass.getpass(prompt='Client ID: ')\n", " tokenInfo.client_secret = getpass.getpass(prompt='Secret: ')\n", "\n", "body_auth = {'username' : tokenInfo.email,\n", " 'client_id': tokenInfo.client_id,\n", " 'client_secret' : tokenInfo.client_secret,\n", " 'password' : tokenInfo.password,\n", " 'grant_type' : 'password',\n", " 'platform' : 'ipynb' }\n", "\n", "#print(body_auth)\n", "\n", "payload = urlencode(body_auth)\n", "res = requests.request(\"POST\", tokenInfo.url, data=payload, headers=tokenInfo.headers)\n", "auth_json = res.json()\n", "\n", "if 'error' in auth_json:\n", " print(\"\\n\\nThere was a problem generating the access token: %s Run the first cell again and enter the credentials.\" % (auth_json.get('error_description', auth_json)))\n", "else:\n", " tokenInfo.access_token = auth_json.get('access_token')\n", " tokenInfo.refresh_token = auth_json.get('refresh_token')\n", " if tokenInfo.access_token and tokenInfo.refresh_token:\n", " print (\"\\n\\nYour access token expires in 60 minutes. After it expires, it should be regenerated automatically. If not, run the cell rerun the first query cell. \\n\\nFor now, skip ahead to the section 'Make a Query'.\")\n", " else:\n", " print(\"\\n\\nAuthentication completed but tokens were not returned. Response: {}\".format(auth_json))\n", "\n", "#print(vars(tokenInfo))\n", "if tokenInfo.access_token and tokenInfo.refresh_token:\n", " print('\\n\\naccess token: ' + tokenInfo.access_token + ' refresh token: ' + tokenInfo.refresh_token)\n", "else:\n", " print('\\n\\nNo access token was generated. Check the messages above for errors.')" ] }, { "attachments": {}, "cell_type": "markdown", "metadata": {}, "source": [ "### Make a query \n", "After the access token confirmation appears above, you can modify the query below and use the **_Cell >> Run_** menu option with the cell **immediately below this text** to run the query for updated results. \n", "\n", "The sample results are from 10+ years of data for companies filing data on the _Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion_ , and may take several minutes to recreate. \n", "\n", "Modify the **_XBRL\\_Elements_** to return data from a different FERC Form 1 Schedule, and/or change the **_entity\\_codes_** to shorten the list. You can also change the base taxonomy corresponding to the Form by year or Form type. The queries to get lists are in the commented lines above each variable.\n", " \n", "Refer to XBRL API documentation at https://xbrlus.github.io/xbrl-api/#/Facts/getFactDetails for other endpoints and parameters to filter and return. " ] }, { "cell_type": "code", "execution_count": null, "metadata": {}, "outputs": [ { "name": "stdout", "output_type": "stream", "text": [ "On Mon Nov 18 16:45:12 2024 info@xbrl.us (client ID: 69e1257c ...) started the query and\n", "up to 5000 records are found so far ...\n", " - this set contained fewer than the 5000 possible, only 4173 records.\n", "\n", "At Mon Nov 18 16:45:28 2024, the query finished with 4173 rows returned in 0:00:15.380693 for \n", "https://api.xbrl.us/api/v1/cube/search?unique&cube.description=200+-+Schedule+-+Summary+of+Utility+Plant+and+Accumulated+Provisions+for+Depreciation,+Amortization+and+Depletion&entity.code=C000533,C001111&fields=period.fiscal-year.sort(DESC),period.fiscal-period.sort(DESC),entity.code,report.entity-name,report.id,cube.description.sort(ASC),cube.tree-sequence.sort(ASC),cube.primary-local-name,fact.value,unit,dimensions.count,dimension-pair.sort(ASC)\n" ] }, { "data": { "text/html": [ "
| \n", " | period.fiscal-year | \n", "period.fiscal-period | \n", "entity.code | \n", "report.entity-name | \n", "report.id | \n", "cube.description | \n", "cube.tree-sequence | \n", "cube.primary-local-name | \n", "fact.value | \n", "unit | \n", "dimensions.count | \n", "dimension-pair | \n", "
|---|---|---|---|---|---|---|---|---|---|---|---|---|
| 0 | \n", "2024 | \n", "2Q | \n", "C000533 | \n", "Kentucky Power Company | \n", "753929 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "3 | \n", "UtilityPlantInServiceClassified | \n", "3,285,598,514.00 | \n", "USD | \n", "0 | \n", "\n", " |
| 1 | \n", "2024 | \n", "2Q | \n", "C000533 | \n", "Kentucky Power Company | \n", "753929 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "3 | \n", "UtilityPlantInServiceClassified | \n", "3,285,598,514.00 | \n", "USD | \n", "1 | \n", "[{'UtilityTypeAxis': 'ElectricUtilityMember'}] | \n", "
| 2 | \n", "2024 | \n", "2Q | \n", "C001111 | \n", "Baltimore Gas and Electric Company | \n", "753841 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "3 | \n", "UtilityPlantInServiceClassified | \n", "10,068,795,285.00 | \n", "USD | \n", "1 | \n", "[{'UtilityTypeAxis': 'ElectricUtilityMember'}] | \n", "
| 3 | \n", "2024 | \n", "2Q | \n", "C001111 | \n", "Baltimore Gas and Electric Company | \n", "753841 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "3 | \n", "UtilityPlantInServiceClassified | \n", "1,244,861,502.00 | \n", "USD | \n", "1 | \n", "[{'UtilityTypeAxis': 'CommonUtilityMember'}] | \n", "
| 4 | \n", "2024 | \n", "2Q | \n", "C001111 | \n", "Baltimore Gas and Electric Company | \n", "753841 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "3 | \n", "UtilityPlantInServiceClassified | \n", "15,404,286,687.00 | \n", "USD | \n", "0 | \n", "\n", " |
| ... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "... | \n", "
| 4168 | \n", "2010 | \n", "Y | \n", "C001111 | \n", "Baltimore Gas and Electric Company | \n", "418249 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "16 | \n", "UtilityPlantNet | \n", "4,541,951,837.00 | \n", "USD | \n", "0 | \n", "\n", " |
| 4169 | \n", "2010 | \n", "Y | \n", "C000533 | \n", "Kentucky Power Company | \n", "420966 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "34 | \n", "AccumulatedProvisionForDepreciationAmortizationAndDepletionOfPlantUtility | \n", "568,441,518.00 | \n", "USD | \n", "0 | \n", "\n", " |
| 4170 | \n", "2010 | \n", "Y | \n", "C000533 | \n", "Kentucky Power Company | \n", "420967 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "34 | \n", "AccumulatedProvisionForDepreciationAmortizationAndDepletionOfPlantUtility | \n", "568,441,518.00 | \n", "USD | \n", "0 | \n", "\n", " |
| 4171 | \n", "2010 | \n", "Y | \n", "C001111 | \n", "Baltimore Gas and Electric Company | \n", "418248 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "34 | \n", "AccumulatedProvisionForDepreciationAmortizationAndDepletionOfPlantUtility | \n", "2,659,775,623.00 | \n", "USD | \n", "0 | \n", "\n", " |
| 4172 | \n", "2010 | \n", "Y | \n", "C001111 | \n", "Baltimore Gas and Electric Company | \n", "418249 | \n", "200 - Schedule - Summary of Utility Plant and Accumulated Provisions for Depreciation, Amortization and Depletion | \n", "34 | \n", "AccumulatedProvisionForDepreciationAmortizationAndDepletionOfPlantUtility | \n", "2,659,775,623.00 | \n", "USD | \n", "0 | \n", "\n", " |