Module:Find sources: Difference between revisions
| imported>Mr. Stradivarius  (start work on a replacement for Template:Find sources multi) | imported>Mr. Stradivarius   (add sanity checks for template config and link-generating code) | ||
| Line 10: | Line 10: | ||
| local TEMPLATE_ROOT = ROOT_PAGE .. '/templates/' | local TEMPLATE_ROOT = ROOT_PAGE .. '/templates/' | ||
| local LINK_ROOT = ROOT_PAGE .. '/links/' | local LINK_ROOT = ROOT_PAGE .. '/links/' | ||
| local DEFAULT_SEPARATOR = ', ' | |||
| local checkType = require('libraryUtils').checkType | |||
| local p = {} | local p = {} | ||
| local function  | local function maybeLoadData(page) | ||
| 	local success,  | 	local success, data = pcall(mw.loadData, page) | ||
| 	if success then | 	if success then | ||
| 		return  | 		return data | ||
| 	else | 	else | ||
| 		return nil | 		return nil | ||
| Line 22: | Line 25: | ||
| end | end | ||
| local function  | local function substituteParams(msg, ...) | ||
| 	local  | 	local params = {...} | ||
| 	if  | 	if params[1] then | ||
| 		return mw.message.newRawMessage(msg):params(params):plain() | |||
| 		return data | |||
| 	else | 	else | ||
| 		return  | 		return msg | ||
| 	end | 	end | ||
| end | end | ||
| function  | local function renderLink(code, searchStrings, display) | ||
| 	-- Get link config. | |||
| 	local templateCfg = maybeLoadData(TEMPLATE_ROOT .. template) | |||
| 	local linkCfg = maybeLoadData(LINK_ROOT .. code) | |||
| 	if not linkCfg then | |||
| 		error(string.format( | |||
| 			"invalid link code '%s'; no link config found at [[%s]]", | |||
| 			code, | |||
| 			LINK_ROOT .. code | |||
| 		)) | |||
| 	end | |||
| 	-- Make URL. | |||
| 	local url | |||
| 	do | |||
| 		local firstFormat = linkCfg.firstFormat or '"$1"' | |||
| 		searchStrings[1] = substituteParams(firstFormat, searchStrings[1]) | |||
| 		local separator = linkCfg.separator or "+" | |||
| 		local searchString = table.concat(searchStrings, separator) | |||
| 		url = linkCfg.url or error(string.format( | |||
| 			"no 'url' field found in the link config at [[%s]]", | |||
| 			LINK_ROOT .. code | |||
| 		)) | |||
| 		url = substituteParams(url, searchString) | |||
| 	end | |||
| 	-- Fetch display. | |||
| 	display = display or linkCfg.display or error(string.format( | |||
| 		"no 'display' field found in the link config at [[%s]]", | |||
| 		LINK_ROOT .. code | |||
| 	)) | |||
| 	return string.format('[%s %s]', url, display) | |||
| end | |||
| local function checkTemplateCfgLinkTable(cfgPage, key, t) | |||
| 	if type(t) ~= 'table' then | |||
| 		error(string.format( | |||
| 			"invalid link table found in key '%s' of template config at [[%s]] " .. | |||
| 				"(expected type table, got type %s)", | |||
| 			key, cfgPage, type(t) | |||
| 		)) | |||
| 	elseif type(t[1]) ~= 'string' then | |||
| 		error(string.format( | |||
| 			"invalid link table found in key '%s' of template config at [[%s]] " .. | |||
| 				"(the first table value must be a string)", | |||
| 			key, cfgPage | |||
| 		)) | |||
| 	elseif t[2] and type(t[2]) ~= 'string' then | |||
| 		error(string.format( | |||
| 			"invalid link table found in key '%s' of template config at [[%s]] " .. | |||
| 				"(if the second table value is specified, it must be a string)", | |||
| 			key, cfgPage | |||
| 		)) | |||
| 	end | |||
| end | |||
| function p._main(template, args) | |||
| 	checkType('_main', 1, template, 'string') | |||
| 	checkType('_main', 2, args, 'table', true) | |||
| 	args = args or {} | |||
| 	local title = mw.title.getCurrentTitle() | |||
| 	-- Get the template config. | |||
| 	local templateCfgPage = TEMPLATE_ROOT .. template | |||
| 	local templateCfg = maybeLoadData(templateCfgPage) | |||
| 	if not templateCfg then | 	if not templateCfg then | ||
| 		error(string.format( | 		error(string.format( | ||
| 			"invalid template name '%s'; no template config found at [[%s]] | 			"invalid template name '%s'; no template config found at [[%s]]", | ||
| 			template, | 			template, templateCfgPage | ||
| 			TEMPLATE_ROOT .. template | |||
| 		)) | 		)) | ||
| 	end | 	end | ||
| 	-- Template config sanity check. | |||
| 	if templateCfg.introLink then | |||
| 		checkTemplateCfgLinkTable(templateCfgPage, 'introLink', templateCfg.introLink) | |||
| 	end | |||
| 	if type(templateCfg.links) ~= 'table' then | |||
| 		error(string.format( | |||
| 			"type error in the 'links' field of the template config at [[%s]] " .. | |||
| 				"(expected table, got %s)", | |||
| 			templateCfgPage, type(templateCfg.links) | |||
| 		)) | |||
| 	end | |||
| 	for _, t in ipairs(templateCfg.links) do | |||
| 		checkTemplateCfgLinkTable(templateCfgPage, 'links', t) | |||
| 	end | |||
| 	-- Get the search strings from the arguments. | |||
| 	local searchStrings = {} | |||
| 	for i, s in ipairs(args) do | |||
| 		searchStrings[i] = mw.uri.encode(s) | |||
| 	end | |||
| 	searchStrings[1] = searchStrings[1] or title.subpageText | |||
| 	-- Make the intro link | |||
| 	local introLink | |||
| 	if templateCfg.introLink then | |||
| 		local code = templateCfg.introLink[1] | |||
| 		local display = templateCfg.introLink[2] | |||
| 		introLink = renderLink(code, searchStrings, display) | |||
| 	else | |||
| 		introLink = '' | |||
| 	end | |||
| 	-- Make the other links | |||
| 	local links = {} | |||
| 	for i, t in ipairs(templateCfg.links) do | |||
| 		local code = t[1] | |||
| 		local display = t[2] | |||
| 		links[i] = renderLink(code, searchStrings, display) | |||
| 	end | |||
| 	links = table.concat(links, templateCfg.separator or DEFAULT_SEPARATOR) | |||
| end | end | ||
| Line 47: | Line 153: | ||
| 			wrappers = mw.site.namespaces[10].name .. ':' .. template | 			wrappers = mw.site.namespaces[10].name .. ':' .. template | ||
| 		}) | 		}) | ||
| 		return t. | 		return t._main(template, args) | ||
| 	end | 	end | ||
| end | end}) | ||
| return p | return p | ||
Revision as of 07:13, 26 September 2014
|  | This Lua module is used on approximately 1,560,000 pages, or roughly 51587% of all pages. To avoid major disruption and server load, any changes should be tested in the module's /sandbox or /testcases subpages, or in your own module sandbox. The tested changes can be added to this page in a single edit. Consider discussing changes on the talk page before implementing them. | 
|  | This module is subject to page protection. It is a highly visible module in use by a very large number of pages, or is substituted very frequently. Because vandalism or mistakes would affect many pages, and even trivial editing might cause substantial load on the servers, it is protected from editing. | 
This module produces a list of links to search engines to help editors find sources about a given subject. It implements {{find sources}} and other similar templates. It is highly extensible; new templates and new link types can be added easily and by any user.
Usage
From wikitext
Usually, from wikitext this module should be used via a template, e.g. {{find sources}}. However, it is also possible to use it directly from #invoke, like this:
{{#invoke:Find sources|template|search term 1|search term 2|...}}
- template is the name of the template that should be called. It must be the full page name of the template, without the namespace, and with the correct capitalisation. So, for Template:Find sources, the template name should be "Find sources"; "find sources" with a lower-case "f" would cause an error. This parameter is required.
- search term 1, search term 2, etc., are the terms to be searched for in each of the search engine links. The first search term will appear in quotation marks, and subsequent search terms will be added without quotation marks. Each term will be separated by spaces. These parameters are all optional. If no parameters are given, the current page name will be used as a search term.
Alternatively the following can be used, which will convert any title to a search query, with the main portion of the title in quotation marks, and any parenthetical disambiguator outside quotation marks.
{{#invoke:Find sources|template|title=title}}
- template as above, is the name of the template that should be called. It must be the full page name of the template, without the namespace, and with the correct capitalisation. So, for Template:Find sources, the template name should be "Find sources"; "find sources" with a lower-case "f" would cause an error. This parameter is required.
- title the title to be converted to a search query. This is processed in the same way as the subpage name of the page if no parameters are used.
From Lua
Load the module with the following code:
local mFindSources = require('Module:Find sources')
You can then produce the list of search links like this:
mFindSources._main(template, searchTerms)
- template is the template name, as outlined in the #From wikitext section above. This parameter is required.
- searchTerms is an array of search terms. Each item in the array corresponds to a numbered search term as outlined in the #From wikitext section above. Alternatively, an empty table with the titlekey set will automatically generate a query as described above. This parameter is optional.
Example syntax:
mFindSources._main('Find sources', {'Albert Einstein', '-"Marilyn Monroe"', 'relativity', 'science'})
Available templates
The following templates are available for use:
Lua error in package.lua at line 80: module 'libraryUtils' not found.
Making new templates
There are two basic ingredients to making a new source-finding template. The first is the template configuration module, and the second is the template invocation on the template page.
Template configuration page
To find the name of the template configuration page, take the page name of your proposed template without the namespace prefix, and add it to the base page of "Module:Find sources/templates/". For example, for Template:Find sources, the configuration page is located at Module:Find sources/templates/Find sources. Note that the template must be capitalised exactly as the page name is, otherwise the arguments will not be passed through from the template page to the module.
The template configuration page should look something like this:
return {
	blurb = "Find sources for $1 – $2",
	introLink = {code = 'google'},
	links = {
		{code = 'google news', display = 'news'},
		{code = 'google newspapers', display = 'newspapers', tooltip = 'Search newspapers with Google Books'}
	},
	separator = ', ',
	isUsedInMainspace = true,
	class = 'custom-class',
	style = 'color: green; font-size: 110%;'
}
- blurb - the text presented to users. There are two parameters available in the blurb, $1 and $2. $1 is the intro link, if specified, and $2 represents the other links. This field is required.
- introLink - an optional introductory link. If specified, this must be a link table. If the display value in the link table is not set, the search terms that the user entered will be used instead.
- links - an array of link tables defining the main links to be used by the template. This field is required.
- separator - the text separating the search engine links. This field is optional; the default value is taken from MediaWiki:Dot-separator.
- isUsedInMainspace - set this to true if the template will be used in the main namespace. If this is not set, when the template is used in the main namespace it will produce an error and be put in Category:Pages with templates in the wrong namespace.
- class - a custom HTML class to apply to the template text. The "plainlinks" class is added by default. This field is optional.
- style - custom CSS to apply to the template text. This field is optional.
Once you have created your template configuration page, you need to add the template to Module:Find sources/templates in order for it to show up in the table of templates on this documentation page.
Link tables
Link tables are used in the "introLink" and "links" fields of the template configuration. They can contain three fields, "code", "display", and "tooltip". The "code" field is required, and must be one of the link codes listed in the table below. The "display" field is optional, and specifies a custom display value for that link to be used by the template. If the display field is not set, a default value is set by the module. The "tooltip" field is also optional, and specifies a custom tooltip for the link. HTML reserved characters are escaped by the module. If the field is absent, no tooltip is displayed. For the "introLink" field, the default value is the search text entered by the user; for the "links" field, the default value is defined in the link configuration modules.
The following table contains all the available link codes, with descriptions and examples.
Lua error in package.lua at line 80: module 'libraryUtils' not found.
Template invocation
The template invocation on the template page itself should look like this:
{{#invoke:Find sources|template}}<noinclude>
{{#invoke:Find sources/autodoc|template}}
</noinclude>
This activates the template code and also provides automatic documentation. template is the name of the template without the namespace prefix, and must be correctly capitalised.
If you want to use custom documentation from a /doc subpage instead of the automatic documentation, use this invocation instead:
{{#invoke:Find sources|template}}<noinclude>
{{documentation}}
<!-- Categories go on the /doc subpage, and interwikis go on Wikidata. -->
</noinclude>
Adding new links
New links are added to Module:Find sources/links. Link codes should be short but descriptive, and should be in lower case. For example, the link code for Google search is "google".
The link configuration page should look something like this:
return {
	url = '//www.google.com/search?as_eq=wikipedia&q=$1',
	display = 'Google',
    description = '[[Google]], the flagship search engine from Google Inc.',
}
| Parameter | Description | 
|---|---|
| url | The url to perform the search. This field takes a parameter $1, which is the URL-encoded search text entered by the user. This field is required. | 
| display | If a display value is not specified in the template configuration, this value is used instead. This field is required. | 
| separator | This is used to separate the search terms entered by the user. This is optional, and defaults to "+" (a URL-encoded space). | 
| description | a brief description of what the search engine link does. This is used in the template documentation to generate the list of link descriptions, and also to make the descriptions in the table of link codes above. This is optional, but recommended. | 
| notes | notes about the search engine link. These are put in the table of link codes above, but they are not put in the template documentation for individual source-finding templates. This field is optional. | 
Automatic documentation
Automatic documentation is provided for templates based on this module; the documentation is generated by Module:Find sources/autodoc. The autodoc module uses the following pages:
- Template:Find sources documentation - the template used for most of the documentation content.
- The template configuration page. Used to find the list of link codes, and to see if the template is used in articles.
- The template documentation configuration page. Used to find template shortcuts.
- The link documentation configuration page. Used to find descriptions of each of the search links.
Template documentation configuration page
This page is located at the "/autodoc" subpage of the template configuration page. It is only used to generate the documentation, not to generate any of the actual template output, so it does not need to be protected. It should look something like this:
return {
	shortcuts = {'shortcut1', 'shortcut2'},
	description = 'This template is used to provide source links on [[WP:AFD|AfD]] pages',
	docIntro = 'This template produces a series of search-engine links to be used on [[WP:AFD|AfD]] pages.'
}
- shortcuts - an array of shortcuts that redirect to the template page. These are displayed on the template documentation page using the {{template shortcut}} template.
- description - a brief description of what the template does. This is displayed on this documentation page in the table of available templates.
- docIntro - text to be used instead of the first sentence of the template documentation for individual source-finding templates. This is optional, and the default value is "This template produces a series of links to various search interfaces to help find additional reference material for articles."
Configuration
The messages used in this module can be found at Module:Find sources/config and Module:Find sources/autodoc/config. This can be helpful for translating this module for use in other languages. Note that any template and link configuration used must also be translated.
-- This module implements {{find sources}} and other similar templates, and also
-- provides a mechanism to easily create new source-finding templates.
--
-- Template settings are found in subpages of [[Module:Find sources/templates]].
-- Link functions are defined in subpages of [[Module:Find sources/links]].
-- Functions shared between the link modules are stored at
-- [[Module:Find sources/shared]].
local ROOT_PAGE = 'Module:Find sources'
local TEMPLATE_ROOT = ROOT_PAGE .. '/templates/'
local LINK_ROOT = ROOT_PAGE .. '/links/'
local DEFAULT_SEPARATOR = ', '
local checkType = require('libraryUtils').checkType
local p = {}
local function maybeLoadData(page)
	local success, data = pcall(mw.loadData, page)
	if success then
		return data
	else
		return nil
	end
end
local function substituteParams(msg, ...)
	local params = {...}
	if params[1] then
		return mw.message.newRawMessage(msg):params(params):plain()
	else
		return msg
	end
end
local function renderLink(code, searchStrings, display)
	-- Get link config.
	local linkCfg = maybeLoadData(LINK_ROOT .. code)
	if not linkCfg then
		error(string.format(
			"invalid link code '%s'; no link config found at [[%s]]",
			code,
			LINK_ROOT .. code
		))
	end
	-- Make URL.
	local url
	do
		local firstFormat = linkCfg.firstFormat or '"$1"'
		searchStrings[1] = substituteParams(firstFormat, searchStrings[1])
		local separator = linkCfg.separator or "+"
		local searchString = table.concat(searchStrings, separator)
		url = linkCfg.url or error(string.format(
			"no 'url' field found in the link config at [[%s]]",
			LINK_ROOT .. code
		))
		url = substituteParams(url, searchString)
	end
	
	-- Fetch display.
	display = display or linkCfg.display or error(string.format(
		"no 'display' field found in the link config at [[%s]]",
		LINK_ROOT .. code
	))
	return string.format('[%s %s]', url, display)
end
local function checkTemplateCfgLinkTable(cfgPage, key, t)
	if type(t) ~= 'table' then
		error(string.format(
			"invalid link table found in key '%s' of template config at [[%s]] " ..
				"(expected type table, got type %s)",
			key, cfgPage, type(t)
		))
	elseif type(t[1]) ~= 'string' then
		error(string.format(
			"invalid link table found in key '%s' of template config at [[%s]] " ..
				"(the first table value must be a string)",
			key, cfgPage
		))
	elseif t[2] and type(t[2]) ~= 'string' then
		error(string.format(
			"invalid link table found in key '%s' of template config at [[%s]] " ..
				"(if the second table value is specified, it must be a string)",
			key, cfgPage
		))
	end
end
function p._main(template, args)
	checkType('_main', 1, template, 'string')
	checkType('_main', 2, args, 'table', true)
	args = args or {}
	local title = mw.title.getCurrentTitle()
	
	-- Get the template config.
	local templateCfgPage = TEMPLATE_ROOT .. template
	local templateCfg = maybeLoadData(templateCfgPage)
	if not templateCfg then
		error(string.format(
			"invalid template name '%s'; no template config found at [[%s]]",
			template, templateCfgPage
		))
	end
	-- Template config sanity check.
	if templateCfg.introLink then
		checkTemplateCfgLinkTable(templateCfgPage, 'introLink', templateCfg.introLink)
	end
	if type(templateCfg.links) ~= 'table' then
		error(string.format(
			"type error in the 'links' field of the template config at [[%s]] " ..
				"(expected table, got %s)",
			templateCfgPage, type(templateCfg.links)
		))
	end
	for _, t in ipairs(templateCfg.links) do
		checkTemplateCfgLinkTable(templateCfgPage, 'links', t)
	end
	-- Get the search strings from the arguments.
	local searchStrings = {}
	for i, s in ipairs(args) do
		searchStrings[i] = mw.uri.encode(s)
	end
	searchStrings[1] = searchStrings[1] or title.subpageText
	
	-- Make the intro link
	local introLink
	if templateCfg.introLink then
		local code = templateCfg.introLink[1]
		local display = templateCfg.introLink[2]
		introLink = renderLink(code, searchStrings, display)
	else
		introLink = ''
	end
	-- Make the other links
	local links = {}
	for i, t in ipairs(templateCfg.links) do
		local code = t[1]
		local display = t[2]
		links[i] = renderLink(code, searchStrings, display)
	end
	links = table.concat(links, templateCfg.separator or DEFAULT_SEPARATOR)
end
setmetatable(p, { __index = function(t, template)
	return function(frame)
		local args = require('Module:Arguments').getArgs(frame, {
			wrappers = mw.site.namespaces[10].name .. ':' .. template
		})
		return t._main(template, args)
	end
end})
return p







