Jump to content
Wikipedia The Free Encyclopedia

Module:Cs1 documentation support

From Wikipedia, the free encyclopedia
Module documentation[view] [edit] [history] [purge]
Warning This Lua module is used on approximately 250 pages and changes may be widely noticed. Test changes in the module's /sandbox or /testcases subpages, or in your own module sandbox. Consider discussing changes on the talk page before implementing them.

This module provides support for cs1|2 documentation by fetching data from the various modules, most notably Module:Citation/CS1/Configuration and rendering that data in a reader-accessible format.

Usage

{{#invoke:Cs1 documentation support|function_name}}

The above documentation is transcluded from Module:Cs1 documentation support/doc. (edit | history)
Editors can experiment in this module's sandbox (edit | diff) and testcases (create) pages.
Subpages of this module.

 require('strict');
 localgetArgs=require('Module:Arguments').getArgs;

 localcfg=mw.loadData('Module:Citation/CS1/Configuration');-- load the configuration module
 localwhitelist=mw.loadData('Module:Citation/CS1/Whitelist');-- load the whitelist module


 localexclusion_lists={-- TODO: move these tables into a separate ~/data module and mw.loadData() it
 ['cite book']={
 ['agency']=true,
 ['air-date']=true,
 ['arxiv']=true,
 ['biorxiv']=true,
 ['citeseerx']=true,
 ['class']=true,
 ['conference']=true,
 ['conference-format']=true,
 ['conference-url']=true,
 ['degree']=true,
 ['department']=true,
 ['display-interviewers']=true,
 ['docket']=true,
 ['episode']=true,
 ['interviewer#']=true,
 ['interviewer-first#']=true,
 ['interviewer-link#']=true,
 ['interviewer-mask#']=true,
 ['ismn']=true,
 ['issn']=true,
 ['issue']=true,
 ['jfm']=true,
 ['journal']=true,
 ['jstor']=true,
 ['mailinglist']=true,
 ['message-id']=true,
 ['minutes']=true,
 ['MR']=true,
 ['network']=true,
 ['number']=true,
 ['RFC']=true,
 ['script-journal']=true,
 ['season']=true,
 ['section']=true,
 ['sections']=true,
 ['series-link']=true,
 ['series-number']=true,
 ['series-separator']=true,
 ['sheet']=true,
 ['sheets']=true,
 ['SSRN']=true,
 ['station']=true,
 ['time']=true,
 ['time-caption']=true,
 ['trans-article']=true,
 ['trans-journal']=true,
 ['transcript']=true,
 ['transcript-format']=true,
 ['transcript-url']=true,
 ['ZBL']=true,
 },
 ['cite journal']={
 ['agency']=true,
 ['air-date']=true,
 ['book-title']=true,
 ['chapter']=true,
 ['chapter-format']=true,
 ['chapter-url']=true,
 ['chapter-url-access']=true,
 ['class']=true,
 ['conference']=true,
 ['conference-format']=true,
 ['conference-url']=true,
 ['contribution']=true,
 ['contributor#']=true,
 ['contributor-first#']=true,
 ['contributor-link#']=true,
 ['contributor-mask#']=true,
 ['degree']=true,
 ['department']=true,
 ['display-interviewers']=true,
 ['docket']=true,
 ['edition']=true,
 ['editor#']=true,
 ['editor-first#']=true,
 ['editor-link#']=true,
 ['editor-mask#']=true,
 ['editors']=true,
 ['encyclopedia']=true,
 ['episode']=true,
 ['ignore-isbn-error']=true,
 ['interviewer#']=true,
 ['interviewer-first#']=true,
 ['interviewer-link#']=true,
 ['interviewer-mask#']=true,
 ['isbn']=true,
 ['ismn']=true,
 ['LCCN']=true,
 ['mailinglist']=true,
 ['message-id']=true,
 ['minutes']=true,
 ['network']=true,
 ['script-chapter']=true,
 ['season']=true,
 ['section']=true,
 ['sections']=true,
 ['series-link']=true,
 ['series-number']=true,
 ['series-separator']=true,
 ['sheet']=true,
 ['sheets']=true,
 ['station']=true,
 ['time']=true,
 ['time-caption']=true,
 ['trans-article']=true,
 ['transcript']=true,
 ['transcript-format']=true,
 ['transcript-url']=true,
 },
 }

 --[[-------------------------< A D D _ T O _ L I S T >---------------------------------------------------------

 adds code/name pair to code_list and name/code pair to name_list; code/name pairs in override_list replace those
 taken from the MediaWiki list; these are marked with a superscripted dagger.

 |script-<param>= lang codes always use override names so dagger is omitted

 ]]

 localfunctionadd_to_list(code_list,name_list,override_list,code,name,dagger)
 iffalse==daggerthen
 dagger='';-- no dagger for |script-<param>= codes and names
 else
 dagger='<sup>†</sup>';-- dagger for all other lists using override
 end

 ifoverride_list[code]then-- look in the override table for this code
 code_list[code]=override_list[code]..dagger;-- use the name from the override table; mark with dagger
 name_list[override_list[code]]=code..dagger;
 else
 code_list[code]=name;-- use the MediaWiki name and code
 name_list[name]=code;
 end
 end


 --[[-------------------------< L I S T _ F O R M A T >---------------------------------------------------------

 formats key/value pair into a string for rendering
 	['k'] = 'v'	→ k: v

 ]]

 localfunctionlist_format(result,list)
 fork,vinpairs(list)do
 table.insert(result,k..': '..v);
 end
 end


 --[[-------------------------< L A N G _ L I S T E R >---------------------------------------------------------

 Module entry point

 Crude documentation tool that returns one of several lists of language codes and names.

 Used in Template:Citation Style documentation/language/doc

 {{#invoke:cs1 documentation support|lang_lister|list=<selector>|lang=<code>}}

 where <selector> is one of the values:
 	2char – list of ISO 639-1 codes and names sorted by code
 	3char – list of ISO 639-2, -3 codes and names sorted by code
 	ietf – list of IETF language tags and names sorted by tag
 	ietf2 – list of ISO 639-1 based IETF language tags and names sorted by tag
 	ietf3 – list of list of ISO 639-2, -3 based IETF language tags and names sorted by tag
 	name – list of language names and codes sorted by name
 	all - list all language codes/tags and names sorted by code/tag

 where <code> is a MediaWiki supported 2, 3, or ietf-like language code; because of fall-back, language names may
 be the English-language names.


 ]]

 localfunctionlang_lister(frame)
 locallang=(frame.args.langand''~=frame.args.lang)andframe.args.langormw.getContentLanguage():getCode()
 localsource_list=mw.language.fetchLanguageNames(lang,'all');
 localoverride=cfg.lang_tag_remap;
 localcode_1_list={};
 localcode_2_list={};
 localietf_list={};
 localietf_list2={};
 localietf_list3={};
 localname_list={};

 ifnot({['2char']=true,['3char']=true,['ietf']=true,['ietf2']=true,['ietf3']=true,['name']=true,['all']=true})[frame.args.list]then
 return'<span style="color:#d33">unknown list selector: '..frame.args.list..'</span>';
 end

 forcode,nameinpairs(source_list)do
 if'all'==frame.args.listthen
 add_to_list(code_1_list,name_list,override,code,name);-- use the code_1_list because why not?
 elseif2==code:len()then
 add_to_list(code_1_list,name_list,override,code,name);
 elseif3==code:len()then
 add_to_list(code_2_list,name_list,override,code,name);
 elseifcode:match('^%a%a%-.+')then-- ietf with 2-character language tag
 add_to_list(ietf_list,name_list,override,code,name);-- add to main ietf list for |list=ietf
 add_to_list(ietf_list2,name_list,override,code,name);-- add to ietf2 list
 elseifcode:match('^%a%a%a%-.+')then-- ietf with 3-character language tag
 add_to_list(ietf_list,name_list,override,code,name);-- add to main ietf list for |list=ietf
 add_to_list(ietf_list3,name_list,override,code,name);-- add to ietf3 list
 end
 end

 localresult={};
 localout={};

 if'2char'==frame.args.listor'all'==frame.args.listthen-- iso 639-1
 list_format(result,code_1_list);
 elseif'3char'==frame.args.listthen-- iso 639-2, 3
 list_format(result,code_2_list);
 elseif'ietf'==frame.args.listthen-- all ietf tags
 list_format(result,ietf_list);
 elseif'ietf2'==frame.args.listthen-- 2-character ietf tags
 list_format(result,ietf_list2);
 elseif'ietf3'==frame.args.listthen-- 3 character ietf tags
 list_format(result,ietf_list3);
 else--must be 'name'
 list_format(result,name_list);
 end

 localtemplatestyles=frame:extensionTag{
 name='templatestyles',args={src="Div col/styles.css"}
 }

 table.sort(result);
 table.insert(result,1,templatestyles..'<div class="div-col" style="column-width:16em">');
 table.insert(out,table.concat(result,'\n*'));
 table.insert(out,'</div>');

 returntable.concat(out,'\n');
 end


 --[[--------------------------< S C R I P T _ L A N G _ L I S T E R >------------------------------------------

 Module entry point

 Crude documentation tool that returns list of language codes and names supported by the various |script-<param>= parameters.

 used in Help:CS1 errors

 {{#invoke:cs1 documentation support|script_lang_lister}}

 ]]

 localfunctionscript_lang_lister(frame)
 locallang_code_src=cfg.script_lang_codes;-- get list of allowed script language codes
 localoverride=cfg.lang_tag_remap;
 localthis_wiki_lang=mw.language.getContentLanguage().code;-- get this wiki's language

 localcode_list={};-- interim list of aliases
 localname_list={};-- not used; defined here so that we can reuse add_to_list() 
 localout={};-- final output (for now an unordered list)

 for_,codeinipairs(lang_code_src)do-- loop through the list of codes
 localname=mw.language.fetchLanguageName(code,this_wiki_lang);-- get the language name associated with this code
 add_to_list(code_list,name_list,override,code,name,false);-- name_list{} not used but provided so that we can reuse add_to_list(); don't add superscript dagger
 end

 localresult={};
 localout={};

 list_format(result,code_list);

 localtemplatestyles=frame:extensionTag{
 name='templatestyles',args={src="Div col/styles.css"}
 }

 table.sort(result);
 table.insert(result,1,templatestyles..'<div class="div-col" style="column-width:16em">');
 table.insert(out,table.concat(result,'\n*'));
 table.insert(out,'</div>');

 returntable.concat(out,'\n');
 end


 --[[--------------------------< A L I A S _ L I S T E R >------------------------------------------------------

 experimental code that lists parameters and their aliases. Perhaps basis for some sort of documentation?

 {{#invoke:cs1 documentation support|alias_lister}}

 ]]

 localfunctionalias_lister()
 localalias_src=cfg.aliases;-- get master list of aliases
 localkey;-- key for k/v in a new table
 locallist={};-- interim list of aliases
 localout={};-- final output (for now an unordered list)

 for_,aliasesinpairs(alias_src)do-- loop throu the master list of aliases
 if'table'==type(aliases)then-- table only when there are aliases
 fori,aliasinipairs(aliases)do-- loop through all of the aliases
 if1==ithen-- first 'alias' is the canonical parameter name
 key=alias;-- so it becomes the key in list
 else
 list[key]=list[key]and(list[key]..', '..alias)oralias;-- make comma-separated list of aliases
 list[alias]='see '..key;-- make a back reference from this alias to the canonical parameter
 end
 end
 end
 end

 fork,vinpairs(list)do-- loop through the list to make a simple unordered list
 table.insert(out,table.concat({'*',k,': ',v}));
 end

 table.sort(out);-- sort it
 returntable.concat(out,'010円');-- concatenate with \n
 --	return (mw.dumpObject (list))
 end


 --[[--------------------------< C A N O N I C A L _ P A R A M _ L I S T E R >----------------------------------

 experimental code that lists canonical parameter names. Perhaps basis for some sort of documentation?

 returns a comma separated, alpha sorted, list of the canonical parameters. If given a template name, excludes
 parameters listed in that template's exclusion_list[<template>]{} table (if a table has been defined).

 {{#invoke:cs1 documentation support|canonical_param_lister|<template>}}

 ]]

 localfunctioncanonical_param_lister(frame)
 localtemplate=frame.args[1];
 if''==templatethen
 template=nil;
 end

 iftemplatethen
 template=mw.text.trim(template:lower());
 end

 localalias_src=cfg.aliases;-- get master list of aliases
 localid_src=cfg.id_handlers;-- get master list of identifiers

 locallist={};-- interim list of aliases
 localout={};-- final output (for now an unordered list)

 for_,aliasesinpairs(alias_src)do-- loop through the master list of aliases
 localname;
 if'table'==type(aliases)then-- table only when there are aliases
 name=aliases[1];-- first member of an aliases table is declared canonical
 else
 name=aliases;-- for those parameters that do not have any aliases, the parameter is declared canonical
 end

 ifnottemplatethen-- no template name, add this parameter
 table.insert(list,name);
 elseifnotexclusion_lists[template]then-- template name but no exclusion list
 table.insert(list,name);
 elseifnotexclusion_lists[template][name]then-- template name and exclusion list but name not in list
 table.insert(list,name);
 end
 end

 fork,idsinpairs(id_src)do-- spin through the list of identifiers
 localname=id_src[k].parameters[1];-- get the first (left-most) parameter name
 localaccess=id_src[k].custom_access;-- get the access-icon parameter if it exists for this identifier
 ifnottemplatethen-- no template name
 table.insert(list,name);-- add this parameter
 ifaccessthen
 table.insert(list,access);-- add this access-icon parameter
 end
 elseifnotexclusion_lists[template]then-- template name but no exclusion list
 table.insert(list,name);
 ifaccessthen
 table.insert(list,access);
 end
 elseifnotexclusion_lists[template][name]then-- template name and exclusion list but name not in list
 table.insert(list,name);
 ifaccessthen
 table.insert(list,access);
 end
 end
 end

 for_,paraminipairs(list)do-- loop through the list to make a simple unordered list
 table.insert(out,table.concat({'*',param}));
 end

 localfunctioncomp(a,b)-- used in following table.sort()
 returna:lower()<b:lower();
 end

 table.sort(out,comp);-- sort the list
 returntable.concat(out,'010円');-- concatenate with \n
 --	return (mw.dumpObject (list))
 end


 --[[--------------------------< C A N O N I C A L _ N A M E _ G E T >------------------------------------------

 returns first (canonical) name when metaparameter is assigned a table of names
 returns name when metaparameter is assigned a single name
 returns empty string when metaparameter name not found in alias_src{}, id_src{}, or id_src[meta].custom_access

 metaparameter <metaparam> is the key in Module:Citation/CS1 aliases{} table or id_handlers{} table. Because access-icon
 don't have <metaparam> keys, per se, we create pseudo <metaparam> keys by appending 'access' to the identifier <metaparam>:
 	the <metaparam> for |doi-access= is, for the purposes of this function, DOIaccess, etc

 Some lists of aliases might be better served when a particular alias is identified as the canonical alias for a 
 particular use case. If, for example, <metaparam> Perodical lists:
 	'journal', 'magazine', 'newspaper', 'periodical', 'website', 'work'
 that order works fine for {{cite journal}} documentation but doesn't work so well for {{cite magazine}}, {{cite news}},
 or {{cite web}}. So, for using this function to document {{cite magazine}} the returned value should be the
 parameter best suited for that template so we can specify magazine in the override (frame.args[2])

 While for this function, it would be just as simple to not use the function, this mechanism is implemented here 
 to match similar functionality in alias_names_get() (there are slight differences)
 	<override> must exist in the alias list
 	does not apply to the access icon parameters (ignored - these have no aliases)

 (and which would be best for {{cite news}}? |newspaper= or |work=? can't solve all of the worlds problems at once).

 output format is controlled by |format=
 	plain - renders in plain text in a <span> tag; may have id attribute
 	para - renders as it would in {{para|<param>}}

 {{#invoke:cs1 documentation support|canonical_name_get|<metaparam>|<override>|id=<attribute>|format=[plain|para]}}

 ]]

 localfunctioncanonical_name_get(frame)
 localalias_src=cfg.aliases;-- get master list of aliases
 localid_src=cfg.id_handlers;-- get master list of identifiers
 localargs=getArgs(frame);

 localname;
 localmeta=args[1]
 localoverride=args[2];

 localaccess;-- for id-access parameters
 ifmeta:match('^(%u+)access')then-- the metaparameter (which is not used in ~/Configuration) is id_handlers key concatenated with access: BIBCODEaccess
 meta,access=meta:gsub('^(%u+)access','%1');-- strip 'access' text from meta and use returned count value as a flag
 end

 ifalias_src[meta]then
 name=alias_src[meta];-- name is a string or a table
 if'table'==type(name)then-- table only when there are aliases
 ifnotoverridethen
 name=name[1];-- first member of an aliases table is declared canonical
 else
 for_,vinipairs(name)do-- here when override is set; spin throu the aliases to make sure override matches alias in table
 ifv==overridethen
 name=v;-- declare override to be the canonical param for this use case
 break;
 end
 end
 end
 end

 elseifid_src[meta]then-- if there is an id handler
 ifaccessthen-- and if this is a request for the handler's custom access parameter
 ifid_src[meta].custom_accessthen-- if there is a custom access parameter
 name=id_src[meta].custom_access;-- use it
 else
 return'';-- nope, return empty string
 end
 else
 ifnotoverridethen
 name=id_src[meta].parameters[1];-- get canonical id handler parameter
 else
 for_,vinipairs(id_src[meta].parameters)do-- here when override is set; spin throu the aliases to make sure override matches alias in table
 ifv==overridethen
 name=v;-- declare override to be the canonical param for this use case
 break;
 end
 end
 end
 end
 else
 return'';-- metaparameter not specified, or no such metaparameter
 end

 if'plain'==args.formatthen-- format and return the output
 ifargs.idthen
 returnstring.format('<span id="%s">%s</span>',args.id,name);-- plain text with id attribute
 else
 returnname;-- plain text
 end
 elseif'para'==args.formatthen
 returnstring.format('<code class="nowrap">|%s=</code>',name);-- same as {{para|<param>}}
 end

 returnstring.format('<b id="%s">%s</b>',args.idor'',name);-- because {{csdoc}} bolds param names
 end


 --[[--------------------------< A L I A S _ N A M E S _ G E T >------------------------------------------------

 returns list of aliases for metaparameter <metaparam>
 returns empty string when there are no aliases
 returns empty string when <metaparam> name not found in alias_src{} or id_src{}; access icon parameters have no aliases so ignored

 metaparameter <metaparam> is the key in Module:Citation/CS1 aliases{} table or id_handlers{} table.

 Some lists of aliases might be better served when a particular alias is identified as the canonical alias for a 
 particular use case. If, for example, <metaparam> Perodical lists:
 	'journal', 'magazine', 'newspaper', 'periodical', 'website', 'work'
 that order works fine for {{cite journal}} documentation but doesn't work so well for {{cite magazine}}, {{cite news}},
 or {{cite web}}. So, for using this function to document {{cite magazine}} the returned value should be the
 aliases that are not best suited for that template so we can specify magazine in the override (frame.args[2])
 to be the canonical parameter so it won't be listed with the rest of the aliases (normal canonical journal will be)

 	<override> must exist in the alias list except:
 		when <override> value is 'all', returns the canonical parameter plus all of the aliases

 output format is controlled by |format=
 	plain - renders in plain text in a <span> tag; may have id attribute
 	para - renders as it would in {{para|<param>}}
 	when not specified, refurns the default bold format used for {{csdoc}}

 {{#invoke:cs1 documentation support|alias_name_get|<metaparam>|<override>|format=[plain|para]}}

 ]]

 localfunctionalias_names_get(frame)
 localalias_src=cfg.aliases;-- get master list of aliases
 localid_src=cfg.id_handlers;-- get master list of identifiers
 localargs=getArgs(frame);

 localmeta=args[1];
 localoverride=args[2];

 localout={};
 localsource;-- selected parameter or id aliases list
 localaliases;

 source=alias_src[meta]or(id_src[meta]andid_src[meta].parameters);
 ifnotsourcethen
 ifmeta:match('%u+access')then
 return'no'==args.noneand''or'none';-- custom access parameters don't have aliases
 else
 return'';-- no such meta
 end
 elseifnotsource[2]then-- id_source[meta] is always a table; if no second member, no aliases
 return'no'==args.noneand''or'none';
 end

 ifnotoverridethen
 aliases=source;-- normal skip-canonical param case
 else
 localflag='all'==overrideandtrueornil;-- so that we know that <override> parameter is a valid alias; spoof when override == 'all'
 aliases={[1]=''};-- spoof to push alias_src[meta][1] and id_src[meta][1] into aliases[2]
 for_,vinipairs(source)do-- here when override is set; spin through the aliases to make sure override matches alias in table
 ifv~=overridethen
 table.insert(aliases,v);-- add all but overridden param to the the aliases list for this use case
 else
 flag=true;-- set the flag so we know that <override> is a valid alias
 end
 end
 ifnotflagthen
 aliases={}-- unset the table as error indicator
 end
 end

 if'table'==type(aliases)then-- table only when there are aliases
 fori,aliasinipairs(aliases)do
 if1~=ithen-- aliases[1] is the canonical name; don't include it
 if'plain'==args.formatthen-- format and return the output
 table.insert(out,alias);-- plain text
 elseif'para'==args.formatthen
 table.insert(out,string.format('<code class="nowrap">|%s=</code>',alias));-- same as {{para|<param>}}
 else
 table.insert(out,string.format("'''%s'''",alias));-- because csdoc bolds param names
 end
 end
 end

 returntable.concat(out,', ');-- make pretty list and quit
 end

 return'no'==args.noneand''or'none';-- no metaparameter with that name or no aliases
 end


 --[[--------------------------< I S _ B O O K _ C I T E _ T E M P L A T E >------------------------------------

 fetch the title of the current page; if it is a preprint template, return true; empty string else

 ]]

 localbook_cite_templates={
 ['citation']=true,
 ['cite book']=true,
 }

 localfunctionis_book_cite_template()
 localtitle=mw.title.getCurrentTitle().rootText;-- get title of current page without namespace and without sub-pages; from Template:Cite book/new -> Cite book

 title=titleandtitle:lower()or'';
 returnbook_cite_templates[title]or'';
 end


 --[[--------------------------< I S _ L I M I T E D _ P A R A M _ T E M P L A T E >----------------------------

 fetch the title of the current page; if it is a preprint template, return true; empty string else

 ]]

 locallimited_param_templates={-- if ever there is a need to fetch info from ~/Whitelist then
 ['cite arxiv']=true,-- this list could also be fetched from there
 ['cite biorxiv']=true,
 ['citeseerx']=true,
 ['ssrn']=true,
 }

 localfunctionis_limited_param_template()
 localtitle=mw.title.getCurrentTitle().rootText;-- get title of current page without namespace and without sub-pages; from Template:Cite book/new -> Cite book

 title=titleandtitle:lower()or'';
 returnlimited_param_templates[title]or'';
 end


 --[[--------------------------< H E A D E R _ M A K E >--------------------------------------------------------

 makes a section header from <header_text> and <level>; <level> defaults to 2; cannot be less than 2

 ]]

 localfunction_header_make(args)
 ifnotargs[1]then
 return'';-- no header text
 end

 locallevel=args[2]andtonumber(args[2])or2;

 level=string.rep('=',level);
 returnlevel..args[1]..level;
 end


 --[[--------------------------< H E A D E R _ M A K E >--------------------------------------------------------

 Entry from an {{#invoke:}}
 makes a section header from <header_text> and <level>; <level> defaults to 2; cannot be less than 2

 ]]

 localfunctionheader_make(frame)
 localargs=getArgs(frame);
 return_header_make(args);
 end


 --[[--------------------------< I D _ L I M I T S _ G E T >----------------------------------------------------

 return the limit values for named identifier parameters that have <id> limits (pmc, pmid, ssrn, s2cid, oclc, osti, rfc); the return
 value used in template documentation and error message help-text

 {{#invoke:Cs1 documentation support|id_limits_get|<id>}}

 ]]

 localfunctionid_limits_get(frame)
 localargs=getArgs(frame);
 localhandlers=cfg.id_handlers;-- get id_handlers {} table from ~/Configuration

 returnargs[1]andhandlers[args[1]:upper()].id_limitor('<span style="color:#d33">No limit defined for identifier: '..(args[1]or'<unknown name>')..'</span>');
 end


 --[[--------------------------< C A T _ L I N K _ M A K E >----------------------------------------------------
 ]]

 localfunctioncat_link_make(cat)
 returntable.concat({'[[:Category:',cat,']]'});
 end


 --[[--------------------------< S C R I P T _ C A T _ L I S T E R >--------------------------------------------

 utility function to get script-language categories

 ]]

 locallang_list_t=mw.language.fetchLanguageNames('en','all');

 localfunctionscript_cat_lister(script_lang_codes_t,lang_tag_remap_t,cats_list_t)
 for_,lang_codeinipairs(script_lang_codes_t)do
 locallang_name=lang_tag_remap_t[lang_code]orlang_list_t[lang_code];-- use remap table to get Bengali instead of Bangla and the like; else use standard MediaWiki names
 localcat='CS1 uses '..lang_name..'-language script ('..lang_code..')';-- build a category name
 cats_list_t[cat]=1;-- and save it
 end
 end


 --[[--------------------------< C S 1 _ C A T _ L I S T E R >--------------------------------------------------

 This is a crude tool that reads the category names from Module:Citation/CS1/Configuration, makes links of them,
 and then lists them in sorted lists. A couple of parameters control the rendering of the output:
 	|select=	-- (required) takes one of three values: error, maint, prop
 	|sandbox=	-- takes one value: no
 	|hdr-lvl=	-- base header level (number of == that make a header); default:2 min:2

 This tool will automatically attempt to load a sandbox version of ~/Configuration if one exists.
 Setting |sandbox=no will defeat this.

 {{#invoke:cs1 documentation support|cat_lister|select=<error|maint|prop>|sandbox=<no>}}

 ]]

 localfunctioncat_lister(frame)
 localargs=getArgs(frame);

 locallist_live_cats={};-- list of live categories
 locallist_sbox_cats={};-- list of sandbox categories

 locallive_sbox_out={}-- list of categories that are common to live and sandbox modules
 locallive_not_in_sbox_out={}-- list of categories in live but not sandbox
 localsbox_not_in_live_out={}-- list of categories in sandbox but not live

 localout={};-- final output assembled here

 localsandbox;-- boolean; true: evaluate the sandbox module
 localhdr_lvl;-- 

 localsb_cfg;
 localsandbox,sb_cfg=pcall(mw.loadData,'Module:Citation/CS1/Configuration/sandbox');-- get sandbox configuration

 localcat;

 localselect=args.select;
 if'no'==args.sandboxthen-- list sandbox?
 sandbox=false;-- no, live only
 end
 ifhdr_lvlthen-- if set and
 iftonumber(hdr_lvl)then-- can be converted to number
 if2>tonumber(hdr_lvl)then-- min is 2
 hdr_lvl=2;-- so set to min
 end
 else-- can't be converted
 hdr_lvl=2;-- so default to min
 end
 else
 hdr_lvl=2;-- not set so default to min
 end

 if'error'==selector'maint'==selectthen-- error and main categorys handling different from poperties cats
 for_,tinpairs(cfg.error_conditions)do-- get the live module's categories
 if('error'==selectandt.message)or('maint'==selectandnott.message)then
 cat=t.category:gsub('|(.*)$','');-- strip sort key if any
 list_live_cats[cat]=1;-- add to the list
 end
 end

 ifsandboxthen-- if ~/sandbox module exists and |sandbox= not set to 'no'
 for_,tinpairs(sb_cfg.error_conditions)do-- get the sandbox module's categories
 if('error'==selectandt.message)or('maint'==selectandnott.message)then
 cat=t.category:gsub('|(.*)$','');-- strip sort key if any
 list_sbox_cats[cat]=1;-- add to the list
 end
 end
 end

 elseif'prop'==selectthen-- prop cats
 for_,catinpairs(cfg.prop_cats)do-- get the live module's categories
 cat=cat:gsub('|(.*)$','');-- strip sort key if any
 list_live_cats[cat]=1;-- add to the list
 end

 script_cat_lister(cfg.script_lang_codes,cfg.lang_tag_remap,list_live_cats);-- get live module's foriegn language script cats

 ifsandboxthen-- if ~/sandbox module exists and |sandbox= not set to 'no'
 for_,catinpairs(sb_cfg.prop_cats)do-- get the sandbox module's categories
 cat=cat:gsub('|(.*)$','');-- strip sort key if any
 list_sbox_cats[cat]=1;-- add to the list
 end

 script_cat_lister(sb_cfg.script_lang_codes,sb_cfg.lang_tag_remap,list_sbox_cats);-- get sandbox module's foriegn language script cats
 end
 else
 return'<span style="color:#d33; font-style:normal;">error: unknown selector: '..select..'</span>'
 end

 fork,_inpairs(list_live_cats)do-- separate live/sbox common cats from cats not in sbox
 ifnotlist_sbox_cats[k]andsandboxthen
 table.insert(live_not_in_sbox_out,cat_link_make(k));-- in live but not in sbox
 else
 table.insert(live_sbox_out,cat_link_make(k));-- in both live and sbox
 end
 end

 fork,_inpairs(list_sbox_cats)do-- separate sbox/live common cats from cats not in live
 ifnotlist_live_cats[k]then
 table.insert(sbox_not_in_live_out,cat_link_make(k));-- in sbox but not in live
 end
 end

 localfunctioncomp(a,b)-- local function for case-agnostic category name sorting
 returna:lower()<b:lower();
 end

 localheader;-- initialize section header with name of selected category list
 if'error'==selectthen
 header='error';
 elseif'maint'==selectthen
 header='maintenance';
 else
 header='properties';
 end

 header=table.concat({-- build the main header
 'Live ',-- always include this
 ((sandboxand'and sandbox ')or''),-- if sandbox evaluated, mention that
 header,-- add the list name
 ' categories (',-- finish the name and add
 #live_sbox_out,-- count of categories listed
 ')'-- close
 })

 localtemplatestyles=frame:extensionTag{
 name='templatestyles',args={src="Div col/styles.css"}
 }

 header=table.concat({-- make a useable header
 _header_make({header,hdr_lvl}),
 '\n'..templatestyles..'<div class="div-col">'-- opening <div> for columns
 });

 table.sort(live_sbox_out,comp);-- sort case agnostic acsending
 table.insert(live_sbox_out,1,header);-- insert the header at the top
 table.insert(out,table.concat(live_sbox_out,'\n*'));-- make a big string of unordered list markup
 table.insert(out,'</div>\n');-- close the </div> and add new line so the next header works

 if0~=#live_not_in_sbox_outthen-- when there is something in the table
 header=table.concat({-- build header for subsection
 'In live but not in sandbox (',
 #live_not_in_sbox_out,
 ')'
 });

 header=table.concat({-- make a useable header
 _header_make({header,hdr_lvl+1}),
 '\n'..templatestyles..'<div class="div-col">'
 });

 table.sort(live_not_in_sbox_out,comp);
 table.insert(live_not_in_sbox_out,1,header);
 table.insert(out,table.concat(live_not_in_sbox_out,'\n*'));
 table.insert(out,'</div>\n');
 end

 if0~=#sbox_not_in_live_outthen-- when there is something in the table
 header=table.concat({-- build header for subsection
 'In sandbox but not in live (',
 #sbox_not_in_live_out,
 ')'
 });

 header=table.concat({-- make a useable header
 _header_make({header,hdr_lvl+1}),
 '\n'..templatestyles..'<div class="div-col">'
 });

 table.sort(sbox_not_in_live_out,comp);
 table.insert(sbox_not_in_live_out,1,header);
 table.insert(out,table.concat(sbox_not_in_live_out,'\n*'));
 table.insert(out,'</div>\n');
 end

 returntable.concat(out);-- concat into a huge string and done
 end


 --[=[--------------------------< H E L P _ T E X T _ C A T S >--------------------------------------------------

 To create category links at the bottom of each error help text section and on the individual error category pages;
 fetches category names from ~/Configuration; replaces this:
 	{{#ifeq:{{FULLPAGENAME}}|Category:CS1 errors: bioRxiv|Category:CS1 errors: bioRxiv|[[:Category:CS1 errors: bioRxiv]]}}
 with this:
 	{{#invoke:Cs1 documentation support|help_text_cats|err_bad_biorxiv}}
 where {{{1}}} is the error_conditions key from Module:Citation/CS1/Configuration

 add |pages=yes to append the number of pages in the category

 ]=]

 localfunctionhelp_text_cats(frame)
 localargs_t=getArgs(frame);
 localerror_conditions_t=cfg.error_conditions;-- get the table of error conditions
 localreplacements_t={};-- table to hold replacement parameters for 1ドル etc placeholders in category names
 fork,vinpairs(args_t)do-- look for |1ドル=<replacement> parameters
 if'string'==type(k)andk:match('^$%d+$')then-- if found
 replacements_t[k]=v;-- save key and value
 end
 end

 ifargs_t[1]anderror_conditions_t[args_t[1]]then-- must have error_condition key and it must exist
 localerror_cat=error_conditions_t[args_t[1]].category;-- get error category from cs1|2 configuration
 iferror_cat:match('$%d')then-- look for placeholders in <error_cat>
 error_cat=error_cat:gsub('$%d',replacements_t)-- replace place holders with matching value from replacements_t
 end

 localtitle_obj=mw.title.getCurrentTitle();-- get a title object for the currently displayed page
 localname_space=title_obj.nsText;
 if('Category'==name_space)and(error_cat==title_obj.text)then-- if this is the category page for the error message
 returntable.concat({'Category:',error_cat});-- no link; just category name
 else-- here when currently displayed page is other than the error message category
 localpages='';-- default empty strin for concatenation
 if'yes'==args_t.pagesthen-- if we should display category page count: TODO: do we need to keep this?
 pages=mw.site.stats.pagesInCategory(error_cat,'all');-- get category page count
 pages=table.concat({' (',mw.language.getContentLanguage():formatNum(pages),' page',(1==pages)and')'or's)'});-- make renderable text
 end
 returntable.concat({'[[:Category:',error_cat,']]',pages});-- link to category with or without page count
 end
 else
 return'<span style="color:#d33">unknown error_conditions key: '..(args_t[1]or'key missing')..'</span>';
 end
 end


 --[[--------------------------< H E L P _ T E X T _ E R R O R _ M E S S A G E >--------------------------------

 to render help text example error messages
 	{{#invoke:Cs1 documentation support|help_text_error_messages|err_bad_biorxiv}}

 assign a single underscore to any of the |$n= parameters to insert an empty string in the error message:
 	{{#invoke:Cs1 documentation support|help_text_error_messages|err_bad_issn|1ドル=_}} -> Check |issn= value
 	{{#invoke:Cs1 documentation support|help_text_error_messages|err_bad_issn|1ドル=e}} -> Check |eissn= value

 error message is rendered at 120% font size; to specify another font size use |size=; must include unit specifier (%, em, etc)

 ]]

 localfunctionhelp_text_error_messages(frame)
 localargs_t=getArgs(frame);
 localerror_conditions=mw.loadData('Module:Citation/CS1/Configuration').error_conditions;
 --	local span_o = '<span class="cs1-visible-error citation-comment">';
 localspan_o='<span class="citation-comment" style="color:#d33; font-size:'..((args_t.sizeandargs_t.size)or'120%')..'">';
 localspan_c='</span>';

 localmessage;
 localout={};-- output goes here

 ifargs_t[1]anderror_conditions[args_t[1]]then-- must have error_condition key and it must exist
 message=error_conditions[args_t[1]].message;
 locali=1;
 localcount;
 localrep;
 repeat
 rep='$'..i
 args_t[rep]=args_t[rep]andargs_t[rep]:gsub('^%s*_%s*$','')ornil;-- replace empty string marker with actual empty string
 message,count=message:gsub(rep,args_t[rep]orrep)
 i=i+1;
 until(0==count);

 table.insert(out,span_o);
 table.insert(out,message);
 table.insert(out,span_c);
 else
 return'<span style="color:#d33">unknown error_conditions key: '..(args_t[1]or'key missing')..'</span>';
 end

 localout_str=table.concat(out);
 returntable.concat({frame:extensionTag('templatestyles','',{src='Module:Citation/CS1/styles.css'}),out_str});
 end


 --[[--------------------------< T E M P L A T E S _ T >--------------------------------------------------------

 This table is a k/v table of sequence tables. The keys in this table are collapsed lowercase form of the cs1|2
 template names ({{ROOTPAGENAME}}):
 	Template:Cite AV media -> citeavmedia

 Each subsequence table holds:
 	[1] documentation page where the TemplateData json is stored ({{cite book}} is the oddball)
 	[2] key to 'preprint_arguments_t' and unique_arguments_t' tables in Module:Citation/CS1/Whitelist; these keys
 		dictate which of the basic or limited arguments and numbered arguments tables will be used to validate
 		the content of the TemplateData

 ]]

 localtemplates_t={
 citearxiv={'Template:Cite_arXiv/doc','arxiv'},-- preprint arguments 
 citeavmedia={'Template:Cite AV media/doc','audio-visual'},-- unique arguments
 citeavmedianotes={'Template:Cite AV media notes/doc'},-- no template data
 citebiorxiv={'Template:Cite bioRxiv/doc','biorxiv'},-- preprint arguments
 citebook={'Template:Cite book/TemplateData'},
 citeciteseerx={'Template:Cite CiteSeerX/doc','citeseerx'},-- no template data; preprint uses limited arguments
 citeconference={'Template:Cite conference/doc','conference'},-- unique arguments
 citedocument={'Template:Cite document/doc','document'},-- special case; uses whitelist.document_parameters_t
 citeencyclopedia={'Template:Cite encyclopedia/doc'},
 citeepisode={'Template:Cite episode/doc','episode'},-- unique arguments
 citeinterview={'Template:Cite interview/doc'},
 citejournal={'Template:Cite journal/doc'},
 citemagazine={'Template:Cite magazine/doc'},
 citemailinglist={'Template:Cite mailing list/doc','mailinglist'},-- unique arguments			-- no template data
 citemap={'Template:Cite map/TemplateData','map'},-- unique arguments
 citemedrxiv={'Template:Cite medRxiv/doc','medrxiv'},-- preprint arguments
 citenews={'Template:Cite news/doc'},
 citenewsgroup={'Template:Cite newsgroup/doc','newsgroup'},-- unique arguments
 citepodcast={'Template:Cite podcast/doc'},
 citepressrelease={'Template:Cite press release/doc'},
 citereport={'Template:Cite report/doc','report'},-- unique arguments
 citeserial={'Template:Cite serial/doc','serial'},-- unique arguments			-- no template data
 citesign={'Template:Cite sign/doc'},
 citespeech={'Template:Cite speech/doc','speech'},-- unique arguments			-- no template data
 citessrn={'Template:Cite SSRN/doc','ssrn'},-- preprint arguments		-- no template data
 citetechreport={'Template:Cite tech report/doc'},
 citethesis={'Template:Cite thesis/doc','thesis'},-- unique arguments
 citeweb={'Template:Cite web/doc'},
 citation={'Template:Citation/doc'},
 }


 --[[--------------------------< N O _ P A G E _ T E M P L A T E S _ T >----------------------------------------


 ]]

 localno_page_templates_t={};


 --[[--------------------------< I D E N T I F I E R _ A L I A S E S _ T >--------------------------------------

 a table of the identifier aliases

 ]]

 localidentifier_aliases_t={}
 foridentifier,handlerinpairs(cfg.id_handlers)do-- for each identifier
 localaliases_t={};-- create a table
 for_,aliasinipairs(handler.parameters)do-- get the alaises
 aliases_t[alias]=true;-- and add them to the table in a form that mimics the whitelist tables
 end
 identifier_aliases_t[identifier:lower()]=aliases_t;-- add new table to the identifier aliases table; use lowercase identifier base name for the key
 end


 --[[--------------------------< T E M P L A T E _ D A T A _ J S O N _ G E T >----------------------------------

 get template doc page content and extract the content of the TemplateData tags (case insensitive)

 <template> is the canonical name of the template doc page (with namespace) that holds the template data; usually
 Template:Cite xxx/doc (except Template:Cite book/TemplateData)

 ]]

 localfunctiontemplate_data_json_get(template)
 localjson=mw.title.new(template):getContent()or'';-- get the content of the article or ''; new pages edited w/ve do not have 'content' until saved; ve does not preview; phab:T221625
 json=json:match('<[Tt]emplate[Dd]ata>(.-)</[Tt]emplate[Dd]ata>');-- remove everything exept the content of the TemplatData tags
 returnjsonandmw.text.jsonDecode(json);-- decode the json string and return as a table; nil if not found
 end


 --[[--------------------------< V A L I D A T E _ D O C U M E N T _ P A R A M >--------------------------------

 looks for <param> (can be the canonical parameter name or can be an alias) in whitelist.document_parameters_t.
 When found, returns true; nil else

 <param> is the parameter's name as listed in the TemplateData

 ]]

 localfunctionvalidate_document_param(param)
 iftrue==whitelist.document_parameters_t[param]then
 returntrue;
 end
 end


 --[[--------------------------< V A L I D A T E _ U N I Q U E _ P A R A M >------------------------------------

 looks for <param> (can be the canonical parameter name or can be an alias) in whitelist.basic_arguments{} and if
 necessary in whitelist.numbered_arguments{}. When found, returns true; nil else

 <param> is the parameter's name as listed in the TemplateData

 ]]

 localfunctionvalidate_basic_param(param)
 iftrue==whitelist.common_parameters_t[param]then
 returntrue;
 end
 end


 --[[--------------------------< V A L I D A T E _ P R E P R I N T _ P A R A M >--------------------------------

 looks for <param> (can be the canonical parameter name or can be an alias) in whitelist.preprint_arguments_t{} or
 whitelist.limited_basic_arguments{} or whitelist.limited_numbered_arguments{}. When found, returns true; nil else

 <param> is the parameter's name as listed in the TemplateData
 <key> is key neccessary to look in the appropriate subtable of whitelist.preprint_arguments_t{}

 ]]

 localfunctionvalidate_preprint_param(param,key)
 iftrue==whitelist.preprint_arguments_t[key][param]or
 true==whitelist.limited_parameters_t[param]then
 --		true == whitelist.limited_basic_arguments_t[param] or
 --		true == whitelist.limited_numbered_arguments_t[param] then
 returntrue;
 end
 end


 --[[--------------------------< V A L I D A T E _ U N I Q U E _ P A R A M >------------------------------------

 looks for <param> (can be the canonical parameter name or can be an alias) in whitelist.unique_arguments_t{} or
 whitelist.basic_arguments{} or whitelist.numbered_arguments{}. When found, returns true; nil else

 <param> is the parameter's name as listed in the TemplateData
 <key> is key neccessary to look in the appropriate subtable of whitelist.unique_arguments_t{}

 ]]

 localfunctionvalidate_unique_param(param,key,cfg_aliases_t)
 iftrue==whitelist.unique_arguments_t[key][param]ortrue==validate_basic_param(param)then
 returntrue;
 end
 end


 --[[--------------------------< V A L I D A T E _ I D _ P A R A M >--------------------------------------------

 looks for <param> <alias> in identifier_aliases_t{}. When found, returns true; nil else

 <param> is the parameter's name as listed in the TemplateData
 <alias> is the alias that we're looking for

 ]]

 localfunctionvalidate_id_alias(param,alias)
 returnidentifier_aliases_t[param]andidentifier_aliases_t[param][alias];
 end


 --[[--------------------------< P A R A M _ E R R O R_ M S G >-------------------------------------------------


 ]]

 localfunctionparam_error_msg(param)
 return'<span style="font-family:"monospace">|'..param..'=</span> is not a valid parameter';
 end


 --[[--------------------------< D U P _ A L I A S _ E R R O R_ M S G >-----------------------------------------


 ]]

 localfunctiondup_alias_error_msg(param,alias)
 return'<span style="font-family:"monospace">|'..param..'=</span> has duplicate aliases: <span font-family:"monospace";>|'..alias..'=</span>';
 end


 --[[--------------------------< D U P _ A L I A S E S _ C H E C K >--------------------------------------------

 create an associative array of <param> aliases. if <alias> already present in <aliases_t> add an error message
 to <out>

 ]]

 localfunctiondup_aliases_check(param,alias,aliases_t,out_t)
 ifnotaliases_t[alias]then
 aliases_t[alias]=true;
 else
 table.insert(out_t,dup_alias_error_msg(param,alias));
 end
 end


 --[[--------------------------< A L I A S _ E R R O R_ M S G >-------------------------------------------------


 ]]

 localfunctionalias_error_msg(param,alias)
 return'<code style="color: inherit; background: inherit; border: none; padding: inherit">|'..alias..'=</code> is not a valid alias of: <code style="color: inherit; background: inherit; border: none; padding: inherit">|'..param..'=</code>';
 end


 --[[--------------------------< C F G _ A L I A S E S _ T _ M A K E >------------------------------------------

 convert this from cfg.aliases{}:
 	['AccessDate'] = {'access-date', 'accessdate'}

 to this in out_t{}
 	['access-date'] = 'AccessDate',
 	['accessdate'] = 'AccessDate',

 to test if |accessdate= is an aliases of |access-date=:
 	if out_t['access-date'] == out_t['accessdate']
 ]]

 localfunctioncfg_aliasts_t_make()
 localout_t={};
 formeta,params_tinpairs(cfg.aliases)do
 if'table'==type(params_t)then-- metaparameters that are assigned string values do not have aliases
 for_,paraminipairs(params_t)do-- for each alias
 param=param:gsub('#','');-- get rid of enumerators
 out_t[param]=meta;-- add it to the output table
 end
 end
 end
 --error (mw.dumpObject (out_t))
 returnout_t;
 end


 --[[--------------------------< T E M P L A T E _ D A T A _ V A L I D A T E >----------------------------------

 compairs parameter names listed in a cs1|2 template's TemplateData structure (everything between <TemplateData>
 and </TemplateData> tag case insensitive). Returns error messages when errors found, empty string else.

 	{{#invoke:Cs1 documentation support|template_data_validate|{{ROOTPAGENAME}}}}

 When called from a different page:
 	{{#invoke:cs1 documentation support|template_data_validate|<canonical template name>}}
 where the <canonical template name> is the template's canonical name with or without namespace and or subpages

 ]]

 localfunctiontemplate_data_validate(frame)
 localargs_t=getArgs(frame);

 ifnotargs_t[1]then
 return'<span style="color:#d33">Error: cs1|2 template name required</span>';
 end

 localtemplate_idx=args_t[1]:lower():match('cit[ae][^/]+');-- args_t[1] has something
 ifnottemplate_idxthen-- but if not a cs1|2 template abandon with error message
 return'<span style="color:#d33">Error: cs1|2 template name required</span>';
 else
 template_idx=template_idx:gsub(' ','');-- is what appears to be a cs1|2 template so strip spaces
 end

 localcfg_aliases_t=cfg_aliasts_t_make();

 localtemplate_t=templates_t[template_idx];
 localout={};

 localtemplate_doc=template_t[1];
 localjson_t=template_data_json_get(template_doc);

 ifnotjson_tthen
 table.insert(out,'Error: can\'t find TemplateData');
 else
 forparam,param_tinpairs(json_t['params'])do
 localparam_i;-- this will be the parameter name that gets validated
 ifparam:find('[Ss]2[Cc][Ii][Dd]')then-- |s2cid*= parameters are not enumerated ...
 param_i=param;-- ... so don't convert the '2' to '#'
 else
 param_i=param:gsub('%d+','#');-- for enumerated parameters, convert the enumerator digits to a single '#' character; all others unmolested
 end

 localparam_is_valid;-- boolean true when param is valid; nil else
 iftemplate_t[2]then-- if template is a preprint or uses unique parameters of 'document' parameters
 if'document'==template_t[2]then-- if a {{cite document}} template
 param_is_valid=validate_document_param(param_i,template_t[2]);
 ifparam_is_validthen
 localaliases_t={};-- used by dup_aliases_check
 ifparam_t['aliases']then
 for_,aliasinipairs(param_t['aliases'])do
 dup_aliases_check(param,alias,aliases_t,out);
 localalias_i=alias:gsub('%d+','#');-- in case an enumerated parameter, convert the enumerator digits to a single '#' character
 ifnotvalidate_document_param(alias_i,template_t[2])then-- is 'alias' a known parameter?
 table.insert(out,alias_error_msg(param,alias));-- may be known but is not supported
 elseifcfg_aliases_t[param_i:gsub('#','')]~=cfg_aliases_t[alias_i:gsub('#','')]then-- is 'alias' known to be an alias of 'param'?
 table.insert(out,alias_error_msg(param,alias));
 end
 end
 end
 else-- here when param not valid preprint param
 table.insert(out,param_error_msg(param))
 end
 elseifwhitelist.preprint_arguments_t[template_t[2]]then-- if a preprint template
 param_is_valid=validate_preprint_param(param_i,template_t[2]);
 ifparam_is_validthen
 localaliases_t={};-- used by dup_aliases_check
 ifparam_t['aliases']then
 for_,aliasinipairs(param_t['aliases'])do
 dup_aliases_check(param,alias,aliases_t,out);
 localalias_i=alias:gsub('%d+','#');-- in case an enumerated parameter, convert the enumerator digits to a single '#' character
 ifnotvalidate_preprint_param(alias_i,template_t[2])then-- is 'alias' a known parameter?
 table.insert(out,alias_error_msg(param,alias));-- may be known but is not supported
 elseifcfg_aliases_t[param_i:gsub('#','')]~=cfg_aliases_t[alias_i:gsub('#','')]then-- is 'alias' known to be an alias of 'param'?
 table.insert(out,alias_error_msg(param,alias));
 end
 end
 end
 else-- here when param not valid preprint param
 table.insert(out,param_error_msg(param))
 end
 elseifwhitelist.unique_arguments_t[template_t[2]]then-- if a unique parameters template
 param_is_valid=validate_unique_param(param_i,template_t[2]);
 ifparam_is_validthen
 localaliases_t={};-- used by dup_aliases_check
 ifparam_t['aliases']then
 for_,aliasinipairs(param_t['aliases'])do
 dup_aliases_check(param,alias,aliases_t,out);
 localalias_i=alias:gsub('%d+','#');-- in case an enumerated parameter, convert the enumerate digits to a single '#' character
 ifnotvalidate_unique_param(alias_i,template_t[2])then-- is 'alias' a known parameter?
 table.insert(out,alias_error_msg(param,alias));
 elseifcfg_aliases_t[param_i:gsub('#','')]~=cfg_aliases_t[alias_i:gsub('#','')]then-- is 'alias' known to be an alias of 'param'?
 table.insert(out,alias_error_msg(param,alias));
 end
 end
 end
 else-- here when param not valid unique parameter
 table.insert(out,param_error_msg(param))
 end
 else-- should never be here if coder is doing the right thing ...
 table.insert(out,'internal error: unexpected keyword in templates_t: '..template_t[2]);
 break;
 end
 else-- here when not unique or preprint
 param_is_valid=validate_basic_param(param_i);
 ifparam_is_validthen
 localaliases_t={};-- used by dup_aliases_check
 ifparam_t['aliases']then
 for_,aliasinipairs(param_t['aliases'])do
 dup_aliases_check(param,alias,aliases_t,out)
 localalias_i=alias:gsub('%d+','#');-- in case an enumerated parameter, convert the enumerate digits to a single '#' character
 ifnotvalidate_basic_param(alias_i)andnotvalidate_id_alias(param,alias)then-- for isbn13 (while still supported) must not mask the digits
 table.insert(out,alias_error_msg(param,alias));
 elseifcfg_aliases_t[param_i:gsub('#','')]~=cfg_aliases_t[alias_i:gsub('#','')]then-- is 'alias' known to be an alias of 'param'?
 table.insert(out,alias_error_msg(param,alias));
 end
 end
 end
 else-- here when param not valid
 table.insert(out,param_error_msg(param))
 end
 end
 end
 end

 ---------- this emits errors when page/pages/at listed in templatedata of templates that don't support those parameters ----------
 --	if json_t then
 --		if ({['citeavmedia']=true, ['citeepisode']=true, ['citemailinglist']=true, ['citenewsgroup']=true, ['citepodcast']=true, ['citeserial']=true, ['citesign']=true, ['citespeech']=true})[template_idx] then
 --			local insource_params_t = {};										-- build sequence of pagination params not supported by these templates
 --			for _, meta_param in ipairs ({'At', 'Page', 'Pages', 'QuotePage', 'QuotePages'}) do
 --				if 'table' == type (cfg.aliases[meta_param]) then
 --					for _, alias in ipairs (cfg.aliases[meta_param]) do			-- metaparameter is a sequence
 --						table.insert (insource_params_t, alias);				-- add the aliases from the metaparameter sequence to the table
 --					end
 --				else															-- metaparameter is plain text
 --					table.insert (insource_params_t, cfg.aliases[meta_param]);	-- add the alias to the table
 --				end
 --			end
 --		
 --			for _, param in ipairs (insource_params_t) do
 --				if json_t.params[param] then
 --					table.insert (out, param_error_msg (param));				-- error; this parameter not supported by this template
 --				end
 --			end
 --		end
 --	end
 ---------- end page/pages/at error detection ----------

 if0~=#outthen
 table.sort(out);
 out[1]='*'..out[1];-- add a splat to the first error message

 --		return table.concat ({'[[' .. template_doc .. ']] TemplateData has errors:<div style="color:#d33; font-style: normal">\n', table.concat (out, '\n*'), '</div>'});
 returntable.concat({
 '[[Template:'..args_t[1]..']] uses ',
 whitelist.preprint_arguments_t[template_t[2]]and'preprint and limited parameter sets'or(whitelist.unique_arguments_t[template_t[2]]and'unique and standard parameter sets'or'standard parameter set'),
 '; TemplateData has errors:\n',
 '<div style="color:#d33; font-style: normal">\n',table.concat(out,'\n*'),'</div>'
 });
 else
 return;-- no errors detected; return nothing
 end
 end


 --[[--------------------------< E R R O R _ C A T _ P A G E _ T A L L Y >--------------------------------------

 loop through Module:Citation/CS1/Configuration error_conditions {} fetching error category names. For each error
 category add the number of pages in the category to the tally. Render the number when done.

 {{#invoke:cs1 documentation support|error_cat_page_tally}}

 ]]

 localfunctionerror_cat_page_tally()
 localerror_conditions_t=cfg.error_conditions;-- get the table of error conditions
 localtally=0;
 localcat_t={};-- some error message share a category; save tallied cats here so we don't recount the already counted
 locali=0;-- number of categories

 fork,v_tinpairs(error_conditions_t)do
 ifk:match('^err')then
 ifnotcat_t[v_t.category]then
 cat_t[v_t.category]=true;
 tally=tally+mw.site.stats.pagesInCategory(v_t.category,'pages');-- get category page count; ignore subcats and files
 i=i+1;
 end
 end
 end

 returnmw.language.getContentLanguage():formatNum(tally)
 end


 --[[--------------------------< M A I N T _ C A T _ P A G E _ T A L L Y >--------------------------------------

 loop through Module:Citation/CS1/Configuration error_conditions {} fetching error category names. For each error
 category add the number of pages in the category to the tally. Render the number when done.

 {{#invoke:cs1 documentation support|maint_cat_page_tally}}

 Dynamic subcats of CS1 maint: DOI inactive not counted because these names come and go as time goes by.

 ]]

 localfunctionmaint_cat_page_tally()
 localerror_conditions_t=cfg.error_conditions;-- get the table of error conditions
 localtally=0;
 localcat_t={};-- some error message share a category; save tallied cats here so we don't recount the already counted
 locali=0;-- number of categories

 fork,v_tinpairs(error_conditions_t)do
 ifnotk:match('^err')then-- if not an error key its a maint key
 ifnotcat_t[v_t.category]then
 cat_t[v_t.category]=true;
 if'maint_mult_names'==kor'maint_numeric_names'==kthen
 localspecial_case_translation_t=cfg.special_case_translation;
 for_,nameinipairs({'AuthorList','ContributorList','EditorList','InterviewerList','TranslatorList'})do
 localcat_name=v_t.category:gsub('1ドル',special_case_translation_t[name]);-- replace 1ドル with translated list name
 tally=tally+mw.site.stats.pagesInCategory(cat_name,'pages');-- get category page count; ignore subcats and files
 i=i+1;
 end
 else
 tally=tally+mw.site.stats.pagesInCategory(v_t.category,'pages');-- get category page count; ignore subcats and files
 i=i+1;
 end
 end
 end
 end

 returnmw.language.getContentLanguage():formatNum(tally)
 end


 --[[--------------------------< U N C A T E G O R I Z E D _ N A M E S P A C E _ L I S T E R >------------------

 For use in the Help:CS1 error §Notes

 Get namespace names and identifiers from MediaWiki. Make a human readable list of namespace names and identifiers
 that cs1|2 does not categorize.

 {{#invoke:cs1 documentation support|uncategorized_namespace_lister}}

 For convenience, 
 {{#invoke:cs1 documentation support|uncategorized_namespace_lister|all=<anything>}}

 returns a list of all namespace names and identifiers used on the current wiki. Any namespace with an
 identifier less than 1, currently Mainspace (0), Special (-1), and Media (-2), is excluded from the list.

 ]]

 localfunctionuncategorized_namespace_lister(frame)
 locallist_t={};
 localfunctioncompare(a,b)-- local function to sort namespaces numerically by the identifiers
 locala_num=tonumber(a:match('%d+'));-- get identifiers and convert to numbers
 localb_num=tonumber(b:match('%d+'));
 returna_num<b_num;-- do the comparison
 end

 fori,_inpairs(mw.site.namespaces)do-- for each namespace in the table
 if''==frame.args.allornotframe.args.allthen-- when |all= not set, make a list of uncategorized namespaces
 ifcfg.uncategorized_namespaces[i]then-- if the identifier is listed in our uncategorized namespace list
 table.insert(list_t,table.concat({mw.site.namespaces[i].name,' (',i,')'}))-- add name and identifier to our local list
 end
 elseif0<ithen-- |all=<anything>: all namespace names and identifiers; ignore identifiers less than 1
 table.insert(list_t,table.concat({'*',mw.site.namespaces[i].name,' (',i,')'}))-- add name and identifier as an unordered list item
 end
 end

 table.sort(list_t,compare);-- ascending numerical sort by identifier

 ifnotframe.args.allthen-- when |all= not set, format list of uncategorized namespaces and identifiers
 list_t[#list_t]='and '..list_t[#list_t];-- add 'and ' to the last name/identifier pair
 returntable.concat(list_t,', ');-- make a big string and done
 else-- make list of all namespaces and identifiers
 returntable.concat(list_t,'\n');-- make a big string and done
 end
 end


 --[[--------------------------< S I N G L E _ L T R _ 2 N D _ L V L _ D O M A I N _ L I S T E R >-------------

 for Help:CS1_errors#bad_url, list the supported top level domains that support single-letter 2nd level names

 	{{#invoke:Module:cs1 documentation support|single_ltr_2nd_lvl_domain_lister}}

 ]]

 localfunctionsingle_ltr_2nd_lvl_domain_lister()
 localout_t={};-- output goes here
 for_,tldinipairs(cfg.single_letter_2nd_lvl_domains_t)do-- fetch each tld
 table.insert(out_t,'.'..tld);-- prefix with a dot and save in out_t{}
 end
 returntable.concat(out_t,', ');-- make a big string and done
 end


 --[[--------------------------< C O D E _ N A M E _ P A I R _ E X I S T S >----------------------------------------

 Returns language code if pair exists, nil if either code doesn't exist or the name doesn't match.
 Intended for use by Template:CS1 language sources/core

 args[1] is language code
 args[2] is language name

 ]]

 localfunctioncode_name_pair_exists(frame)
 localwiki_language=mw.getContentLanguage():getCode()
 localsource_list=mw.language.fetchLanguageNames(wiki_language,'all');
 localcode_list={};
 localname_list={};
 localoverride=cfg.lang_tag_remap;

 forcode,nameinpairs(source_list)do
 add_to_list(code_list,name_list,override,code,name);
 end

 localargs=getArgs(frame);
 locallanguage_code=args[1]
 locallanguage_name=args[2]
 -- Check if the language code exists and the corresponding name matches
 ifcode_list[language_code]==language_namethen
 -- Both code and name are a valid pair
 returnlanguage_code
 else
 ifoverride[language_code]==language_namethen
 -- Code and name are a valid pair found in override table
 returnlanguage_code
 else
 -- Either code doesn't exist or the name doesn't match
 returnnil
 end
 end
 end


 --[[--------------------------< E X P O R T E D F U N C T I O N S >------------------------------------------
 ]]

 return{
 alias_lister=alias_lister,
 alias_names_get=alias_names_get,
 canonical_param_lister=canonical_param_lister,
 canonical_name_get=canonical_name_get,
 cat_lister=cat_lister,
 code_name_pair_exists=code_name_pair_exists,
 error_cat_page_tally=error_cat_page_tally,
 header_make=header_make,
 help_text_cats=help_text_cats,
 help_text_error_messages=help_text_error_messages,
 id_limits_get=id_limits_get,
 is_book_cite_template=is_book_cite_template,
 is_limited_param_template=is_limited_param_template,
 lang_lister=lang_lister,
 maint_cat_page_tally=maint_cat_page_tally,
 script_lang_lister=script_lang_lister,
 single_ltr_2nd_lvl_domain_lister=single_ltr_2nd_lvl_domain_lister,
 template_data_validate=template_data_validate,
 uncategorized_namespace_lister=uncategorized_namespace_lister,
 };

AltStyle によって変換されたページ (->オリジナル) /