Skip to content


Subversion checkout URL

You can clone with
Download ZIP
DuckDuckGo instant answer plugins based on JavaScript APIs
JavaScript Perl

Fetching latest commit…

Cannot retrieve the latest commit at this time

Failed to load latest commit information.

Spice Zero-click Info Plugins

See the contribution wiki for a general overview on contributing to DuckDuckGo.

This repository is for contributing JavaScript-based Zero-click Info plugins. Each spice plugin will generally involve at least one HTTP(S) request to a third-party API, though in some cases could be completely stand-alone.

We maintain a list of requested spice plugins, which are colored purple on the Trello board, but whatever you want to attempt is welcome!


First off, thank you!


1) Develop your plugin using the Structure below in either a fork or a branch (if a collaborator).

2) Test your plugin via the Testing procedure below.

3) Submit a pull request.

Feel free to ask questions!


Each spice plugin has its own directory. Some of the directories are in use on the live system, and some are still in development.

Each directory has a structure like this:

# Main file, which gets called by the client at the appropriate time. 
# To understand the flow, look at example/spice.js 

# Calls the js file and is used for testing. 
# Look at example/example.html for extensive workflow comments.

# The js functions and files get segemented by a short namespace prefix.
# This is usually two or three letters, e.g. xk for xkcd.
# Just make something up you think makes sense.

# Nginx conf to call the relevant external API.
# To prevent search leakage (and for caching), we run
# all calls through nginx.
# Start with the xkcd conf and try modifying it appropriately.

# Perl block to determine when to call the spice plugin.
# See xkcd plugin for a good example to start with.

# Example JSON object (returned from third-party API).


You should be able to test your spice plugin via the spice.html file in your plugin directory. That is, it should be able to run in your Web browser and display the information you want it to display in a stand-alone fashion.

In the same token, you should be able to run the example.html to test that you have the repository set up right. If it works, you should get a line about weather at the top.

You do not need a Web server to test, though it is of course fine if you do. That is, you should just be able to open the files locally, i.e. drag or open the appropriate HTML in your browser.

spice.js flow

The overall flow is as follows:

1) An external API is called with a callback function.

2) That callback function is defined by you, and takes the JSON object from the external API and parses out the information needed for display.

3) Your callback function calls the nra function with the appropriate variables. That is the internal function we use to display the Zero-click Info box for spice plugins. You pass nra a object that takes the following parameters.

// Requried snippet (abstract). It can be pure HTML in which case it is set via innerHTML, but better is it is an object, in which case onclick and other event handlers won't be destroyed.
items[0]['a'] = snippet;

// Optional header. If there is a relevant (and relatively short) title, then set it here.
items[0]['h'] = title;

// Required source name and url. 
// These are used to make the More at X link in all Zero-click Info boxes.
// 's' should be the main name.
items[0]['s'] = 'XKCD';
items[0]['u'] = url

// Optional force of a bigger box. Usually the box is auto-resized smaller with an expansion UI if needed. Generally you shouldn't force it to be bigger, but in the XKCD case you don't want the big image to be cutoff.
items[0]['f'] = 1

// Optional image. If there is a thumbnail image, we will display it on the right.
items[0]['i'] = image_url


1) Look at the xkcd/spice.js file for a working live example.

2) If you use internal variables you should put a var statement at the top of the function so they don't leak scope.

3) If you make html, e.g. by createElement, note d is a global shortcut for document, i.e. d.createElement.

4) Any functions should exist in your namespace. For example, for twitter the namespace is tr, so the main callback would named nrtr and a helper function would be nrtr_helper_function.

5) The image is automatically right-floated by default. To avoid looking bad, use and

(if you need line breaks) instead of

. Also it is good to end with a so the More at X line is on the same line. See the twitter plugin for an example.

6) Don't use jQuery. We use YUI2 internally. To set styles you can do:


If the whole Zero-click Info is an image (like in the XKCD case) you can use this class on the img:

Something went wrong with that request. Please try again.