Scholar Code Samples

R

require(httr)
getScholarlyData <- function(token, query){
	url <- 'https://api.lens.org/scholarly/search'
	headers <- c('Authorization' = token, 'Content-Type' = 'application/json')
	httr::POST(url = url, add_headers(.headers=headers), body = query)
}
token <- 'your-access-token'
request <- '{
	"query": {
		"match_phrase": {
			"author.affiliation.name": "Harvard University"
		}
	},
	"size": 1,
	"sort": [{
		"year_published": "desc"
	}]
}'
data <- getScholarlyData(token, request)
content(data, "text")

R - Cursor Based Pagination

Packages <- c("dplyr", "httr", "jsonlite")
lapply(Packages, library, character.only = TRUE)

token <- 'Your Access Token'

# this sets the maximum number of records returned per query
max_results <- 100 

getLensData<- function(token, query){
  url <- 'https://api.lens.org/scholarly/search'
  headers <- c('Authorization' = token, 'Content-Type' = 'application/json')
  httr::POST(url = url, add_headers(.headers=headers), body = query)
}

request <- paste0('{
	"query":  "malaria",
	"size": "',max_results,'",
	"scroll": "1m",
	"include": ["lens_id", "authors", "publication_type", "title"]
}')


data <- getLENSData(token, request)

record_json <- content(data, "text")

# convert json output from article search to list
record_list <- fromJSON(record_json) 

#convert it to a data frame
record_df <- data.frame(record_list) 
total<- record_list[["total"]]


# if a result contains more than the max number of records per request, use cursor based pagination
if(total > max_results) {
  
  #calculate the number of queries needed for those with more than the max number of results
  sets <- ceiling(record_list[["total"]] / max_results) 
  
  # extract the scroll id from the query to go back to the same search
  scroll_id <- record_list[["scroll_id"]] 
  
  # loop through the sets of results needed to bring back all records into a data frame
  for (i in 2:sets){ 
    #extract the latest scroll_id from the last query
    scroll_id <- record_list[["scroll_id"]] 
    
    # new query based on scroll_id and including 'include' for efficiency
    request <- paste0('{"scroll_id": "', 
                      scroll_id,
                      '", "include": ["lens_id", "authors", "publication_type", "title"]
                      }')
    
    # perform article search and extract text results
    data <- getLENSData(token, request)
    record_json <- httr::content(data, "text")
    
    # convert json output from article search to list
    record_list <- jsonlite::fromJSON(record_json) 
    new_df <- data.frame(record_list)
    
    # bind the latest search data frame to the previous data frame
    record_df <- dplyr::bind_rows(record_df,new_df) 
  } 
}
Credit: Neal Haddaway

Python

import requests
url = 'https://api.lens.org/scholarly/search'
data = '''{
     "query": {
           "match_phrase":{
                "author.affiliation.name": "Harvard University"
           }
     },
     "size": 1,
     "sort": [
           {
                "year_published": "desc"
           }
     ]
}'''
headers = {'Authorization': 'Bearer your-access-token', 'Content-Type': 'application/json'}
response = requests.post(url, data=data, headers=headers)
if response.status_code != requests.codes.ok:
  print(response.status_code)
else:
  print(response.text)

Python - Cursor Based Pagination

import requests
import time
import sys
url = 'https://api.lens.org/scholarly/search'

# include fields
include = '''["patent_citations", "lens_id"]'''
# request body with scroll time of 1 minute
request_body = '''{
     "query": "Malaria",
     "size": 100,
     "scroll":"1m",
     "include": %s
}''' % include
headers = {'Authorization': 'Bearer YOUR-TOKEN', 'Content-Type': 'application/json'}

# Recursive function to scroll through paginated results
def scroll(scroll_id):
  # Change the request_body to prepare for next scroll api call
  # Make sure to append the include fields to make faster response
  if scroll_id is not None:
    global request_body
    request_body = '''{"scroll_id": "%s", "include": %s, "scroll": "1m"}''' % (scroll_id, include)

  # make api request
  response = requests.post(url, data=request_body, headers=headers) 

  # If rate-limited, wait for n seconds and proceed the same scroll id
  # Since scroll time is 1 minutes, it will give sufficient time to wait and proceed
  if response.status_code == requests.codes.too_many_requests:
    time.sleep(8)
    scroll(scroll_id)
  # If the response is not ok here, better to stop here and debug it
  elif response.status_code != requests.codes.ok:
      if response.status_code != requests.codes.no_content:
          print(response.json())
      sys.exit(1)
  # If the response is ok, do something with the response, take the new scroll id and iterate
  else:
      json = response.json()
      if json.get('results') is not None and json['results'] > 0:
          scroll_id = json['scroll_id'] # Extract the new scroll id from response
          print(json['data']) #DO something with your data
          scroll(scroll_id)

# start recursive scrolling
scroll(scroll_id=None)

Python - Cursor Based Pagination - List of Identifiers

Use this script of you have a large list of Lens identifiers to send in the request. Sends 5,000 identifiers per request.

import requests
import time
import itertools
import json
import sys

url = 'https://api.lens.org/scholarly/search'
headers = {'Authorization': 'Bearer YOUR-TOKEN', 'Content-Type': 'application/json'}

# Recursive function to scroll through paginated results
def scroll(scroll_id, request_body):
  # Change the request_body to prepare for next scroll api call
  # Make sure to append the include fields to make faster response
  if scroll_id is not None:
    request_body = '''{"scroll_id": "%s", "include": %s}''' % (scroll_id, include)

  # make api request
  response = requests.post(url, data=request_body, headers=headers) 

  # If rate-limited, wait for n seconds and proceed the same scroll id
  # Since scroll time is 1 minutes, it will give sufficient time to wait and proceed
  if response.status_code == requests.codes.too_many_requests:
    time.sleep(8)
    scroll(scroll_id, request_body)
  # If the response is not ok here, better to stop here and debug it
  elif response.status_code != requests.codes.ok:
    if response.status_code != requests.codes.no_content:
        print(response.json())
    sys.exit(1)
  # If the response is ok, do something with the response, take the new scroll id and iterate
  else:
      json = response.json()
      scroll_id = json['scroll_id'] # Extract the new scroll id from response
      print(json['data']) #DO something with your data
      scroll(scroll_id, request_body)

include = '''["lens_id", "patent_citations"]'''
identifiers = [list of Lens identifiers which can be more than 10K]
# take 5000 at a time from the list and scroll for the 5000
for ids in itertools.batched(identifiers, 5000):
  request_body = '''{
    "query": {
        "terms":  {
          "lens_id":'''+(json.dumps(ids))+'''
      }
    },
    "include": %s
  }''' % include

  # start recursive scrolling
  scroll(scroll_id=None, request_body=request_body)

Java

import java.io.*;
import java.net.HttpURLConnection;
import java.net.URL;
import java.nio.charset.StandardCharsets;

public class JavaSample {

    public static void main(String[] args) {
        try {
            URL url = new URL("https://api.lens.org/scholarly/search");
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            conn.setConnectTimeout(30000);
            conn.setDoOutput(true);
            conn.setRequestMethod("POST");
            conn.setRequestProperty("Content-Type", "application/json");
            conn.setRequestProperty("Authorization", "Bearer your-access-token");

            String request = "{\"query\":{\"match_phrase\":{\"author.affiliation.name\":\"Harvard University\"}},\"size\":10,\"sort\":[{\"year_published\":\"desc\"}]}";
            conn.getOutputStream().write(request.getBytes(StandardCharsets.UTF_8));

            if (conn.getResponseCode() != 200) {
                throw new RuntimeException(conn.getResponseCode());
            }

            BufferedReader br = new BufferedReader(new InputStreamReader((conn.getInputStream())));
            String output;
            while ((output = br.readLine()) != null) {
                System.out.println(output);
            }
            conn.disconnect();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

Java - Cursor Based Pagination - Aggregate by Journal

Use this script of you have a list of journals and want to aggregate output by journal to calculate the total scholarly citations per journal.

Note: If you have a large list of journals or result set, it would be better to use the Aggregation API as this script is using recursion.

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonNode;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.node.ArrayNode;
import com.fasterxml.jackson.databind.node.ObjectNode;

import java.io.IOException;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.atomic.AtomicLong;

/**
 * Note: This example is using `jackson-databind` for working with json request/response.
 * To keep it simple, its using HttpURLConnection to send the http request.
 *
 */
public class ScholarlyScrollSample {

    private final static ObjectMapper objectMapper = new ObjectMapper();
    private static final String TOKEN = "YOUR_TOKEN";

    public static void main(String[] args) throws IOException {
        Map<String, CitationOutput> output = new HashMap<>();
        AtomicLong recordCounter = new AtomicLong();

        LensApiClient client = new LensApiClient(new URL("https://api.lens.org/scholarly/search")) {
            /**
             * Writing the response to Map in this example. Could be a database write if required.
             * If it's a heavy operation, a non-blocking implementation might be required.
             * @param response response from the api call
             */
            @Override
            public void consumeResponse(JsonNode response) {
                long numRecords = response.get("results").longValue();
                ArrayNode records = (ArrayNode) response.get("data");
                records.forEach(record -> {
                    String lensId = record.get("lens_id").asText();
                    String sourceTitle = record.path("source").path("title").asText();
                    int scholarlyCitationCounts = record.has("scholarly_citations_count") ? record.get("scholarly_citations_count").asInt() : 0;
                    CitationOutput existing = output.getOrDefault(sourceTitle, new CitationOutput());
                    output.put(sourceTitle, existing.merge(lensId, scholarlyCitationCounts));
                });
                recordCounter.addAndGet(numRecords);
            }
        };
        //Call the API and handle the response. Pass scrollTime e.g. "1m" since we want to scroll.
        client.scrollAndConsume(createRequest(), "1m");

        System.out.println("Completed processing " + recordCounter.get() + " records.");
        output.forEach((k, v) ->
                System.out.println(k + ": (citation count: " + v.getCitationCount() + ", records count: " + v.getRecordsCount() + ")")
        );
    }

    private static ObjectNode createRequest() throws JsonProcessingException {
        String request = "{" +
                "    \"include\": [" +
                "        \"lens_id\"," +
                "        \"scholarly_citations_count\"," +
                "        \"source.title\""+
                "    ]," +
                "    \"size\": 1000," +
                "    \"query\": {" +
                "        \"terms\": {" +
                "            \"source.title.exact\": [" +
                "                \"The Journal of Medical Sciences\"," +
                "                \"An International Journal of Otorhinolaryngology Clinics\"" +
                "            ]" +
                "        }" +
                "    }" +
                "}";
        return objectMapper.readValue(request, ObjectNode.class);
    }

    private abstract static class LensApiClient {
        private final URL url;

        private LensApiClient(URL url) {
            this.url = url;
        }

        /**
         * Perform http call and subsequent recursive calls for scrolling
         * @param request request body
         * @param scrollTime scroll context alive time e.g. 1m. Can be null if no scrolling is required
         * @throws IOException
         */
        public void scrollAndConsume(ObjectNode request, String scrollTime) throws IOException {
            if(scrollTime != null && !scrollTime.isBlank()) {
                request.put("scroll", scrollTime);
            }
            HttpURLConnection conn = (HttpURLConnection) url.openConnection();
            try {
                conn.setConnectTimeout(30000);
                conn.setDoOutput(true);
                conn.setRequestMethod("POST");
                conn.setRequestProperty("Content-Type", "application/json");
                conn.setRequestProperty("Authorization", "Bearer " + TOKEN);
                conn.getOutputStream().write(objectMapper.writeValueAsBytes(request));

                //If rate-limited, wait until the next allowed call
                if (conn.getResponseCode() == 429) {
                    int sleepSeconds = Integer.parseInt(conn.getHeaderField("x-rate-limit-retry-after-seconds"));
                    System.out.println("Rate limited - sleeping for " + sleepSeconds + " seconds ...");
                    Thread.sleep(sleepSeconds * 1000L);
                    scrollAndConsume(request, scrollTime);
                } else if (conn.getResponseCode() == 200) {
                    JsonNode output = objectMapper.readTree(conn.getInputStream());
                    consumeResponse(output);
                    if (output.has("results") && output.get("results").intValue() > 0) {
                        if(output.has("scroll_id")) {
                            System.out.println("Fetched " + output.get("results") + " records. Scrolling next batch ...");
                            String scrollId = output.get("scroll_id").asText();
                            scrollAndConsume(createScrollRequest(request, scrollId), scrollTime);
                        }
                    }
                } else if(conn.getResponseCode() == 204) {
                    System.out.println("No more records found.");
                } else {
                    System.out.println("Failed with status: (" + conn.getResponseCode() + ")");
                }
            } catch (InterruptedException e) {
                throw new RuntimeException(e);
            } finally {
                conn.disconnect();
            }
        }

        /**
         * Implement this to handle the response.
         * Note: if the execution of implemented method is a heavy operation, the scroll context might get expired.
         * In such case, you should make the implementation non-blocking
         * @param response response from the api call
         */
        public abstract void consumeResponse(JsonNode response);

        /**
         * Create request for scroll based pagination
         * @param originalRequest previous request required to copy over pagination to next scroll request
         * @param scrollId scroll id from the response
         * @return scroll request
         */
        private ObjectNode createScrollRequest(ObjectNode originalRequest, String scrollId) {
            ObjectNode request = objectMapper.createObjectNode();
            JsonNode projection = originalRequest.get("include");
            request.set("include", projection);
            request.put("scroll_id", scrollId);
            return request;
        }
    }

    //Sample model schema to get citations count and records
    private static class CitationOutput {
        private Integer citationCount;
        private List<String> lensIds;

        public CitationOutput() {
            this.citationCount = 0;
            this.lensIds = new ArrayList<>();
        }

        public CitationOutput merge(String lensId, Integer citationCount) {
            this.citationCount += citationCount;
            this.lensIds.add(lensId);
            return this;
        }

        public Integer getCitationCount() {
            return citationCount;
        }

        public Integer getRecordsCount() {
            return lensIds.size();
        }
    }
}

NodeJs

var request = require('request');

var endPoint = 'https://api.lens.org/scholarly/search';
var token = 'your-access-token';
var query = {
     "query": {
           "match_phrase":{
                "author.affiliation.name": "Harvard University"
           }
     },
     "size": 10,
     "sort": [
           {
                "year_published": "desc"
           }
     ]
};

var options = {
     url: endPoint,
     body: query,
     json: true,
     headers: {
           "Authorization": "Bearer " + token
     }
}

request.post(options, function(err, res, data) {
     if (err) {
           console.log(err);
     }

     console.log(data);
});

cURL

curl -X POST \
  https://api.lens.org/scholarly/search \
  -H 'Authorization: Bearer your-access-token' \
  -H 'Content-Type: application/json' \
  -d '{
	"query": {
		"bool": {
			"must": {
				"match_phrase": {
					"author.affiliation.name": "Harvard University"
				}
			},
			"filter": {
				"range": {
					"year_published": {
						"gte": "1999",
						"lte": "2000"
					}
				}
			}
		}
	},
	"size": 50
}'