Skip to content

Instantly share code, notes, and snippets.

@AbhiOnGithub
Last active August 22, 2018 15:22
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save AbhiOnGithub/841fa8990f63e1e415954877ffd2b5f9 to your computer and use it in GitHub Desktop.
Save AbhiOnGithub/841fa8990f63e1e415954877ffd2b5f9 to your computer and use it in GitHub Desktop.
Code to Showcase Bing Search APIs - Reference Microsoft Documentation
<!DOCTYPE html>
<!-- saved from url=(0014)about:internet -->
<!-- the above Mark of the Web lets IE run this page in the Internet security zone,
avoiding the permission prompt for running active content such as JavaScript -->
<html>
<head>
<meta charset="UTF-8">
<title>Bing Web Search API Demo</title>
<base target="_blank">
<style type="text/css">
html, body, div, p, h1, h2 {font-family: Verdana, "Lucida Sans", sans-serif; color: #000;}
html, body, div, p {font-weight: normal;}
body {background-color: #fff;}
h1, h2, h3 {font-weight: bold; color: #087;}
sup {font-weight: normal;}
html, body, div, p {font-size: 12px;}
h1 {font-size: 20px; margin-top: 30px;}
h2 {font-size: 16px; clear: left;}
h3 {font-size: 14px; clear: left;}
#sidebar {font-size: 10px; text-align: right; display: flex; padding: 0px 10px;
float: right; margin-left: 15px; margin-right: 0px;}
#sidebar p {margin-top: 0px; font-size: 10px;}
#sidebar img {display: inline-block; float: none; padding-right: 0px;}
#pole p {font-size: 14px;}
#pole, #mainline, #json, #http, #sidebar, #error, #paging1, #paging2
{display: none;}
#term {width: 100%;}
#logo {padding: 15px; float: right; border-left: 2px solid #ccc;}
#query {float: left;}
img {vertical-align: top; float: left; margin-right: 10px; margin-bottom: 10px;}
p.images img {float: none;}
p.relatedSearches {clear: none;}
#logo p, p.news, p.webPages, p.images {clear: left;}
a[href="#"]:link {color: blue;}
a[href="#"]:visited {color: blue;}
h3 a:visited {color: #087 !important;}
h3 a:link {color: #087 !important;}
</style>
<script type="text/javascript">
// cookie names for data we store
// YOUR API KEY DOES NOT GO IN THIS CODE; don't paste it in.
API_KEY_COOKIE = "bing-search-api-key";
CLIENT_ID_COOKIE = "bing-search-client-id";
// Various browsers differ in their support for persistent storage by local
// HTML files (IE won't use localStorage, but Chrome won't use cookies). So
// use localStorage if we can, otherwise use cookies.
try {
localStorage.getItem; // try localStorage
window.retrieveValue = function (name) {
return localStorage.getItem(name) || "";
}
window.storeValue = function(name, value) {
localStorage.setItem(name, value);
}
} catch (e) {
window.retrieveValue = function (name) {
var cookies = document.cookie.split(";");
for (var i = 0; i < cookies.length; i++) {
var keyvalue = cookies[i].split("=");
if (keyvalue[0].trim() === name) return keyvalue[1];
}
return "";
}
window.storeValue = function (name, value) {
var expiry = new Date();
expiry.setFullYear(expiry.getFullYear() + 1);
document.cookie = name + "=" + value.trim() + "; expires=" + expiry.toUTCString();
}
}
// get stored API subscription key, or prompt if it's not found
function getSubscriptionKey() {
var key = retrieveValue(API_KEY_COOKIE);
while (key.length !== 32) {
key = prompt("Enter Bing Search API subscription key:", "").trim();
}
// always set the cookie in order to update the expiration date
storeValue(API_KEY_COOKIE, key);
return key;
}
// invalidate stored API subscription key so user will be prompted again
function invalidateSubscriptionKey() {
storeValue(API_KEY_COOKIE, "");
}
// escape quotes to HTML entities for use in HTML tag attributes
function escapeQuotes(text) {
return text.replace(/'/g, "&apos;").replace(/"/g, "&quot;");
}
// get the host portion of a URL, strpping out search result formatting and www too
function getHost(url) {
return url.replace(/<\/?b>/g, "").replace(/^https?:\/\//, "").split("/")[0].replace(/^www\./, "");
}
// format plain text for display as an HTML <pre> element
function preFormat(text) {
text = "" + text;
return "<pre>" + text.replace(/&/g, "&amp;").replace(/</g, "&lt;") + "</pre>"
}
// put HTML markup into a <div> and reveal it
function showDiv(id, html) {
var content = document.getElementById("_" + id)
if (content) content.innerHTML = html;
var wrapper = document.getElementById(id);
if (wrapper) wrapper.style.display = html.trim() ? "block" : "none";
}
// hides the specified <div>s
function hideDivs() {
for (var i = 0; i < arguments.length; i++) {
var element = document.getElementById(arguments[i])
if (element) element.style.display = "none";
}
}
// render functions for various types of search results
searchItemRenderers = {
// render Web page result
webPages: function(item) {
var html = [];
html.push("<p class='webPages'><a href='" + item.url + "'>" + item.name + "</a>");
html.push(" (" + getHost(item.displayUrl) + ")");
html.push("<br>" + item.snippet);
if ("deepLinks" in item) {
var links = [];
for (var i = 0; i < item.deepLinks.length; i++) {
links.push("<a href='" + item.deepLinks[i].url + "'>" +
item.deepLinks[i].name + "</a>");
}
html.push("<br>" + links.join(" - "));
}
return html.join("");
},
// render news story
news: function(item) {
var html = [];
html.push("<p class='news'>");
if (item.image) {
width = 60;
height = Math.round(width * item.image.thumbnail.height / item.image.thumbnail.width);
html.push("<img src='" + item.image.thumbnail.contentUrl +
"&h=" + height + "&w=" + width + "' width=" + width + " height=" + height+ ">");
}
html.push("<a href='" + item.url + "'>" + item.name + "</a>");
if (item.category) html.push(" - " + item.category);
if (item.contractualRules) { // MUST display source attributions
html.push(" (");
var rules = [];
for (var i = 0; i < item.contractualRules.length; i++)
rules.push(item.contractualRules[i].text);
html.push(rules.join(", "));
html.push(")");
}
html.push(" (" + getHost(item.url) + ")");
html.push("<br>" + item.description);
return html.join("");
},
// render image result using thumbnail
images: function(item, section, index, count) {
var height = 60;
var width = Math.round(height * item.thumbnail.width / item.thumbnail.height);
var html = [];
if (section === "sidebar") {
if (index) html.push("<br>");
} else {
if (!index) html.push("<p class='images'>");
}
html.push("<a href='" + item.hostPageUrl + "'>");
var title = escapeQuotes(item.name) + "\n" + getHost(item.hostPageDisplayUrl);
html.push("<img src='"+ item.thumbnailUrl + "&h=" + height + "&w=" + width +
"' height=" + height + " width=" + width + " title='" + title + "' alt='" + title + "'>");
html.push("</a>");
return html.join("");
},
// render video result using thumbnail
videos: function(item, section, index, count) {
// videos are rendered like images
return searchItemRenderers.images(item, section, index, count);
},
relatedSearches: function(item, section, index, count) {
var html = [];
if (section !== "sidebar") html.push(index === 0 ? "<h2>Related</h2>": " - ");
else html.push("<p class='relatedSearches'>");
html.push("<a href='#' onclick='return doRelatedSearch(&quot;" +
escapeQuotes(item.text) + "&quot;)'>");
html.push(item.displayText + "</a>");
return html.join("");
}
}
// render search results from rankingResponse object in specified order
function renderResultsItems(section, results) {
var items = results.rankingResponse[section].items;
var html = [];
for (var i = 0; i < items.length; i++) {
var item = items[i];
// collection name has lowercase first letter while answerType has uppercase
// e.g. `WebPages` rankingResult type is in the `webPages` top-level collection
var type = item.answerType[0].toLowerCase() + item.answerType.slice(1);
// must have results of the given type AND a renderer for it
if (type in results && type in searchItemRenderers) {
var render = searchItemRenderers[type];
// this ranking item refers to ONE result of the specified type
if ("resultIndex" in item) {
html.push(render(results[type].value[item.resultIndex], section));
// this ranking item refers to ALL results of the specified type
} else {
var len = results[type].value.length;
for (var j = 0; j < len; j++) {
html.push(render(results[type].value[j], section, j, len));
}
}
}
}
return html.join("\n\n");
}
// render the search results given the parsed JSON response
function renderSearchResults(results) {
// if spelling was corrected, update search field
if (results.queryContext.alteredQuery)
document.forms.bing.query.value = results.queryContext.alteredQuery;
// add Prev / Next links with result count
var pagingLinks = renderPagingLinks(results);
showDiv("paging1", pagingLinks);
showDiv("paging2", pagingLinks);
// for each possible section, render the resuts from that section
for (section in {pole: 0, mainline: 0, sidebar: 0}) {
if (results.rankingResponse[section])
showDiv(section, renderResultsItems(section, results));
}
}
function renderErrorMessage(message) {
showDiv("error", preFormat(message));
showDiv("noresults", "No results.");
}
// handle Bing search request results
function handleOnLoad() {
hideDivs("noresults");
var json = this.responseText.trim();
var jsobj = {};
// try to parse JSON results
try {
if (json.length) jsobj = JSON.parse(json);
} catch(e) {
renderErrorMessage("Invalid JSON response");
}
// show raw JSON and headers
showDiv("json", preFormat(JSON.stringify(jsobj, null, 2)));
showDiv("http", preFormat("GET " + this.responseURL + "\n\nStatus: " + this.status + " " +
this.statusText + "\n" + this.getAllResponseHeaders()));
// if HTTP response is 200 OK, try to render search results
if (this.status === 200) {
var clientid = this.getResponseHeader("X-MSEdge-ClientID");
if (clientid) retrieveValue(CLIENT_ID_COOKIE, clientid);
if (json.length) {
if (jsobj._type === "SearchResponse" && "rankingResponse" in jsobj) {
renderSearchResults(jsobj);
} else {
renderErrorMessage("No search results in JSON response");
}
} else {
renderErrorMessage("Empty response (are you sending too many requests too quickly?)");
}
}
// Any other HTTP response is an error
else {
// 401 is unauthorized; force re-prompt for API key for next request
if (this.status === 401) invalidateSubscriptionKey();
// some error responses don't have a top-level errors object, so gin one up
var errors = jsobj.errors || [jsobj];
var errmsg = [];
// display HTTP status code
errmsg.push("HTTP Status " + this.status + " " + this.statusText + "\n");
// add all fields from all error responses
for (var i = 0; i < errors.length; i++) {
if (i) errmsg.push("\n");
for (var k in errors[i]) errmsg.push(k + ": " + errors[i][k]);
}
// also display Bing Trace ID if it isn't blocked by CORS
var traceid = this.getResponseHeader("BingAPIs-TraceId");
if (traceid) errmsg.push("\nTrace ID " + traceid);
// and display the error message
renderErrorMessage(errmsg.join("\n"));
}
}
function bingWebSearch(query, options, key) {
// scroll to top of window
window.scrollTo(0, 0);
if (!query.trim().length) return false; // empty query, do nothing
showDiv("noresults", "Working. Please wait.");
hideDivs("pole", "mainline", "sidebar", "_json", "_headers", "paging1", "paging2", "error");
var endpoint = "https://api.cognitive.microsoft.com/bing/v7.0/search";
var request = new XMLHttpRequest();
var queryurl = endpoint + "?q=" + encodeURIComponent(query) + "&" + options;
//
try {
request.open("GET", queryurl);
}
catch (e) {
renderErrorMessage("Bad request (invalid URL)\n" + queryurl);
return false;
}
// add request headers
request.setRequestHeader("Ocp-Apim-Subscription-Key", key);
request.setRequestHeader("Accept", "application/json");
var clientid = retrieveValue(CLIENT_ID_COOKIE);
if (clientid) request.setRequestHeader("X-MSEdge-ClientID", clientid);
// event handler for successful response
request.addEventListener("load", handleOnLoad);
// event handler for erorrs
request.addEventListener("error", function() {
renderErrorMessage("Error completing request");
});
// event handler for aborted request
request.addEventListener("abort", function() {
renderErrorMessage("Request aborted");
});
// send the request
request.send();
return false;
}
// build query options from the HTML form
function bingSearchOptions(form) {
var options = [];
options.push("mkt=" + form.where.value);
options.push("SafeSearch=" + (form.safe.checked ? "strict" : "off"));
if (form.when.value.length) options.push("freshness=" + form.when.value);
var what = [];
for (var i = 0; i < form.what.length; i++)
if (form.what[i].checked) what.push(form.what[i].value);
if (what.length) {
options.push("promote=" + what.join(","));
options.push("answerCount=9");
}
options.push("count=" + form.count.value);
options.push("offset=" + form.offset.value);
options.push("textDecorations=true");
options.push("textFormat=HTML");
return options.join("&");
}
// toggle display of a div (used by JSON/HTTP expandos)
function toggleDisplay(id) {
var element = document.getElementById(id);
if (element) {
var display = element.style.display;
if (display === "none") {
element.style.display = "block";
window.scrollBy(0, 200);
} else {
element.style.display = "none";
}
}
return false;
}
// perform a related search (used by related search links)
function doRelatedSearch(query) {
var bing = document.forms.bing;
bing.query.value = query;
bing.offset.value = 0;
return bingWebSearch(query, bingSearchOptions(bing), getSubscriptionKey());
}
// generate the HTML for paging links (prev/next)
function renderPagingLinks(results) {
var html = [];
var bing = document.forms.bing;
var offset = parseInt(bing.offset.value, 10);
var count = parseInt(bing.count.value, 10);
html.push("<p class='paging'><i>Results " + (offset + 1) + " to " + (offset + count));
html.push(" of about " + results.webPages.totalEstimatedMatches + ".</i> ");
html.push("<a href='#' onclick='return doPrevSearchPage()'>Prev</a> | ");
html.push("<a href='#' onclick='return doNextSearchPage()'>Next</a>");
return html.join("");
}
// go to the next page (used by next page link)
function doNextSearchPage() {
var bing = document.forms.bing;
var query = bing.query.value;
var offset = parseInt(bing.offset.value, 10);
var count = parseInt(bing.count.value, 10);
offset += count;
bing.offset.value = offset;
return bingWebSearch(query, bingSearchOptions(bing), getSubscriptionKey());
}
// go to the previous page (used by previous page link)
function doPrevSearchPage() {
var bing = document.forms.bing;
var query = bing.query.value;
var offset = parseInt(bing.offset.value, 10);
var count = parseInt(bing.count.value, 10);
if (offset) {
offset -= count;
if (offset < 0) offset = 0;
bing.offset.value = offset;
return bingWebSearch(query, bingSearchOptions(bing), getSubscriptionKey());
}
alert("You're already at the beginning!");
return false;
}
// --></script>
</head>
<body onload="document.forms.bing.query.focus();">
<form name="bing" onsubmit="this.offset.value = 0; return bingWebSearch(this.query.value,
bingSearchOptions(this), getSubscriptionKey())">
<div id="logo"><!-- logo block including search market/language -->
<img src="">
<I>api</I>
<p><select name="where">
<option value="es-AR">Argentina (Spanish)</option>
<option value="en-AU">Australia (English)</option>
<option value="de-AT">Austria (German)</option>
<option value="nl-BE">Belgium (Dutch)</option>
<option value="fr-BE">Belgium (French)</option>
<option value="pt-BR">Brazil (Portuguese)</option>
<option value="en-CA">Canada (English)</option>
<option value="fr-CA">Canada (French)</option>
<option value="es-CL">Chile (Spanish)</option>
<option value="da-DK">Denmark (Danish)</option>
<option value="fi-FI">Finland (Finnish)</option>
<option value="fr-FR">France (French)</option>
<option value="de-DE">Germany (German)</option>
<option value="zh-HK">Hong Kong (Traditional Chinese)</option>
<option value="en-IN">India (English)</option>
<option value="en-ID">Indonesia (English)</option>
<option value="it-IT">Italy (Italian)</option>
<option value="ja-JP">Japan (Japanese)</option>
<option value="ko-KR">Korea (Korean)</option>
<option value="en-MY">Malaysia (English)</option>
<option value="es-MX">Mexico (Spanish)</option>
<option value="nl-NL">Netherlands (Dutch)</option>
<option value="en-NZ">New Zealand (English)</option>
<option value="no-NO">Norway (Norwegian)</option>
<option value="zh-CN">People's Republic of China (Chinese)</option>
<option value="pl-PL">Poland (Polish)</option>
<option value="pt-PT">Portugal (Portuguese)</option>
<option value="en-PH">Philippines (English)</option>
<option value="ru-RU">Russia (Russian)</option>
<option value="ar-SA">Saudi Arabia (Arabic)</option>
<option value="en-ZA">South Africa (English)</option>
<option value="es-ES">Spain (Spanish)</option>
<option value="sv-SE">Sweden (Swedish)</option>
<option value="fr-CH">Switzerland (French)</option>
<option value="de-CH">Switzerland (German)</option>
<option value="zh-TW">Taiwan (Traditional Chinese)</option>
<option value="tr-TR">Turkey (Turkish)</option>
<option value="en-GB">United Kingdom (English)</option>
<option value="en-US" selected>United States (English)</option>
<option value="es-US">United States (Spanish)</option>
</select>
<p>from Microsoft Cognitive Services
</div>
<div id="query"><!-- query controls including search field and options -->
<h1>Bing Web Search API demo</h2>
<input type="text" name="query" id="term" placeholder="Search the Web" autocomplete=off>
<p>Promote
<input type=checkbox name="what" id="webpages" value="webpages">
<label for="webpages">Web pages</label>
<input type=checkbox name="what" id="news" value="news">
<label for="news">News</label>
<input type=checkbox name="what" id="images" value="images">
<label for="images">Images</label>
<input type=checkbox name="what" id="videos" value="videos">
<label for="videos">Videos</label>
&nbsp;&nbsp;&nbsp;From
<select name="when">
<option value="" selected>All time</option>
<option value="month">Past month</option>
<option value="week">Past week</option>
<option value="day">Last 24 hours</option>
</select>
&nbsp;&nbsp;&nbsp;<input type=checkbox id="safe" name="safe" value="on" checked><label for="safe">SafeSearch</label>
<!-- these hidden fields control paging -->
<input type=hidden name="count" value="25">
<input type=hidden name="offset" value="0">
</form>
</div>
<div id="error">
<h2>Error</h2>
<div id="_error">
</div>
</div>
<h2>Results</h2>
<div id="paging1">
<div id="_paging1"></div>
</div>
<div id="noresults">
<div id="_noresults">None yet.</div>
</div>
<div id="sidebar">
<div id="_sidebar"></div>
</div>
<div id="pole">
<div id="_pole"></div>
</div>
<div id="mainline">
<div id="_mainline"></div>
</div>
<div id="paging2">
<div id="_paging2"></div>
</div>
<div id="json">
<h3><a href="#" onclick="return toggleDisplay('_json')">JSON</a></h3>
<div id="_json" style="display: none"></div>
</div>
<div id="http">
<h3><a href="#" onclick="return toggleDisplay('_http')">HTTP</a></h3>
<div id="_http" style="display: none"></div>
</div>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment