Module:Citation/CS1 explained

require ('strict');

----------------------------< F O R W A R D D E C L A R A T I O N S >--------------------------------------

each of these counts against the Lua upvalue limit

local validation; -- functions in Module:Citation/CS1/Date_validation

local utilities; -- functions in Module:Citation/CS1/Utilitieslocal z = ; -- table of tables in Module:Citation/CS1/Utilities

local identifiers; -- functions and tables in Module:Citation/CS1/Identifierslocal metadata; -- functions in Module:Citation/CS1/COinSlocal cfg = ; -- table of configuration tables that are defined in Module:Citation/CS1/Configurationlocal whitelist = ; -- table of tables listing valid template parameter names; defined in Module:Citation/CS1/Whitelist

--------------------< P A G E S C O P E V A R I A B L E S >---------------

declare variables here that have page-wide scope that are not brought in fromother modules; that are created here and used here

local added_deprecated_cat; -- Boolean flag so that the category is added only oncelocal added_vanc_errs; -- Boolean flag so we only emit one Vancouver error / categorylocal added_generic_name_errs; -- Boolean flag so we only emit one generic name error / category and stop testing names once an error is encounteredlocal added_numeric_name_errs; -- Boolean flag so we only emit one numeric name error / category and stop testing names once an error is encounteredlocal added_numeric_name_maint; -- Boolean flag so we only emit one numeric name maint category and stop testing names once a category has been emittedlocal Frame; -- holds the module's frame tablelocal is_preview_mode; -- true when article is in preview mode; false when using 'Preview page with this template' (previewing the module)local is_sandbox; -- true when using sandbox modules to render citation

----------------------------< F I R S T _ S E T >------------------------------------------------------------

Locates and returns the first set value in a table of values where the order established in the table,left-to-right (or top-to-bottom), is the order in which the values are evaluated. Returns nil if none are set.

This version replaces the original 'for _, val in pairs do' and a similar version that used ipairs. With the pairsversion the order of evaluation could not be guaranteed. With the ipairs version, a nil value would terminatethe for-loop before it reached the actual end of the list.

local function first_set (list, count) local i = 1; while i <= count do -- loop through all items in list if utilities.is_set(list[i]) then return list[i]; -- return the first set list member end i = i + 1; -- point to next endend

----------------------------< A D D _ V A N C _ E R R O R >----------------------------------------------------

Adds a single Vancouver system error message to the template's output regardless of how many error actually exist.To prevent duplication, added_vanc_errs is nil until an error message is emitted.

added_vanc_errs is a Boolean declared in page scope variables above

local function add_vanc_error (source, position) if added_vanc_errs then return end added_vanc_errs = true; -- note that we've added this category utilities.set_message ('err_vancouver',);end

--

local function is_scheme (scheme) return scheme and scheme:match ('^%a[%a%d%+%.%-]*:'); -- true if scheme is set and matches the patternend

--[=[-------------------------< I S _ D O M A I N _ N A M E >-------------------------------------------------- Does this thing that purports to be a domain name seem to be a valid domain name? Syntax defined here: http://tools.ietf.org/html/rfc1034#section-3.5 BNF defined here: https://tools.ietf.org/html/rfc4234 Single character names are generally reserved; see https://tools.ietf.org/html/draft-ietf-dnsind-iana-dns-01#page-15; see also [[Single-letter second-level domain]]list of TLDs: https://www.iana.org/domains/root/db

RFC 952 (modified by RFC 1123) requires the first and last character of a hostname to be a letter or a digit. Betweenthe first and last characters the name may use letters, digits, and the hyphen.

Also allowed are IPv4 addresses. IPv6 not supported

domain is expected to be stripped of any path so that the last character in the last character of the TLD. tldis two or more alpha characters. Any preceding '//' (from splitting a URL with a scheme) will be strippedhere. Perhaps not necessary but retained in case it is necessary for IPv4 dot decimal.

There are several tests: the first character of the whole domain name including subdomains must be a letter or a digit internationalized domain name (ASCII characters with .xn-- ASCII Compatible Encoding (ACE) prefix xn-- in the TLD) see https://tools.ietf.org/html/rfc3490 single-letter/digit second-level domains in the .org, .cash, and .today TLDs q, x, and z SL domains in the .com TLD i and q SL domains in the .net TLD single-letter SL domains in the ccTLDs (where the ccTLD is two letters) two-character SL domains in gTLDs (where the gTLD is two or more letters) three-plus-character SL domains in gTLDs (where the gTLD is two or more letters) IPv4 dot-decimal address format; TLD not allowed

returns true if domain appears to be a proper name and TLD or IPv4 address, else false

]=]

local function is_domain_name (domain) if not domain then return false; -- if not set, abandon end domain = domain:gsub ('^//', ); -- strip '//' from domain name if present; done here so we only have to do it once if not domain:match ('^[%w]') then -- first character must be letter or digit return false; end

if domain:match ('^%a+:') then -- hack to detect things that look like s:Page:Title where Page: is namespace at Wikisource return false; end

local patterns =

for _, pattern in ipairs (patterns) do -- loop through the patterns list if domain:match (pattern) then return true; -- if a match then we think that this thing that purports to be a URL is a URL end end

for _, d in ipairs (cfg.single_letter_2nd_lvl_domains_t) do -- look for single letter second level domain names for these top level domains if domain:match ('%f[%w][%w]%.' .. d) then return true end end return false; -- no matches, we don't know what this thing isend

----------------------------< I S _ U R L >------------------------------------------------------------------

returns true if the scheme and domain parts of a URL appear to be a valid URL; else false.

This function is the last step in the validation process. This function is separate because there are cases thatare not covered by split_url, for example is_parameter_ext_wikilink which is looking for bracketted externalwikilinks.

local function is_url (scheme, domain) if utilities.is_set (scheme) then -- if scheme is set check it and domain return is_scheme (scheme) and is_domain_name (domain); else return is_domain_name (domain); -- scheme not set when URL is protocol-relative endend

--

local function split_url (url_str) local scheme, authority, domain; url_str = url_str:gsub ('([%a%d])%.?[/%?#].*$', '%1'); -- strip FQDN terminator and path(/), query(?), fragment (#) (the capture prevents false replacement of '//')

if url_str:match ('^//%S*') then -- if there is what appears to be a protocol-relative URL domain = url_str:match ('^//(%S*)') elseif url_str:match ('%S-:/*%S+') then -- if there is what appears to be a scheme, optional authority indicator, and domain name scheme, authority, domain = url_str:match ('(%S-:)(/*)(%S+)'); -- extract the scheme, authority indicator, and domain portions if utilities.is_set (authority) then authority = authority:gsub ('//', , 1); -- replace place 1 pair of '/' with nothing; if utilities.is_set(authority) then -- if anything left (1 or 3+ '/' where authority should be) then return scheme; -- return scheme only making domain nil which will cause an error message end else if not scheme:match ('^news:') then -- except for news:..., MediaWiki won't link URLs that do not have authority indicator; TODO: a better way to do this test? return scheme; -- return scheme only making domain nil which will cause an error message end end domain = domain:gsub ('(%a):%d+', '%1'); -- strip port number if present end return scheme, domain;end

--title-link=, |series-link=, |author-link=, etc. for properly formatted content: no wikilinks, no URLs

Link parameters are to hold the title of a Wikipedia article, so none of the WP:TITLESPECIALCHARACTERS are allowed: # < > [] | _except the underscore which is used as a space in wiki URLs and # which is used for section links

returns false when the value contains any of these characters.

When there are no illegal characters, this function returns TRUE if value DOES NOT appear to be a valid URL (the|-link= parameter is ok); else false when value appears to be a valid URL (the |-link= parameter is NOT ok).

local function link_param_ok (value) local scheme, domain; if value:find ('[<>%[%]|]') then -- if any prohibited characters return false; end

scheme, domain = split_url (value); -- get scheme or nil and domain or nil from URL; return not is_url (scheme, domain); -- return true if value DOES NOT appear to be a valid URLend

---link= value and its matching |= value.<p>|<title>= may be wiki-linked but not when |<param>-link= has a value. This function emits an error message whenthat condition exists</p> <p>check <link> for inter-language interwiki-link prefix. prefix must be a MediaWiki-recognized languagecode and must begin with a colon.</p> </p> <p>local function link_title_ok (link, lorig, title, torig) local orig; if utilities.is_set (link) then -- don't bother if <param>-link doesn't have a value if not link_param_ok (link) then -- check |<param>-link= markup orig = lorig; -- identify the failing link parameter elseif title:find ('%[%[') then -- check |title= for wikilink markup orig = torig; -- identify the failing |title= parameter elseif link:match ('^%a+:') then -- if the link is what looks like an interwiki local prefix = link:match ('^(%a+):'):lower; -- get the interwiki prefix if cfg.inter_wiki_map[prefix] then -- if prefix is in the map, must have preceding colon orig = lorig; -- flag as error end end end</p> <p>if utilities.is_set (orig) then link = <i>; -- unset utilities.set_message ('err_bad_paramlink', orig); -- URL or wikilink in |title= with |title-link=; end return link; -- link if ok, empty string elseend</i></p> <p>--</p> <p>local function check_url(url_str) if nil</p> <h2>url_str:match ("^%S+$") then -- if there are any spaces in |url=value it can't be a proper URL return false; end local scheme, domain;</h2> <p>scheme, domain = split_url (url_str); -- get scheme or nil and domain or nil from URL; if 'news:'</p> <h2>scheme then -- special case for newsgroups return domain:match('^[%a%d%+%-_]+%.[%a%d%+%-_%.]*[%a%d%+%-_]$'); end return is_url (scheme, domain); -- return true if value appears to be a valid URLend</h2> <p>--[=[-------------------------< I S _ P A R A M E T E R _ E X T _ W I K I L I N K >---------------------------- Return true if a parameter value has a string that begins and ends with square brackets [ and ] and the firstnon-space characters following the opening bracket appear to be a URL. The test will also find external wikilinksthat use protocol-relative URLs. Also finds bare URLs.</p> <p>The frontier pattern prevents a match on interwiki-links which are similar to scheme:path URLs. The tests thatfind bracketed URLs are required because the parameters that call this test (currently |title=, |chapter=, |work=,and |publisher=) may have wikilinks and there are articles or redirects like '//Hus' so, while uncommon, |title=<a href="../%2f%2fHus/" title="Owela is the Oshiwambo name of a traditional mancala board game played by the Nama people, Herero people, Rukwangali ...">//Hus</a>is possible as might be .</p> <p>]=]</p> <p>local function is_parameter_ext_wikilink (value)local scheme, domain;</p> <p>if value:match ('%f[%[]%[%a%S*:%S+.*%]') then -- if ext. wikilink with scheme and domain: [xxxx://yyyyy.zzz] scheme, domain = split_url (value:match ('%f[%[]%[(%a%S*:%S+).*%]')); elseif value:match ('%f[%[]%[//%S+.*%]') then -- if protocol-relative ext. wikilink: [//yyyyy.zzz] scheme, domain = split_url (value:match ('%f[%[]%[(//%S+).*%]')); elseif value:match ('%a%S*:%S+') then -- if bare URL with scheme; may have leading or trailing plain text scheme, domain = split_url (value:match ('(%a%S*:%S+)')); elseif value:match ('//%S+') then -- if protocol-relative bare URL: //yyyyy.zzz; may have leading or trailing plain text scheme, domain = split_url (value:match ('(//%S+)')); -- what is left should be the domain else return false; -- didn't find anything that is obviously a URL end</p> <p>return is_url (scheme, domain); -- return true if value appears to be a valid URLend</p> <p>---------------------------< C H E C K _ F O R _ U R L >-----------------------------------------------------<p>loop through a list of parameters and their values. Look at the value and if it has an external link, emit an error message.</p> </p> <p>local function check_for_url (parameter_list, error_list) for k, v in pairs (parameter_list) do -- for each parameter in the list if is_parameter_ext_wikilink (v) then -- look at the value; if there is a URL add an error message table.insert (error_list, utilities.wrap_style ('parameter', k)); end endend</p> <p>----------------------------< S A F E _ F O R _ U R L >------------------------------------------------------<p>Escape sequences for content that will be used for URL descriptions</p> </p> <p>local function safe_for_url(str) if str:match("%[%[.-%]%]") ~= nil then utilities.set_message ('err_wikilink_in_url',); end return str:gsub('[%[%]\n]',);end</p> <p>----------------------------< E X T E R N A L _ L I N K >----------------------------------------------------<p>Format an external link with error checking</p> </p> <p>local function external_link (URL, label, source, access) local err_msg = <i>; local domain; local path; local base_url;</i></p> <p>if not utilities.is_set (label) then label = URL; if utilities.is_set (source) then utilities.set_message ('err_bare_url_missing_title',); else error (cfg.messages["bare_url_no_origin"]); -- programmer error; valid parameter name does not have matching meta-parameter end end if not check_url (URL) then utilities.set_message ('err_bad_url',); end domain, path = URL:match ('^([/%.%-%+:%a%d]+)([/%?#].*)$'); -- split the URL into scheme plus domain and path if path then -- if there is a path portion path = path:gsub ('[%[%]]',); -- replace '[' and ']' with their percent-encoded values URL = table.concat ; -- and reassemble end</p> <p>base_url = table.concat ; -- assemble a wiki-markup URL</p> <p>if utilities.is_set (access) then -- access level (subscription, registration, limited) base_url = utilities.substitute (cfg.presentation['ext-link-access-signal'],); -- add the appropriate icon end</p> <p>return base_url;end</p> <p>----------------------------< D E P R E C A T E D _ P A R A M E T E R >--------------------------------------<p>Categorize and emit an error message when the citation contains one or more deprecated parameters. The function includes theoffending parameter name to the error message. Only one error message is emitted regardless of the number of deprecatedparameters in the citation.</p> <p>added_deprecated_cat is a Boolean declared in page scope variables above</p> </p> <p>local function deprecated_parameter(name) if not added_deprecated_cat then added_deprecated_cat = true; -- note that we've added this category utilities.set_message ('err_deprecated_params',); -- add error message endend</p> <p>--[=[-------------------------< K E R N _ Q U O T E S >-------------------------------------------------------- Apply kerning to open the space between the quote mark provided by the module and a leading or trailing quote mark contained in a |title= or |chapter= parameter's value. This function will positive kern either single or double quotes: "'Unkerned title with leading and trailing single quote marks'" " 'Kerned title with leading and trailing single quote marks' " (in real life the kerning isn't as wide as this example) Double single quotes (italic or bold wiki-markup) are not kerned. Replaces Unicode quote marks in plain text or in the label portion of a [[L|D]] style wikilink with typewriterquote marks regardless of the need for kerning. Unicode quote marks are not replaced in simple <a href="../D/" title="D is the fourth letter of the Latin alphabet, used in the modern English alphabet, the alphabets of other western European ...">D</a> wikilinks.</p> <p>Call this function for chapter titles, for website titles, etc.; not for book titles.</p> <p>]=]</p> <p>local function kern_quotes (str) local cap = <i>; local wl_type, label, link;</i></p> <p>wl_type, label, link = utilities.is_wikilink (str); -- wl_type is: 0, no wl (text in label variable); 1, <a href="../D/" title="D is the fourth letter of the Latin alphabet, used in the modern English alphabet, the alphabets of other western European ...">D</a>; 2, <a href="../L/" title="L is the twelfth letter of the Latin alphabet, used in the modern English alphabet, the alphabets of other western European ...">D</a> if 1</p> <h2>wl_type then -- <a href="../D/" title="D is the fourth letter of the Latin alphabet, used in the modern English alphabet, the alphabets of other western European ...">D</a> simple wikilink with or without quote marks if mw.ustring.match (str, '%[%[[\"“”\'‘’].+[\"“”\'‘’]%]%]') then -- leading and trailing quote marks str = utilities.substitute (cfg.presentation['kern-left'], str); str = utilities.substitute (cfg.presentation['kern-right'], str); elseif mw.ustring.match (str, '%[%[[\"“”\'‘’].+%]%]') then -- leading quote marks str = utilities.substitute (cfg.presentation['kern-left'], str); elseif mw.ustring.match (str, '%[%[.+[\"“”\'‘’]%]%]') then -- trailing quote marks str = utilities.substitute (cfg.presentation['kern-right'], str); end</h2> <p>else -- plain text or <a href="../L/" title="L is the twelfth letter of the Latin alphabet, used in the modern English alphabet, the alphabets of other western European ...">D</a>; text in label variable label = mw.ustring.gsub (label, '[“”]', '\"'); -- replace “” (U+201C & U+201D) with " (typewriter double quote mark) label = mw.ustring.gsub (label, '[‘’]', '\<i>); -- replace ‘’ (U+2018 & U+2019) with ' (typewriter single quote mark)</i></p> <p>cap = mw.ustring.match (label, "^([\"\'][^\'].+)"); -- match leading double or single quote but not doubled single quotes (italic markup) if utilities.is_set (cap) then label = utilities.substitute (cfg.presentation['kern-left'], cap); end cap = mw.ustring.match (label, "^(.+[^\'][\"\'])$") -- match trailing double or single quote but not doubled single quotes (italic markup) if utilities.is_set (cap) then label = utilities.substitute (cfg.presentation['kern-right'], cap); end if 2</p> <h2>wl_type then str = utilities.make_wikilink (link, label); -- reassemble the wikilink else str = label; end end return str;end</h2> <p>--script-title= holds title parameters that are not written in Latin-based scripts: Chinese, Japanese, Arabic, Hebrew, etc. These scripts shouldnot be italicized and may be written right-to-left. The value supplied by |script-title= is concatenated onto Title after Title has been wrappedin italic markup.<p>Regardless of language, all values provided by |script-title= are wrapped in <bdi>...</bdi> tags to isolate RTL languages from the English left to right.</p> <p>|script-title= provides a unique feature. The value in |script-title= may be prefixed with a two-character ISO 639-1 language code and a colon: |script-title=ja:*** *** (where * represents a Japanese character)Spaces between the two-character code and the colon and the colon and the first script character are allowed: |script-title=ja : *** *** |script-title=ja: *** *** |script-title=ja :*** ***Spaces preceding the prefix are allowed: |script-title = ja:*** ***</p> <p>The prefix is checked for validity. If it is a valid ISO 639-1 language code, the lang attribute (lang="ja") is added to the <bdi> tag so that browsers canknow the language the tag contains. This may help the browser render the script more correctly. If the prefix is invalid, the lang attributeis not added. At this time there is no error message for this condition.</p> <p>Supports |script-title=, |script-chapter=, |script-<periodical>=</p> </p> <p>local function format_script_value (script_value, script_param) local lang=<i>; -- initialize to empty string local name; if script_value:match('^%l%l%l?%s*:') then -- if first 3 or 4 non-space characters are script language prefix lang = script_value:match('^(%l%l%l?)%s*:%s*%S.*'); -- get the language prefix or nil if there is no script if not utilities.is_set (lang) then utilities.set_message ('err_script_parameter',); -- prefix without 'title'; add error message return </i>; -- script_value was just the prefix so return empty string end -- if we get this far we have prefix and script name = cfg.lang_tag_remap[lang] or mw.language.fetchLanguageName(lang, cfg.this_wiki_code); -- get language name so that we can use it to categorize if utilities.is_set (name) then -- is prefix a proper ISO 639-1 language code? script_value = script_value:gsub ('^%l+%s*:%s*', <i>); -- strip prefix from script -- is prefix one of these language codes? if utilities.in_array (lang, cfg.script_lang_codes) then utilities.add_prop_cat ('script',) else utilities.set_message ('err_script_parameter',); -- unknown script-language; add error message end lang = ' lang="' .. lang .. '" '; -- convert prefix into a lang attribute else utilities.set_message ('err_script_parameter',); -- invalid language code; add error message lang = </i>; -- invalid so set lang to empty string end else utilities.set_message ('err_script_parameter',); -- no language code prefix; add error message end script_value = utilities.substitute (cfg.presentation['bdi'], <i></i>); -- isolate in case script is RTL</p> <p>return script_value;end</p> <p>--title= and |script-title=, this function concatenates those two parameter values after the scriptvalue has been wrapped in <bdi> tags.</p> <p>local function script_concatenate (title, script, script_param) if utilities.is_set (script) then script = format_script_value (script, script_param); -- <bdi> tags, lang attribute, categorization, etc.; returns empty string on error if utilities.is_set (script) then title = title .. ' ' .. script; -- concatenate title and script title end end return title;end</p> <p>----------------------------< W R A P _ M S G >--------------------------------------------------------------<p>Applies additional message text to various parameter values. Supplied string is wrapped using a message_listconfiguration taking one argument. Supports lower case text for templates. Additional text takenfrom citation_config.messages - the reason this function is similar to but separate from wrap_style.</p> </p> <p>local function wrap_msg (key, str, lower) if not utilities.is_set (str) then return ""; end if true</p> <h2>lower then local msg; msg = cfg.messages[key]:lower; -- set the message to lower case before return utilities.substitute (msg, str); -- including template text else return utilities.substitute (cfg.messages[key], str); end end</h2> <p>--chapter= (or aliases) or |title= or |title-link=</p> <p>local function wikisource_url_make (str) local wl_type, D, L; local ws_url, ws_label; local wikisource_prefix = table.concat ;</p> <p>wl_type, D, L = utilities.is_wikilink (str); -- wl_type is 0 (not a wikilink), 1 (simple wikilink), 2 (complex wikilink)</p> <p>if 0</p> <h2>wl_type then -- not a wikilink; might be from |title-link= str = D:match ('^[Ww]ikisource:(.+)') or D:match ('^[Ss]:(.+)'); -- article title from interwiki link with long-form or short-form namespace if utilities.is_set (str) then ws_url = table.concat ; ws_label = str; -- label for the URL end elseif 1</h2> <p>wl_type then -- simple wikilink: str = D:match ('^[Ww]ikisource:(.+)') or D:match ('^[Ss]:(.+)'); -- article title from interwiki link with long-form or short-form namespace if utilities.is_set (str) then ws_url = table.concat ; ws_label = str; -- label for the URL end elseif 2</p> <h2>wl_type then -- non-so-simple wikilink: displayed text (<a href="../L/" title="L is the twelfth letter of the Latin alphabet, used in the modern English alphabet, the alphabets of other western European ...">D</a>) str = L:match ('^[Ww]ikisource:(.+)') or L:match ('^[Ss]:(.+)'); -- article title from interwiki link with long-form or short-form namespace if utilities.is_set (str) then ws_label = D; -- get ws article name from display portion of interwiki link ws_url = table.concat ; end end</h2> <p>if ws_url then ws_url = mw.uri.encode (ws_url, 'WIKI'); -- make a usable URL ws_url = ws_url:gsub ('%%23', '#'); -- undo percent-encoding of fragment marker end</p> <p>return ws_url, ws_label, L or D; -- return proper URL or nil and a label or nilend</p> <p>--script-<periodical>=, |<periodical>=,and |trans-<periodical>= into a single Periodical meta-parameter.</p> <p>local function format_periodical (script_periodical, script_periodical_source, periodical, trans_periodical)</p> <p>if not utilities.is_set (periodical) then periodical = <i>; -- to be safe for concatenation else periodical = utilities.wrap_style ('italic-title', periodical); -- style end</i></p> <p>periodical = script_concatenate (periodical, script_periodical, script_periodical_source); -- <bdi> tags, lang attribute, categorization, etc.; must be done after title is wrapped</p> <p>if utilities.is_set (trans_periodical) then trans_periodical = utilities.wrap_style ('trans-italic-title', trans_periodical); if utilities.is_set (periodical) then periodical = periodical .. ' ' .. trans_periodical; else -- here when trans-periodical without periodical or script-periodical periodical = trans_periodical; utilities.set_message ('err_trans_missing_title',); end end</p> <p>return periodical;end</p> <p>--script-chapter=, |chapter=, |trans-chapter=,and |chapter-url= into a single chapter meta- parameter (chapter_url_source usedfor error messages).</p> <p>local function format_chapter_title (script_chapter, script_chapter_source, chapter, chapter_source, trans_chapter, trans_chapter_source, chapter_url, chapter_url_source, no_quotes, access) local ws_url, ws_label, L = wikisource_url_make (chapter); -- make a wikisource URL and label from a wikisource interwiki link if ws_url then ws_label = ws_label:gsub ('_', ' '); -- replace underscore separators with space characters chapter = ws_label; end</p> <p>if not utilities.is_set (chapter) then chapter = <i>; -- to be safe for concatenation else if false </i></p> <h2>no_quotes then chapter = kern_quotes (chapter); -- if necessary, separate chapter title's leading and trailing quote marks from module provided quote marks chapter = utilities.wrap_style ('quoted-title', chapter); end end</h2> <p>chapter = script_concatenate (chapter, script_chapter, script_chapter_source); -- <bdi> tags, lang attribute, categorization, etc.; must be done after title is wrapped</p> <p>if utilities.is_set (chapter_url) then chapter = external_link (chapter_url, chapter, chapter_url_source, access); -- adds bare_url_missing_title error if appropriate elseif ws_url then chapter = external_link (ws_url, chapter .. ' ', 'ws link in chapter'); -- adds bare_url_missing_title error if appropriate; space char to move icon away from chap text; TODO: better way to do this? chapter = utilities.substitute (cfg.presentation['interwiki-icon'],); end</p> <p>if utilities.is_set (trans_chapter) then trans_chapter = utilities.wrap_style ('trans-quoted-title', trans_chapter); if utilities.is_set (chapter) then chapter = chapter .. ' ' .. trans_chapter; else -- here when trans_chapter without chapter or script-chapter chapter = trans_chapter; chapter_source = trans_chapter_source:match ('trans%-?(.+)'); -- when no chapter, get matching name from trans-<param> utilities.set_message ('err_trans_missing_title',); end end</p> <p>return chapter;end</p> <p>------------------< H A S _ I N V I S I B L E _ C H A R S >-------------------<p>This function searches a parameter's value for non-printable or invisible characters.The search stops at the first match.</p> <p>This function will detect the visible replacement character when it is part of the Wikisource.</p> <p>Detects but ignores nowiki and math stripmarkers. Also detects other named stripmarkers(gallery, math, pre, ref) and identifies them with a slightly different error message.See also coins_cleanup.</p> <p>Output of this function is an error message that identifies the character or theUnicode group, or the stripmarker that was detected along with its position (or,for multi-byte characters, the position of its first byte) in the parameter value.</p> </p> <p>local function has_invisible_chars (param, v) local position = <i>; -- position of invisible char or starting position of stripmarker local capture; -- used by stripmarker detection to hold name of the stripmarker local stripmarker; -- boolean set true when a stripmarker is found</i></p> <p>capture = string.match (v, '[%w%p ]*'); -- test for values that are simple ASCII text and bypass other tests if true if capture</p> <h2>v then -- if same there are no Unicode characters return; end</h2> <p>for _, invisible_char in ipairs (cfg.invisible_chars) do local char_name = invisible_char[1]; -- the character or group name local pattern = invisible_char[2]; -- the pattern used to find it position, _, capture = mw.ustring.find (v, pattern); -- see if the parameter value contains characters that match the pattern if position and (cfg.invisible_defs.zwj</p> <h2>capture) then -- if we found a zero-width joiner character if mw.ustring.find (v, cfg.indic_script) then -- it's ok if one of the Indic scripts position = nil; -- unset position elseif cfg.emoji_t[mw.ustring.codepoint (v, position+1)] then -- is zwj followed by a character listed in emoji? position = nil; -- unset position end end if position then if 'nowiki'</h2> <p>capture or 'math'</p> <h2>capture or -- nowiki and math stripmarkers (not an error condition) ('templatestyles'</h2> <p>capture and utilities.in_array (param,)) then -- templatestyles stripmarker allowed in these parameters stripmarker = true; -- set a flag elseif true</p> <h2>stripmarker and cfg.invisible_defs.del</h2> <p>capture then -- because stripmakers begin and end with the delete char, assume that we've found one end of a stripmarker position = nil; -- unset else local err_msg; if capture and not (cfg.invisible_defs.del</p> <h2>capture or cfg.invisible_defs.zwj</h2> <p>capture) then err_msg = capture .. ' ' .. char_name; else err_msg = char_name .. ' ' .. 'character'; end</p> <p>utilities.set_message ('err_invisible_char',); -- add error message return; -- and done with this parameter end end endend</p> <p>---------------------< A R G U M E N T _ W R A P P E R >----------------------<p>Argument wrapper. This function provides support for argument mapping definedin the configuration file so that multiple names can be transparently aliased tosingle internal variable.</p> </p> <p>local function argument_wrapper (args) local origin = ; return setmetatable;end</p> <p>--</p> <p>local function nowrap_date (date) local cap = <i>; local cap2 = </i>;</p> <p>if date:match("^%d%d%d%d%-%d%d%-%d%d$") then date = utilities.substitute (cfg.presentation['nowrap1'], date); elseif date:match("^%a+%s*%d%d?,%s+%d%d%d%d$") or date:match ("^%d%d?%s*%a+%s+%d%d%d%d$") then cap, cap2 = string.match (date, "^(.*)%s+(%d%d%d%d)$"); date = utilities.substitute (cfg.presentation['nowrap2'],); end return date;end</p> <p>--type=<default value>) for those templates that have defaults. Also handles thespecial case where it is desirable to omit the title type from the rendered citation(|type=none).</p> <p>local function set_titletype (cite_class, title_type) if utilities.is_set (title_type) then if 'none'</p> <h2>cfg.keywords_xlate[title_type] then title_type = <i>; -- if |type=none then type parameter not displayed end return title_type; -- if |type= has been set to any other value use that value end</i></h2> <p>return cfg.title_types [cite_class] or <i>; -- set template's default title type; else empty string for concatenationend</i></p> <p>----------------------------< S A F E _ J O I N >-----------------------------<p>Joins a sequence of strings together while checking for duplicate separation characters.</p> </p> <p>local function safe_join(tbl, duplicate_char) local f = ; -- create a function table appropriate to type of 'duplicate character' if 1</p> <h2>#duplicate_char then -- for single byte ASCII characters use the string library functions f.gsub = string.gsub f.match = string.match f.sub = string.sub else -- for multi-byte characters use the ustring library functions f.gsub = mw.ustring.gsub f.match = mw.ustring.match f.sub = mw.ustring.sub end</h2> <p>local str = <i>; -- the output string local comp = </i>; -- what does 'comp' mean? local end_chr = <i>; local trim; for _, value in ipairs(tbl) do if value </i></p> <h2>nil then value = <i>; end if str </i></h2> <p><i>then -- if output string is empty str = value; -- assign value to it (first time through the loop) elseif value ~= </i> then if value:sub(1, 1)</p> <h2>'<' then -- special case of values enclosed in spans and other markup. comp = value:gsub("%b<>", ""); -- remove HTML markup (</h2> <span>string</span> -> string) else comp = value; end -- typically duplicate_char is sepc if f.sub(comp, 1, 1) <h2>duplicate_char then -- is first character same as duplicate_char? why test first character? -- Because individual string segments often (always?) begin with terminal punct for the -- preceding segment: 'First element' .. 'sepc next element' .. etc.? trim = false; end_chr = f.sub(str, -1, -1); -- get the last character of the output string -- str = str .. "<HERE(enchr=" .. end_chr .. ")" -- debug stuff? if end_chr</h2> <p>duplicate_char then -- if same as separator str = f.sub(str, 1, -2); -- remove it elseif end_chr</p> <h2>"'" then -- if it might be wiki-markup if f.sub(str, -3, -1)</h2> <p>duplicate_char .. "<i>" then -- if last three chars of str are sepc</i> str = f.sub(str, 1, -4) .. "<i>"; -- remove them and add back </i> elseif f.sub(str, -5, -1)</p> <h2>duplicate_char .. "]]<i>" then -- if last five chars of str are sepc]]</i> trim = true; -- why? why do this and next differently from previous? elseif f.sub(str, -4, -1)</h2> <p>duplicate_char .. "]<i>" then -- if last four chars of str are sepc]</i> trim = true; -- same question end elseif end_chr</p> <h2>"]" then -- if it might be wiki-markup if f.sub(str, -3, -1)</h2> <p>duplicate_char .. "]]" then -- if last three chars of str are sepc]] wikilink trim = true; elseif f.sub(str, -3, -1)</p> <h2>duplicate_char .. '"]' then -- if last three chars of str are sepc"] quoted external link trim = true; elseif f.sub(str, -2, -1)</h2> <p>duplicate_char .. "]" then -- if last two chars of str are sepc] external link trim = true; elseif f.sub(str, -4, -1)</p> <h2>duplicate_char .. "<i>]" then -- normal case when |url=something & |title=Title. trim = true; end elseif end_chr </i></h2> <p>" " then -- if last char of output string is a space if f.sub(str, -2, -1)</p> <h2>duplicate_char .. " " then -- if last two chars of str are <sepc><space> str = f.sub(str, 1, -3); -- remove them both end end</h2> <p>if trim then if value ~= comp then -- value does not equal comp when value contains HTML markup local dup2 = duplicate_char; if f.match(dup2, "%A") then dup2 = "%" .. dup2; end -- if duplicate_char not a letter then escape it value = f.gsub(value, "(%b<>)" .. dup2, "%1", 1) -- remove duplicate_char if it follows HTML markup else value = f.sub(value, 2, -1); -- remove duplicate_char when it is first character end end end str = str .. value; -- add it to the output string end end return str;end</p> <p>----------------------------< I S _ S U F F I X >-----------------------------<p>returns true if suffix is properly formed Jr, Sr, or ordinal in the range 1–9.Puncutation not allowed.</p> </p> <p>local function is_suffix (suffix) if utilities.in_array (suffix,) or suffix:match ('^%dth$') then return true; end return false;end</p> <p>--first= and |last= names to contain any of the letters definedin the four Unicode Latin character sets <a href="http://www.unicode.org/charts/PDF/U0000.pdf" rel="nofollow">C0 Controls and Basic Latin</a> 0041–005A, 0061–007A <a href="http://www.unicode.org/charts/PDF/U0080.pdf" rel="nofollow">C1 Controls and Latin-1 Supplement</a> 00C0–00D6, 00D8–00F6, 00F8–00FF <a href="http://www.unicode.org/charts/PDF/U0100.pdf" rel="nofollow">Latin Extended-A</a> 0100–017F <a href="http://www.unicode.org/charts/PDF/U0180.pdf" rel="nofollow">Latin Extended-B</a> 0180–01BF, 01C4–024F<p>|lastn= also allowed to contain hyphens, spaces, and apostrophes. (http://www.ncbi.nlm.nih.gov/books/NBK7271/box/A35029/)|firstn= also allowed to contain hyphens, spaces, apostrophes, and periods</p> <p>This original test: if nil</p> <h2>mw.ustring.find (last, "^[A-Za-zÀ-ÖØ-öø-ƿDŽ-ɏ%-%s%']*$") or nil</h2> <p>mw.ustring.find (first, "^[A-Za-zÀ-ÖØ-öø-ƿDŽ-ɏ%-%s%'%.]+[2-6%a]*$") thenwas written outside of the code editor and pasted here because the code editorgets confused between character insertion point and cursor position. The test hasbeen rewritten to use decimal character escape sequence for the individual bytesof the Unicode characters so that it is not necessary to use an external editorto maintain this code.</p> <p>\195\128-\195\150 – À-Ö (U+00C0–U+00D6 – C0 controls) \195\152-\195\182 – Ø-ö (U+00D8-U+00F6 – C0 controls) \195\184-\198\191 – ø-ƿ (U+00F8-U+01BF – C0 controls, Latin extended A & B) \199\132-\201\143 – DŽ-ɏ (U+01C4-U+024F – Latin extended B)</p> </p> <p>local function is_good_vanc_name (last, first, suffix, position) if not suffix then if first:find ('[,%s]') then -- when there is a space or comma, might be first name/initials + generational suffix first = first:match ('(.-)[,%s]+'); -- get name/initials suffix = first:match ('[,%s]+(.+)$'); -- get generational suffix end end if utilities.is_set (suffix) then if not is_suffix (suffix) then add_vanc_error (cfg.err_msg_supl.suffix, position); return false; -- not a name with an appropriate suffix end end if nil</p> <h2>mw.ustring.find (last, "^[A-Za-z\195\128-\195\150\195\152-\195\182\195\184-\198\191\199\132-\201\143\225\184\128-\225\187\191%-%s%']*$") or nil</h2> <p>mw.ustring.find (first, "^[A-Za-z\195\128-\195\150\195\152-\195\182\195\184-\198\191\199\132-\201\143\225\184\128-\225\187\191%-%s%'%.]*$") then add_vanc_error (cfg.err_msg_supl['non-Latin char'], position); return false; -- not a string of Latin characters; Vancouver requires Romanization end; return true;end</p> <p>--name-list-style=vanc. <p>Names in |firstn= may be separated by spaces or hyphens, or for initials, a period.See http://www.ncbi.nlm.nih.gov/books/NBK7271/box/A35062/.</p> <p>Vancouver style requires family rank designations (Jr, II, III, etc.) to be renderedas Jr, 2nd, 3rd, etc. See http://www.ncbi.nlm.nih.gov/books/NBK7271/box/A35085/.This code only accepts and understands generational suffix in the Vancouver formatbecause Roman numerals look like, and can be mistaken for, initials.</p> <p>This function uses ustring functions because firstname initials may be any of theUnicode Latin characters accepted by is_good_vanc_name .</p> </p> <p>local function reduce_to_initials (first, position) if first:find (',', 1, true) then return first; -- commas not allowed; abandon end</p> <p>local name, suffix = mw.ustring.match (first, "^(%u+) ([%dJS][%drndth]+)$");</p> <p>if not name then -- if not initials and a suffix name = mw.ustring.match (first, "^(%u+)$"); -- is it just initials? end</p> <p>if name then -- if first is initials with or without suffix if 3 > mw.ustring.len (name) then -- if one or two initials if suffix then -- if there is a suffix if is_suffix (suffix) then -- is it legitimate? return first; -- one or two initials and a valid suffix so nothing to do else add_vanc_error (cfg.err_msg_supl.suffix, position); -- one or two initials with invalid suffix so error message return first; -- and return first unmolested end else return first; -- one or two initials without suffix; nothing to do end end end -- if here then name has 3 or more uppercase letters so treat them as a word</p> <p>local initials_t, names_t =, ; -- tables to hold name parts and initials local i = 1; -- counter for number of initials</p> <p>names_t = mw.text.split (first, '[%s%-]+'); -- split into a sequence of names and possible suffix</p> <p>while names_t[i] do -- loop through the sequence if 1 < i and names_t[i]:match ('[%dJS][%drndth]+%.?$') then -- if not the first name, and looks like a suffix (may have trailing dot) names_t[i] = names_t[i]:gsub ('%.', <i>); -- remove terminal dot if present if is_suffix (names_t[i]) then -- if a legitimate suffix table.insert (initials_t, ' ' .. names_t[i]); -- add a separator space, insert at end of initials sequence break; -- and done because suffix must fall at the end of a name end -- no error message if not a suffix; possibly because of Romanization end if 3 > i then table.insert (initials_t, mw.ustring.sub (names_t[i], 1, 1)); -- insert the initial at end of initials sequence end i = i + 1; -- bump the counter end return table.concat (initials_t); -- Vancouver format does not include spaces.end</i></p> <p>--</p> <p>local function interwiki_prefixen_get (value, is_link) if not value:find (':%l+:') then -- if no prefix return false; -- abandon; boolean here to distinguish from nil fail returns later end</p> <p>local prefix_patterns_linked_t =</p><div class="Footer"> <div class="robots-noindex"><p>This article is licensed under the <a href="http://www.gnu.org/copyleft/fdl.html" rel="nofollow">GNU Free Documentation License</a>. It uses material from the Wikipedia article "<a href="http://en.wikipedia.org/wiki/Module%3aCitation%2fCS1" rel="nofollow">Module:Citation/CS1</a>".</p></div> <p>Except where otherwise indicated, Everything.Explained.Today is © Copyright 2009-2024, A B Cryer, All Rights Reserved. <a href="http://explained.today/cookie_policy.htm">Cookie policy</a>.</p> </div> </div> </body> </html>