Import value from JSON in google sheets - json
I'm new to JSON and I found this custom script to import a JSON array in Google Sheets, but now I'm interested to filter out a certain value. How do I do this? I have no idea how to query this.
Example: I'm interested in the Pairs Priceusd value of this URL: https://api.dexscreener.io/latest/dex/tokens/0x333fd139caef6aa31056cc905987b77b1044d259.
Credits for the script go to: https://gist.github.com/paulgambill/cacd19da95a1421d3164
/**
* Retrieves all the rows in the active spreadsheet that contain data and logs the
* values for each row.
* For more information on using the Spreadsheet API, see
* https://developers.google.com/apps-script/service_spreadsheet
*/
function readRows() {
var sheet = SpreadsheetApp.getActiveSheet();
var rows = sheet.getDataRange();
var numRows = rows.getNumRows();
var values = rows.getValues();
for (var i = 0; i <= numRows - 1; i++) {
var row = values[i];
Logger.log(row);
}
};
/**
* Adds a custom menu to the active spreadsheet, containing a single menu item
* for invoking the readRows() function specified above.
* The onOpen() function, when defined, is automatically invoked whenever the
* spreadsheet is opened.
* For more information on using the Spreadsheet API, see
* https://developers.google.com/apps-script/service_spreadsheet
*/
function onOpen() {
var sheet = SpreadsheetApp.getActiveSpreadsheet();
var entries = [{
name : "Read Data",
functionName : "readRows"
}];
sheet.addMenu("Script Center Menu", entries);
};
/*====================================================================================================================================*
ImportJSON by Trevor Lohrbeer (#FastFedora)
====================================================================================================================================
Version: 1.1
Project Page: http://blog.fastfedora.com/projects/import-json
Copyright: (c) 2012 by Trevor Lohrbeer
License: GNU General Public License, version 3 (GPL-3.0)
http://www.opensource.org/licenses/gpl-3.0.html
------------------------------------------------------------------------------------------------------------------------------------
A library for importing JSON feeds into Google spreadsheets. Functions include:
ImportJSON For use by end users to import a JSON feed from a URL
ImportJSONAdvanced For use by script developers to easily extend the functionality of this library
Future enhancements may include:
- Support for a real XPath like syntax similar to ImportXML for the query parameter
- Support for OAuth authenticated APIs
Or feel free to write these and add on to the library yourself!
------------------------------------------------------------------------------------------------------------------------------------
Changelog:
1.1 Added support for the noHeaders option
1.0 Initial release
*====================================================================================================================================*/
/**
* Imports a JSON feed and returns the results to be inserted into a Google Spreadsheet. The JSON feed is flattened to create
* a two-dimensional array. The first row contains the headers, with each column header indicating the path to that data in
* the JSON feed. The remaining rows contain the data.
*
* By default, data gets transformed so it looks more like a normal data import. Specifically:
*
* - Data from parent JSON elements gets inherited to their child elements, so rows representing child elements contain the values
* of the rows representing their parent elements.
* - Values longer than 256 characters get truncated.
* - Headers have slashes converted to spaces, common prefixes removed and the resulting text converted to title case.
*
* To change this behavior, pass in one of these values in the options parameter:
*
* noInherit: Don't inherit values from parent elements
* noTruncate: Don't truncate values
* rawHeaders: Don't prettify headers
* noHeaders: Don't include headers, only the data
* debugLocation: Prepend each value with the row & column it belongs in
*
* For example:
*
* =ImportJSON("http://gdata.youtube.com/feeds/api/standardfeeds/most_popular?v=2&alt=json", "/feed/entry/title,/feed/entry/content",
* "noInherit,noTruncate,rawHeaders")
*
* #param {url} the URL to a public JSON feed
* #param {query} a comma-separated lists of paths to import. Any path starting with one of these paths gets imported.
* #param {options} a comma-separated list of options that alter processing of the data
*
* #return a two-dimensional array containing the data, with the first row containing headers
* #customfunction
**/
function ImportJSON(url, query, options) {
return ImportJSONAdvanced(url, query, options, includeXPath_, defaultTransform_);
}
/**
* An advanced version of ImportJSON designed to be easily extended by a script. This version cannot be called from within a
* spreadsheet.
*
* Imports a JSON feed and returns the results to be inserted into a Google Spreadsheet. The JSON feed is flattened to create
* a two-dimensional array. The first row contains the headers, with each column header indicating the path to that data in
* the JSON feed. The remaining rows contain the data.
*
* Use the include and transformation functions to determine what to include in the import and how to transform the data after it is
* imported.
*
* For example:
*
* =ImportJSON("http://gdata.youtube.com/feeds/api/standardfeeds/most_popular?v=2&alt=json",
* "/feed/entry",
* function (query, path) { return path.indexOf(query) == 0; },
* function (data, row, column) { data[row][column] = data[row][column].toString().substr(0, 100); } )
*
* In this example, the import function checks to see if the path to the data being imported starts with the query. The transform
* function takes the data and truncates it. For more robust versions of these functions, see the internal code of this library.
*
* #param {url} the URL to a public JSON feed
* #param {query} the query passed to the include function
* #param {options} a comma-separated list of options that may alter processing of the data
* #param {includeFunc} a function with the signature func(query, path, options) that returns true if the data element at the given path
* should be included or false otherwise.
* #param {transformFunc} a function with the signature func(data, row, column, options) where data is a 2-dimensional array of the data
* and row & column are the current row and column being processed. Any return value is ignored. Note that row 0
* contains the headers for the data, so test for row==0 to process headers only.
*
* #return a two-dimensional array containing the data, with the first row containing headers
**/
function ImportJSONAdvanced(url, query, options, includeFunc, transformFunc) {
var jsondata = UrlFetchApp.fetch(url);
var object = JSON.parse(jsondata.getContentText());
return parseJSONObject_(object, query, options, includeFunc, transformFunc);
}
/**
* Encodes the given value to use within a URL.
*
* #param {value} the value to be encoded
*
* #return the value encoded using URL percent-encoding
*/
function URLEncode(value) {
return encodeURIComponent(value.toString());
}
/**
* Parses a JSON object and returns a two-dimensional array containing the data of that object.
*/
function parseJSONObject_(object, query, options, includeFunc, transformFunc) {
var headers = new Array();
var data = new Array();
if (query && !Array.isArray(query) && query.toString().indexOf(",") != -1) {
query = query.toString().split(",");
}
if (options) {
options = options.toString().split(",");
}
parseData_(headers, data, "", 1, object, query, options, includeFunc);
parseHeaders_(headers, data);
transformData_(data, options, transformFunc);
return hasOption_(options, "noHeaders") ? (data.length > 1 ? data.slice(1) : new Array()) : data;
}
/**
* Parses the data contained within the given value and inserts it into the data two-dimensional array starting at the rowIndex.
* If the data is to be inserted into a new column, a new header is added to the headers array. The value can be an object,
* array or scalar value.
*
* If the value is an object, it's properties are iterated through and passed back into this function with the name of each
* property extending the path. For instance, if the object contains the property "entry" and the path passed in was "/feed",
* this function is called with the value of the entry property and the path "/feed/entry".
*
* If the value is an array containing other arrays or objects, each element in the array is passed into this function with
* the rowIndex incremeneted for each element.
*
* If the value is an array containing only scalar values, those values are joined together and inserted into the data array as
* a single value.
*
* If the value is a scalar, the value is inserted directly into the data array.
*/
function parseData_(headers, data, path, rowIndex, value, query, options, includeFunc) {
var dataInserted = false;
if (isObject_(value)) {
for (key in value) {
if (parseData_(headers, data, path + "/" + key, rowIndex, value[key], query, options, includeFunc)) {
dataInserted = true;
}
}
} else if (Array.isArray(value) && isObjectArray_(value)) {
for (var i = 0; i < value.length; i++) {
if (parseData_(headers, data, path, rowIndex, value[i], query, options, includeFunc)) {
dataInserted = true;
rowIndex++;
}
}
} else if (!includeFunc || includeFunc(query, path, options)) {
// Handle arrays containing only scalar values
if (Array.isArray(value)) {
value = value.join();
}
// Insert new row if one doesn't already exist
if (!data[rowIndex]) {
data[rowIndex] = new Array();
}
// Add a new header if one doesn't exist
if (!headers[path] && headers[path] != 0) {
headers[path] = Object.keys(headers).length;
}
// Insert the data
data[rowIndex][headers[path]] = value;
dataInserted = true;
}
return dataInserted;
}
/**
* Parses the headers array and inserts it into the first row of the data array.
*/
function parseHeaders_(headers, data) {
data[0] = new Array();
for (key in headers) {
data[0][headers[key]] = key;
}
}
/**
* Applies the transform function for each element in the data array, going through each column of each row.
*/
function transformData_(data, options, transformFunc) {
for (var i = 0; i < data.length; i++) {
for (var j = 0; j < data[i].length; j++) {
transformFunc(data, i, j, options);
}
}
}
/**
* Returns true if the given test value is an object; false otherwise.
*/
function isObject_(test) {
return Object.prototype.toString.call(test) === '[object Object]';
}
/**
* Returns true if the given test value is an array containing at least one object; false otherwise.
*/
function isObjectArray_(test) {
for (var i = 0; i < test.length; i++) {
if (isObject_(test[i])) {
return true;
}
}
return false;
}
/**
* Returns true if the given query applies to the given path.
*/
function includeXPath_(query, path, options) {
if (!query) {
return true;
} else if (Array.isArray(query)) {
for (var i = 0; i < query.length; i++) {
if (applyXPathRule_(query[i], path, options)) {
return true;
}
}
} else {
return applyXPathRule_(query, path, options);
}
return false;
};
/**
* Returns true if the rule applies to the given path.
*/
function applyXPathRule_(rule, path, options) {
return path.indexOf(rule) == 0;
}
/**
* By default, this function transforms the value at the given row & column so it looks more like a normal data import. Specifically:
*
* - Data from parent JSON elements gets inherited to their child elements, so rows representing child elements contain the values
* of the rows representing their parent elements.
* - Values longer than 256 characters get truncated.
* - Values in row 0 (headers) have slashes converted to spaces, common prefixes removed and the resulting text converted to title
* case.
*
* To change this behavior, pass in one of these values in the options parameter:
*
* noInherit: Don't inherit values from parent elements
* noTruncate: Don't truncate values
* rawHeaders: Don't prettify headers
* debugLocation: Prepend each value with the row & column it belongs in
*/
function defaultTransform_(data, row, column, options) {
if (!data[row][column]) {
if (row < 2 || hasOption_(options, "noInherit")) {
data[row][column] = "";
} else {
data[row][column] = data[row-1][column];
}
}
if (!hasOption_(options, "rawHeaders") && row == 0) {
if (column == 0 && data[row].length > 1) {
removeCommonPrefixes_(data, row);
}
data[row][column] = toTitleCase_(data[row][column].toString().replace(/[\/\_]/g, " "));
}
if (!hasOption_(options, "noTruncate") && data[row][column]) {
data[row][column] = data[row][column].toString().substr(0, 256);
}
if (hasOption_(options, "debugLocation")) {
data[row][column] = "[" + row + "," + column + "]" + data[row][column];
}
}
/**
* If all the values in the given row share the same prefix, remove that prefix.
*/
function removeCommonPrefixes_(data, row) {
var matchIndex = data[row][0].length;
for (var i = 1; i < data[row].length; i++) {
matchIndex = findEqualityEndpoint_(data[row][i-1], data[row][i], matchIndex);
if (matchIndex == 0) {
return;
}
}
for (var i = 0; i < data[row].length; i++) {
data[row][i] = data[row][i].substring(matchIndex, data[row][i].length);
}
}
/**
* Locates the index where the two strings values stop being equal, stopping automatically at the stopAt index.
*/
function findEqualityEndpoint_(string1, string2, stopAt) {
if (!string1 || !string2) {
return -1;
}
var maxEndpoint = Math.min(stopAt, string1.length, string2.length);
for (var i = 0; i < maxEndpoint; i++) {
if (string1.charAt(i) != string2.charAt(i)) {
return i;
}
}
return maxEndpoint;
}
/**
* Converts the text to title case.
*/
function toTitleCase_(text) {
if (text == null) {
return null;
}
return text.replace(/\w\S*/g, function(word) { return word.charAt(0).toUpperCase() + word.substr(1).toLowerCase(); });
}
/**
* Returns true if the given set of options contains the given option.
*/
function hasOption_(options, option) {
return options && options.indexOf(option) >= 0;
}```
You can retrieve the value by
function dexscreener(token) {
var url = 'https://api.dexscreener.io/latest/dex/tokens/' + token
var obj = JSON.parse(UrlFetchApp.fetch(url).getContentText())
return obj.pairs[0].priceUsd
}
put token in A1, and formula in B1 as follows
=dexscreener(A1)
Was able to make it work with this:
function DEXPRICE3 (url) {
var obj = JSON.parse(UrlFetchApp.fetch(url).getContentText())
return (obj.pairs[0].priceUsd)
}
Thanks for the help!
Related
Search through all sheets for a string which have a specific value in the cell next to it
I have a custom formula script that loops through all sheets containing "#" in their tab name. In there, I then look through a range to find the string that I am searching for (given as an argument). If I find that string in the range, I run some other code to check other cells. The important part here is finding the string in the range. Here is my code for doing that. /** * Searches through the box scores and counts Goaltender Wins. * * #param {"Hollywood|Hattricks|HOL"} teamName The team to search for. * #param {"Z1"} dummy The cell to store random number in order to refresh values on editing the box scores. * #return {number} The number of times the item appears in the range(s). * #customfunction */ function calcGoaltenderWins(username, dummy) { var count = 0; SpreadsheetApp.getActive().getSheets().forEach(function (s) { //regular season sheets if (s.getName().indexOf("#") >= 0 && s.getName().indexOf(":") === -1) { //if sheet is finalized if (s.getRange("U37").getDisplayValue() != "Live ⬤") { //if the home team won if (s.getRange("Q11").getValue() < s.getRange("Q12").getValue()) { //check if goaltender is on the player list for that game s.getRange("Y5:Y12") .getValues() .reduce(function (a, b) { return a.concat(b); }) .forEach(function (v) { if (v === username) count += 1; }); } //if the away team won if (s.getRange("Q12").getValue() < s.getRange("Q11").getValue()) { //check if goaltender is on the player list for that game s.getRange("D5:D12") .getValues() .reduce(function (a, b) { return a.concat(b); }) .forEach(function (v) { if (v === username) count += 1; }); } } } }); return count; } The thing I want to figure out is how to check if the string I am checking for has a specific value 2 cells to the left of it. This value should be "G". For example, I used the formula =calcGoaltenderWins("Name", dummy). If the name "Name" is found inside the range D5:D12 (specifically in D5), then it should also only check if B5 (the cell 2 to the left) if it equals "G" in order to continue running the code. If it does not equal "G", then it should not respond to the string it found.
Try this: function calcGoaltenderWins(username, dummy) { var count = 0; const ss=SpreadsheetApp.getActive(); const shts=ss.getSheets() shts.forEach(function(s) { if (s.getName().indexOf("#")>=0 && s.getName().indexOf(":")==-1) { if (s.getRange("U37").getDisplayValue()!="Live ⬤") { if (s.getRange("Q11").getValue() < s.getRange("Q12").getValue()) { s.getRange("Y5:Y12").getValues().reduce(function(a,b){return a.concat(b);}).forEach(function(v){if (v==username)count+=1;}); } if(s.getRange("Q12").getValue() < s.getRange("Q11").getValue()) { let v=s.getRange("B5:B12").getValues();//added s.getRange("D5:D12").getValues().reduce(function(a,b,i) {if(v[i][0]=="G"){return a.concat(b)}else{return a;};}).forEach(function(v){if (v==username)count += 1;//edited }); } } } }); return count; }
How to find out the correct link of the web table to copy with Urlfetch in GAS
I'm trying to get data from a table of a website. I was reading about the URL Fetch Service of Google App Scripts, I understand it, but cannot find what is the correct link to evaluate. I tried this: https://www.codelco.com/prontus_codelco/site/edic/base/port/licitaciones_enproceso.html This is the basic code: (when I find the correct link I will develop more) function getCodelcoTable() { var html = UrlFetchApp.fetch('https://www.codelco.com/prontus_codelco/site/edic/base/port/licitaciones_enproceso.html'); Logger.log(html.getContentText()); }
Code It seems there is what you need /** * Parses the first html-table from the content. max 7 columns * * #param {string} content A html string * #returns {Array.<Array.<object>>} */ function parseHtmlTableToArray_(content) { var inline = content.replace(/[\n\r]/g, ' '); var table = inline.split(/<table.*?>/)[1].split(/<\/table>/)[0]; return table.split(/<tr.*?>/).map(function(row) { return ( row .split(/(<td.*?>|<th.*?>)/) .slice(2) .reduce(function(p, cell, i) { if (~i % 2) { p.push( cell .replace(/<.*?>/g, ' ') .replace(/\s+/g, ' ') .replace(/^\s+/g, '') .replace(/\s+$/g, '') ); } return p; }, []) .join(PLACEHOLDER) + PLACEHOLDER.repeat(7) ) .split(PLACEHOLDER) .slice(0, 7); }); } You need to define PLACEHOLDER and pass your data to parseHtmlTableToArray_. Run For an example you can exec this like: var PLACEHOLDER = '----YADA-YADA----'; function userActionRun() { var content = UrlFetchApp.fetch( 'https://www.codelco.com/prontus_codelco/site/edic/base/port/licitaciones_enproceso.html' ).getContentText(); var data = parseHtmlTableToArray_(content); SpreadsheetApp.openById('XXX') .getSheets()[0] .clearContents() .getRange(1, 1, data.length, data[0].length) .setValues(data); } Result Notes Don't miss add the String.repeat polyfill Links The full snippet String.repeat The full project example
Need to total up selections
I am new to Google scripting and am trying to create an order form that totals on a new sheet. I found some tutorials that have gotten me this far. https://docs.google.com/spreadsheets/d/1gABaW1qaJyaL1k4QSh0nqkJ7uCCyc8NmL02wcn1klpA/edit?usp=sharing The current script allows styles and color/size options to be populated on the Ranges sheet that then is auto-populated on the Order Form sheet based on their selection. Since the styles and color/size options can change I need a script to create the Order Totals sheet which will contain the totals for each Style, Color/Size option that is populated on the Order Form sheet. With the sample data that I have in the Order form the Order Totals would look like: Style, Color/Size, Quantity = Adult Unisex, Grey - Small, 3 - Youth, Grey - Large, 1 - Youth, Grey - X-Large, 3 - Adult Unisex, Grey - Medium, 1 -
You can probably do this somehow with the Sheets built-in functions, but here is an Apps Script solution that is working for me on the sheet you provided. This is a general solution for n columns of item descriptors before a final quantity column. /** * #param {string} A1RangeString - e.g. "A1:C" * #returns {string[][]} suitable for Range.setValues */ function getOrderTotals(A1RangeString) { var orders = SpreadsheetApp.getActive().getSheetByName("Order Form") .getRange(A1RangeString).getValues().filter(function (row) { return row[0]; // i.e. it's not blank }); var totals = orders.reduce(readOrders, {}); return writeTotals(totals); } /** * #param {obj} orderBook - nested dictionary, see readOrders * #returns {string[][]} for use with Range.setValues */ function writeTotals(orderBook) { if (typeof orderBook === "number") { return [orderBook]; } var items = []; Object.keys(orderBook).forEach(function(key) { var result = writeTotals(orderBook[key]); result.forEach(function (description) { items.push([key].concat(description)); }); }); return items; } /** * readOrders is a callback for Array.prototype.reduce * #param {obj} orderBook - a nested dictionary * #param {row} string[] */ function readOrders(orderBook, row) { var item = orderBook; for (var i = 0; i < row.length - 2; ++i) { if (! item[row[i]]) { item[row[i]] = {}; } item = item[row[i]]; } if (! item[row[row.length - 2]]) { item[row[row.length - 2]] = row[row.length - 1]; } else { item[row[row.length - 2]] += row[row.length - 1]; } return orderBook; } function onEdit() { var orders = getOrderTotals("C3:E"); SpreadsheetApp.getActive().getSheetByName("Order Totals") .getRange(1, 1, orders.length, orders[0].length) .setValues(orders); // other onEdit stuff continues below }
pull json api to google sheets no oauth
I'm attempting to pull data from an API we are using and put it into Google Sheets. I've gotten this code from FastFedora (see code below), but that stuff currently isn't working or passing the credentials through unfortunately. The way that i'm calling this currently in Google Sheets is by utilizing this command: =ImportJSONBasicAuthentication("https://WEBSITE.com/builds/ids?include_shares=true&app_id=*","noInherit,noTruncate,rawHeaders") Within the script.google, but it's returning an error of "Code 401". Any help is appreciated. I'm happy to answer any questions as well. /** * Retrieves all the rows in the active spreadsheet that contain data and logs the * values for each row. * For more information on using the Spreadsheet API, see * https://developers.google.com/apps-script/service_spreadsheet */ function readRows() { var sheet = SpreadsheetApp.getActiveSheet(); var rows = sheet.getDataRange(); var numRows = rows.getNumRows(); var values = rows.getValues(); for (var i = 0; i <= numRows - 1; i++) { var row = values[i]; Logger.log(row); } }; /** * Adds a custom menu to the active spreadsheet, containing a single menu item * for invoking the readRows() function specified above. * The onOpen() function, when defined, is automatically invoked whenever the * spreadsheet is opened. * For more information on using the Spreadsheet API, see * https://developers.google.com/apps-script/service_spreadsheet */ function onOpen() { var sheet = SpreadsheetApp.getActiveSpreadsheet(); var entries = [{ name : "Read Data", functionName : "readRows" }]; sheet.addMenu("Script Center Menu", entries); }; /*====================================================================================================================================* ImportJSON by Trevor Lohrbeer (#FastFedora) ==================================================================================================================================== Version: 1.1 Project Page: http://blog.fastfedora.com/projects/import-json Copyright: (c) 2012 by Trevor Lohrbeer License: GNU General Public License, version 3 (GPL-3.0) http://www.opensource.org/licenses/gpl-3.0.html ------------------------------------------------------------------------------------------------------------------------------------ A library for importing JSON feeds into Google spreadsheets. Functions include: ImportJSON For use by end users to import a JSON feed from a URL ImportJSONAdvanced For use by script developers to easily extend the functionality of this library Future enhancements may include: - Support for a real XPath like syntax similar to ImportXML for the query parameter - Support for OAuth authenticated APIs Or feel free to write these and add on to the library yourself! ------------------------------------------------------------------------------------------------------------------------------------ Changelog: 1.1 Added support for the noHeaders option 1.0 Initial release *====================================================================================================================================*/ /** * Imports a JSON feed and returns the results to be inserted into a Google Spreadsheet. The JSON feed is flattened to create * a two-dimensional array. The first row contains the headers, with each column header indicating the path to that data in * the JSON feed. The remaining rows contain the data. * * By default, data gets transformed so it looks more like a normal data import. Specifically: * * - Data from parent JSON elements gets inherited to their child elements, so rows representing child elements contain the values * of the rows representing their parent elements. * - Values longer than 256 characters get truncated. * - Headers have slashes converted to spaces, common prefixes removed and the resulting text converted to title case. * * To change this behavior, pass in one of these values in the options parameter: * * noInherit: Don't inherit values from parent elements * noTruncate: Don't truncate values * rawHeaders: Don't prettify headers * noHeaders: Don't include headers, only the data * debugLocation: Prepend each value with the row & column it belongs in * * For example: * * =ImportJSON("http://gdata.youtube.com/feeds/api/standardfeeds/most_popular?v=2&alt=json", "/feed/entry/title,/feed/entry/content", * "noInherit,noTruncate,rawHeaders") * * #param {url} the URL to a public JSON feed * #param {query} a comma-separated lists of paths to import. Any path starting with one of these paths gets imported. * #param {options} a comma-separated list of options that alter processing of the data * * #return a two-dimensional array containing the data, with the first row containing headers * #customfunction **/ function ImportJSON(url, query, options) { return ImportJSONAdvanced(url, query, options, includeXPath_, defaultTransform_); } /** * An advanced version of ImportJSON designed to be easily extended by a script. This version cannot be called from within a * spreadsheet. * * Imports a JSON feed and returns the results to be inserted into a Google Spreadsheet. The JSON feed is flattened to create * a two-dimensional array. The first row contains the headers, with each column header indicating the path to that data in * the JSON feed. The remaining rows contain the data. * * Use the include and transformation functions to determine what to include in the import and how to transform the data after it is * imported. * * For example: * * =ImportJSON("http://gdata.youtube.com/feeds/api/standardfeeds/most_popular?v=2&alt=json", * "/feed/entry", * function (query, path) { return path.indexOf(query) == 0; }, * function (data, row, column) { data[row][column] = data[row][column].toString().substr(0, 100); } ) * * In this example, the import function checks to see if the path to the data being imported starts with the query. The transform * function takes the data and truncates it. For more robust versions of these functions, see the internal code of this library. * * #param {url} the URL to a public JSON feed * #param {query} the query passed to the include function * #param {options} a comma-separated list of options that may alter processing of the data * #param {includeFunc} a function with the signature func(query, path, options) that returns true if the data element at the given path * should be included or false otherwise. * #param {transformFunc} a function with the signature func(data, row, column, options) where data is a 2-dimensional array of the data * and row & column are the current row and column being processed. Any return value is ignored. Note that row 0 * contains the headers for the data, so test for row==0 to process headers only. * * #return a two-dimensional array containing the data, with the first row containing headers **/ function ImportJSONAdvanced(url, query, options, includeFunc, transformFunc) { var jsondata = UrlFetchApp.fetch(url); var object = JSON.parse(jsondata.getContentText()); return parseJSONObject_(object, query, options, includeFunc, transformFunc); } /** * Encodes the given value to use within a URL. * * #param {value} the value to be encoded * * #return the value encoded using URL percent-encoding */ function URLEncode(value) { return encodeURIComponent(value.toString()); } /** * Parses a JSON object and returns a two-dimensional array containing the data of that object. */ function parseJSONObject_(object, query, options, includeFunc, transformFunc) { var headers = new Array(); var data = new Array(); if (query && !Array.isArray(query) && query.toString().indexOf(",") != -1) { query = query.toString().split(","); } if (options) { options = options.toString().split(","); } parseData_(headers, data, "", 1, object, query, options, includeFunc); parseHeaders_(headers, data); transformData_(data, options, transformFunc); return hasOption_(options, "noHeaders") ? (data.length > 1 ? data.slice(1) : new Array()) : data; } /** * Parses the data contained within the given value and inserts it into the data two-dimensional array starting at the rowIndex. * If the data is to be inserted into a new column, a new header is added to the headers array. The value can be an object, * array or scalar value. * * If the value is an object, it's properties are iterated through and passed back into this function with the name of each * property extending the path. For instance, if the object contains the property "entry" and the path passed in was "/feed", * this function is called with the value of the entry property and the path "/feed/entry". * * If the value is an array containing other arrays or objects, each element in the array is passed into this function with * the rowIndex incremeneted for each element. * * If the value is an array containing only scalar values, those values are joined together and inserted into the data array as * a single value. * * If the value is a scalar, the value is inserted directly into the data array. */ function parseData_(headers, data, path, rowIndex, value, query, options, includeFunc) { var dataInserted = false; if (isObject_(value)) { for (key in value) { if (parseData_(headers, data, path + "/" + key, rowIndex, value[key], query, options, includeFunc)) { dataInserted = true; } } } else if (Array.isArray(value) && isObjectArray_(value)) { for (var i = 0; i < value.length; i++) { if (parseData_(headers, data, path, rowIndex, value[i], query, options, includeFunc)) { dataInserted = true; rowIndex++; } } } else if (!includeFunc || includeFunc(query, path, options)) { // Handle arrays containing only scalar values if (Array.isArray(value)) { value = value.join(); } // Insert new row if one doesn't already exist if (!data[rowIndex]) { data[rowIndex] = new Array(); } // Add a new header if one doesn't exist if (!headers[path] && headers[path] != 0) { headers[path] = Object.keys(headers).length; } // Insert the data data[rowIndex][headers[path]] = value; dataInserted = true; } return dataInserted; } /** * Parses the headers array and inserts it into the first row of the data array. */ function parseHeaders_(headers, data) { data[0] = new Array(); for (key in headers) { data[0][headers[key]] = key; } } /** * Applies the transform function for each element in the data array, going through each column of each row. */ function transformData_(data, options, transformFunc) { for (var i = 0; i < data.length; i++) { for (var j = 0; j < data[i].length; j++) { transformFunc(data, i, j, options); } } } /** * Returns true if the given test value is an object; false otherwise. */ function isObject_(test) { return Object.prototype.toString.call(test) === '[object Object]'; } /** * Returns true if the given test value is an array containing at least one object; false otherwise. */ function isObjectArray_(test) { for (var i = 0; i < test.length; i++) { if (isObject_(test[i])) { return true; } } return false; } /** * Returns true if the given query applies to the given path. */ function includeXPath_(query, path, options) { if (!query) { return true; } else if (Array.isArray(query)) { for (var i = 0; i < query.length; i++) { if (applyXPathRule_(query[i], path, options)) { return true; } } } else { return applyXPathRule_(query, path, options); } return false; }; /** * Returns true if the rule applies to the given path. */ function applyXPathRule_(rule, path, options) { return path.indexOf(rule) == 0; } /** * By default, this function transforms the value at the given row & column so it looks more like a normal data import. Specifically: * * - Data from parent JSON elements gets inherited to their child elements, so rows representing child elements contain the values * of the rows representing their parent elements. * - Values longer than 256 characters get truncated. * - Values in row 0 (headers) have slashes converted to spaces, common prefixes removed and the resulting text converted to title * case. * * To change this behavior, pass in one of these values in the options parameter: * * noInherit: Don't inherit values from parent elements * noTruncate: Don't truncate values * rawHeaders: Don't prettify headers * debugLocation: Prepend each value with the row & column it belongs in */ function defaultTransform_(data, row, column, options) { if (!data[row][column]) { if (row < 2 || hasOption_(options, "noInherit")) { data[row][column] = ""; } else { data[row][column] = data[row-1][column]; } } if (!hasOption_(options, "rawHeaders") && row == 0) { if (column == 0 && data[row].length > 1) { removeCommonPrefixes_(data, row); } data[row][column] = toTitleCase_(data[row][column].toString().replace(/[\/\_]/g, " ")); } if (!hasOption_(options, "noTruncate") && data[row][column]) { data[row][column] = data[row][column].toString().substr(0, 256); } if (hasOption_(options, "debugLocation")) { data[row][column] = "[" + row + "," + column + "]" + data[row][column]; } } /** * If all the values in the given row share the same prefix, remove that prefix. */ function removeCommonPrefixes_(data, row) { var matchIndex = data[row][0].length; for (var i = 1; i < data[row].length; i++) { matchIndex = findEqualityEndpoint_(data[row][i-1], data[row][i], matchIndex); if (matchIndex == 0) { return; } } for (var i = 0; i < data[row].length; i++) { data[row][i] = data[row][i].substring(matchIndex, data[row][i].length); } } /** * Locates the index where the two strings values stop being equal, stopping automatically at the stopAt index. */ function findEqualityEndpoint_(string1, string2, stopAt) { if (!string1 || !string2) { return -1; } var maxEndpoint = Math.min(stopAt, string1.length, string2.length); for (var i = 0; i < maxEndpoint; i++) { if (string1.charAt(i) != string2.charAt(i)) { return i; } } return maxEndpoint; } /** * Converts the text to title case. */ function toTitleCase_(text) { if (text == null) { return null; } return text.replace(/\w\S*/g, function(word) { return word.charAt(0).toUpperCase() + word.substr(1).toLowerCase(); }); } /** * Returns true if the given set of options contains the given option. */ function hasOption_(options, option) { return options && options.indexOf(option) >= 0; } function ImportJSONBasicAuthentication(url, query, parseOptions, username, password) { var fetchOptions = { headers : { 'authorization': "Bearer THIS_IS_MY_TOKEN_NUMBER", 'accept': "application/json" }, muteHttpExceptions: true } return ImportJSONAdvanced(url, fetchOptions, query, parseOptions, includeXPath_, defaultTransform_); }
I know its a little bit late, but maybe I can help some one else. You have to add a parameter here: function ImportJSONAdvanced(url, header, query, options, includeFunc, transformFunc) //line 126 Here I add the parameter "header" to get "fetchOptions" from your function ImportJSONBasicAuthentication. After that you have to add this value(header) to this: var jsondata = UrlFetchApp.fetch(url,header) //line 127 after that the code should be work fine and no 401 should be come there
Google spreadsheets custom function code not working: =RootDomain
Hi I am trying to program a Google spreadsheets custom function. It receives a range and spits out the clean rootdomain. I just ran into an "too many executions" - I have to run this on my whole sheet. So I added a range. Now the feedback is "internal error function" .... Help appreciated .... this must be possible! /** * Generates clean root domains * * #param {input} input The value to change to a root domain. * #return The clean root domain. * #RootDomain */ function RootDomain(input) { if (input == null) return ''; if (input.map) { // Test whether input is an array. return input.map(RootDomain); // Recurse over array if so. } else { if (input = '') return ''; regex = new RegExp(/((www)\.)?.*(\w+)\.([\w\.]{2,6})/); return regex.exec(input)[0].replace(/^http(s)?:\/\//i, "").replace(/^www\./i, "").replace(/\/.*$/, ""); } }
Do this instead: function RootDomain(input) { if (input == null || input === '') { return ''; } else if (input.map) { // Test whether input is an array. return input.map(RootDomain); // Recurse over array if so. } var regex = new RegExp(/((www)\.)?.*(\w+)\.([\w\.]{2,6})/); return regex.exec(input)[0].replace(/^http(s)?:\/\//i, "").replace(/^www\./i, "").replace(/\/.*$/, ""); }