Module:WikidataIB and Module:WikidataIB/sandbox1: Difference between pages
(Difference between pages)
imported>Jonesey95 (Undid revision 1059165330 by Neveselbert (talk). Please stop making breaking changes to live templates without discussion. Use a sandbox, or discuss, or both.) |
imported>RexxS (update from main and remove span around pen icon) |
||
Line 1: | Line 1: | ||
-- Version: | -- Version: 2020-09-22 | ||
-- Module to implement use of a blacklist and whitelist for infobox fields | -- Module to implement use of a blacklist and whitelist for infobox fields | ||
-- Can take a named parameter |qid which is the Wikidata ID for the article | -- Can take a named parameter |qid which is the Wikidata ID for the article | ||
Line 14: | Line 14: | ||
local cdate -- initialise as nil and only load _complex_date function if needed | local cdate -- initialise as nil and only load _complex_date function if needed | ||
-- Module:Complex date is loaded lazily and has the following dependencies: | -- [[Module:Complex date]] is loaded lazily and has the following dependencies: | ||
-- Module: | -- Module:I18n/complex date, Module:ISOdate, Module:DateI18n (alternative for Module:Date), | ||
-- Module:Formatnum, Module:I18n/date, Module:Yesno, Module:Linguistic, Module:Calendar | |||
-- Module: | |||
-- The following, taken from https://www.mediawiki.org/wiki/Wikibase/DataModel#Dates_and_times, | -- The following, taken from https://www.mediawiki.org/wiki/Wikibase/DataModel#Dates_and_times, | ||
-- is needed to use Module:Complex date which seemingly requires date precision as a string. | -- is needed to use Module:Complex date which seemingly requires date precision as a string. | ||
Line 574: | Line 565: | ||
-- shortname is boolean switch to use P1813 (short name) instead of label if true. | -- shortname is boolean switch to use P1813 (short name) instead of label if true. | ||
-- lang is the current language code. | -- lang is the current language code. | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- Dependencies: labelOrId(); donotlink[] | -- Dependencies: labelOrId(); donotlink[] | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
local linkedItem = function(id, | local linkedItem = function(id, lprefix, lpostfix, prefix, postfix, dtxt, shortname, lang) | ||
lprefix = lprefix or "" -- toughen against nil values passed | |||
lpostfix = lpostfix or "" | |||
prefix = prefix or "" | |||
postfix = postfix or "" | |||
lang = lang or "en" -- fallback to default if missing | |||
-- see if item might need italics or quotes | -- see if item might need italics or quotes | ||
local fmt = "" | local fmt = "" | ||
for k, v in ipairs( mw.wikibase.getBestStatements(id, "P31") ) do | |||
if v.mainsnak.datavalue and formats[v.mainsnak.datavalue.value.id] then | |||
fmt = formats[v.mainsnak.datavalue.value.id] | |||
break -- pick the first match | |||
end | end | ||
end | end | ||
Line 628: | Line 606: | ||
if sitelink then | if sitelink then | ||
if not (dtxt or shortname) then | if not (dtxt or shortname) then | ||
-- strip any namespace or dab from the sitelink | |||
local pos = sitelink:find(":") or 0 | |||
local slink = sitelink | |||
if pos > 0 then | |||
local prefix = sitelink:sub(1,pos-1) | |||
if mw.site.namespaces[prefix] then -- that prefix is a valid namespace, so remove it | |||
slink = sitelink:sub(pos+1) | |||
end | end | ||
end | |||
-- remove stuff after commas or inside parentheses - ie. dabs | |||
slink = slink:gsub("%s%(.+%)$", ""):gsub(",.+$", "") | |||
-- use that as label, preserving label case - find("^%u") is true for 1st char uppercase | |||
if label:find("^%u") then | |||
label = slink:gsub("^(%l)", string.upper) | |||
else | |||
label = slink:gsub("^(%u)", string.lower) | |||
end | end | ||
end | end | ||
Line 658: | Line 630: | ||
end | end | ||
elseif islabel then | elseif islabel then | ||
-- no sitelink, label exists, so check if a redirect with that title exists | -- no sitelink, label exists, so check if a redirect with that title exists | ||
local artitle = mw.title.new(label, 0) -- only nil if label has invalid chars | |||
if not donotlink[label] and artitle and artitle.redirectTarget then | |||
-- there's a redirect with the same title as the label, so let's link to that | |||
disp = "[[".. lprefix .. label .. lpostfix .. "|" .. prefix .. fmt .. label .. fmt .. postfix .. "]]" | |||
else | |||
-- either (donotlink is true) or (no sitelink, label exists, not redirect) | |||
-- so output unlinked label with italics or quotes as needed | |||
disp = prefix .. fmt .. label .. fmt .. postfix | |||
end -- test if article title exists as redirect on current Wiki | end -- test if article title exists as redirect on current Wiki | ||
else | else | ||
Line 823: | Line 795: | ||
-- createicon assembles the "Edit at Wikidata" pen icon. | -- createicon assembles the "Edit at Wikidata" pen icon. | ||
-- It returns a wikitext string inside a span class="penicon" | -- It returns a wikitext string inside a span class="penicon" | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- Dependencies: i18n[]; | -- Dependencies: i18n[]; | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
local createicon = function(langcode, entityID, propertyID) | local createicon = function(langcode, entityID, propertyID) | ||
local icon = " [[" | |||
-- "<span data-bridge-edit-flow='overwrite' class='penicon'> [[" -> enable Wikidata Bridge | |||
local icon = " | |||
-- " | |||
.. i18n["filespace"] | .. i18n["filespace"] | ||
.. ":OOjs UI icon edit-ltr-progressive.svg |frameless |text-top |10px |alt=" | .. ":OOjs UI icon edit-ltr-progressive.svg |frameless |text-top |10px |alt=" | ||
.. i18n["editonwikidata"] | .. i18n["editonwikidata"] | ||
.. "|link=https://www.wikidata.org/wiki/" .. entityID | .. "|link=https://www.wikidata.org/wiki/" .. entityID | ||
.. "?uselang=" .. langcode | |||
if propertyID | if propertyID then icon = icon .. "#" .. propertyID end | ||
icon = icon .. "|" .. i18n["editonwikidata"] .. "]] | icon = icon .. "|" .. i18n["editonwikidata"] .. "]]" | ||
return icon | return icon | ||
end | end | ||
Line 848: | Line 815: | ||
-- assembleoutput takes the sequence table containing the property values | -- assembleoutput takes the sequence table containing the property values | ||
-- and formats it according to switches given. It returns a string or nil. | -- and formats it according to switches given. It returns a string or nil. | ||
-- It | -- It needs the entityID and propertyID to create a link in the pen icon. | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- Dependencies: parseParam(); | -- Dependencies: parseParam(); | ||
Line 972: | Line 939: | ||
local qnumber = dv.id | local qnumber = dv.id | ||
if linked then | if linked then | ||
val = linkedItem(qnumber, args) | val = linkedItem(qnumber, lpre, lpost, pre, post, dtxt, shortname, args.lang) | ||
else -- no link wanted so check for display-text, otherwise test for lang code | else -- no link wanted so check for display-text, otherwise test for lang code | ||
local label, islabel | local label, islabel | ||
Line 1,280: | Line 1,247: | ||
args.pd = pd | args.pd = pd | ||
-- allow qualifiers to have a different date format; default to year | -- allow qualifiers to have a different date format; default to year | ||
args.qdf = args.qdf or args.qualifierdateformat or args.df or | args.qdf = args.qdf or args.qualifierdateformat or args.df or "y" | ||
local lang = args.lang or | local lang = args.lang or findlang().code | ||
-- qualID is a string list of wanted qualifiers or "ALL" | -- qualID is a string list of wanted qualifiers or "ALL" | ||
Line 1,408: | Line 1,375: | ||
elseif t ~= "" then | elseif t ~= "" then | ||
if qualsonly then | if qualsonly then | ||
out[#out+1] = timestart .. dsep .. timeend | |||
else | else | ||
out[#out] = out[#out] .. " (" .. timestart .. dsep .. timeend .. ")" | out[#out] = out[#out] .. " (" .. timestart .. dsep .. timeend .. ")" | ||
Line 1,494: | Line 1,455: | ||
local uabbr = parseParam(args.unitabbr or args.uabbr, false) | local uabbr = parseParam(args.unitabbr or args.uabbr, false) | ||
local filter = (args.unit or ""):upper() | local filter = (args.unit or ""):upper() | ||
if filter == "" then filter = nil end | if filter == "" then filter = nil end | ||
Line 1,526: | Line 1,486: | ||
return nil | return nil | ||
end -- of check for string | end -- of check for string | ||
end -- of loop through values of propertyID | end -- of loop through values of propertyID | ||
return assembleoutput(out, frame.args, qid, propertyID) | return assembleoutput(out, frame.args, qid, propertyID) | ||
Line 1,569: | Line 1,528: | ||
end | end | ||
for i2, v2 in ipairs(proptbl) do | for i2, v2 in ipairs(proptbl) do | ||
parttbl = v2.qualifiers and v2.qualifiers.P518 | |||
if parttbl then | if parttbl then | ||
-- this higher location has qualifier 'applies to part' (P518) | -- this higher location has qualifier 'applies to part' (P518) | ||
Line 1,601: | Line 1,560: | ||
if #inst > 0 then | if #inst > 0 then | ||
for k, v in ipairs(inst) do | for k, v in ipairs(inst) do | ||
local instid = | local instid = v.mainsnak.datavalue.value.id | ||
-- stop if it's a country (or a country within the United Kingdom if skip is true) | -- stop if it's a country (or a country within the United Kingdom if skip is true) | ||
if instid == "Q6256" or instid == "Q3624078" or (skip and instid == "Q3336843") then | if instid == "Q6256" or instid == "Q3624078" or (skip and instid == "Q3336843") then | ||
Line 1,613: | Line 1,572: | ||
if prop and prop.mainsnak.datavalue then | if prop and prop.mainsnak.datavalue then | ||
if not skip or count == 0 then | if not skip or count == 0 then | ||
out[#out+1] = linkedItem(qid, ":", "", "", "") -- get a linked value if we can | |||
out[#out+1] = linkedItem(qid, | |||
end | end | ||
qid, prevqid = prop.mainsnak.datavalue.value.id, qid | qid, prevqid = prop.mainsnak.datavalue.value.id, qid | ||
Line 1,846: | Line 1,804: | ||
-- getCoords is used to get coordinates for display in an infobox | -- getCoords is used to get coordinates for display in an infobox | ||
-- whitelist and blacklist are implemented | -- whitelist and blacklist are implemented | ||
-- optional 'display' parameter is allowed, defaults to | -- optional 'display' parameter is allowed, defaults to "inline, title" | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- Dependencies: setRanks(); parseInput(); decimalPrecision(); | -- Dependencies: setRanks(); parseInput(); decimalPrecision(); | ||
Line 1,854: | Line 1,812: | ||
-- if there is a 'display' parameter supplied, use it | -- if there is a 'display' parameter supplied, use it | ||
-- otherwise default to | -- otherwise default to "inline, title" | ||
local disp = frame.args.display or "" | local disp = frame.args.display or "" | ||
if disp == "" then | if disp == "" then | ||
disp = | disp = "inline, title" | ||
end | end | ||
Line 1,915: | Line 1,873: | ||
-- whose value is to be returned is passed in named parameter |qual= | -- whose value is to be returned is passed in named parameter |qual= | ||
local qualifierID = frame.args.qual | local qualifierID = frame.args.qual | ||
-- onlysourced is a boolean passed to return qualifiers | -- onlysourced is a boolean passed to return qualifiers | ||
Line 1,949: | Line 1,898: | ||
for k1, v1 in pairs(props) do | for k1, v1 in pairs(props) do | ||
if v1.mainsnak.snaktype == "value" and v1.mainsnak.datavalue.type == "wikibase-entityid" then | if v1.mainsnak.snaktype == "value" and v1.mainsnak.datavalue.type == "wikibase-entityid" then | ||
-- It's a wiki-linked value, so check if it's the target (in propvalue) and if it has qualifiers | -- It's a wiki-linked value, so check if it's the target (in propvalue) | ||
-- and if it has qualifiers | |||
if v1.mainsnak.datavalue.value.id == propvalue and v1.qualifiers then | if v1.mainsnak.datavalue.value.id == propvalue and v1.qualifiers then | ||
if onlysrc == false or sourced(v1) then | if onlysrc == false or sourced(v1) then | ||
-- if we've got this far, we have a (sourced) claim with qualifiers | -- if we've got this far, we have a (sourced) claim with qualifiers | ||
-- which matches the target, so | -- which matches the target, so find the value(s) of the qualifier we want | ||
local quals = v1.qualifiers[qualifierID] | |||
if quals then | |||
-- can't reference qualifer, so set onlysourced = "no" (not boolean) | |||
local qargs = frame.args | |||
qargs.onlysourced = "no" | |||
local vals = propertyvalueandquals(quals, qargs, qid) | |||
for k, v in ipairs(vals) do | |||
out[#out + 1] = v | |||
end | end | ||
end | end | ||
Line 2,146: | Line 2,094: | ||
-- Dependencies: parseParam; setRanks; parseInput; sourced; propertyvalueandquals assembleoutput; | -- Dependencies: parseParam; setRanks; parseInput; sourced; propertyvalueandquals assembleoutput; | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
p. | p.getPropertyIDs = function(frame) | ||
local args = frame.args | |||
args.reqranks = setRanks(args.rank) | args.reqranks = setRanks(args.rank) | ||
args.langobj = findLang(args.lang) | args.langobj = findLang(args.lang) | ||
Line 2,163: | Line 2,112: | ||
local maxvals = tonumber(args.maxvals) or 0 | local maxvals = tonumber(args.maxvals) or 0 | ||
out = {} | |||
for i, v in ipairs(props) do | for i, v in ipairs(props) do | ||
local snak = v.mainsnak | local snak = v.mainsnak | ||
Line 2,177: | Line 2,126: | ||
return assembleoutput(out, args, qid, pid) | return assembleoutput(out, args, qid, pid) | ||
end | end | ||
Line 2,217: | Line 2,161: | ||
qlist = qlist:gsub("[%p%s]+", " ") .. " " | qlist = qlist:gsub("[%p%s]+", " ") .. " " | ||
out = {} | |||
for i, v in ipairs(props) do | for i, v in ipairs(props) do | ||
local snak = v.mainsnak | local snak = v.mainsnak | ||
Line 2,251: | Line 2,195: | ||
-- Dependencies: parseParam; setRanks; parseInput; sourced; propertyvalueandquals assembleoutput; | -- Dependencies: parseParam; setRanks; parseInput; sourced; propertyvalueandquals assembleoutput; | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
p. | p.getPropOfProp = function(frame) | ||
frame.args.reqranks = setRanks(frame.args.rank) | |||
frame.args.langobj = findLang(frame.args.lang) | |||
frame.args.lang = frame.args.langobj.code | |||
local args = frame.args | |||
local pid1 = args.prop1 or args.pid1 or "" | |||
args.langobj = findLang(args.lang) | |||
args.lang = args.langobj.code | |||
local pid1 = args.prop1 or args.pid1 or "" | |||
local pid2 = args.prop2 or args.pid2 or "" | local pid2 = args.prop2 or args.pid2 or "" | ||
local localval = mw.text.trim(args[1] or "") | |||
if pid1 == "" or pid2 == "" then return nil end | if pid1 == "" or pid2 == "" then return nil end | ||
local qid1, statements1 = parseInput(frame, localval, pid1) | |||
if not qid1 then return localval end | |||
local qid1, statements1 = parseInput( | |||
if not qid1 then return | |||
local onlysrc = parseParam(args.onlysourced or args.osd, true) | local onlysrc = parseParam(args.onlysourced or args.osd, true) | ||
local maxvals = tonumber(args.maxvals) or 0 | local maxvals = tonumber(args.maxvals) or 0 | ||
Line 2,310: | Line 2,231: | ||
end -- of loop through values of property1 | end -- of loop through values of property1 | ||
return assembleoutput(out, args, qid1, pid1) | return assembleoutput(out, args, qid1, pid1) | ||
end | end | ||
Line 2,500: | Line 2,411: | ||
if maxvals > 0 and #cat2 >= maxvals then break end | if maxvals > 0 and #cat2 >= maxvals then break end | ||
end | end | ||
out = {} | |||
for k1, v1 in ipairs(cat1) do | for k1, v1 in ipairs(cat1) do | ||
for k2, v2 in ipairs(cat2) do | for k2, v2 in ipairs(cat2) do | ||
Line 2,784: | Line 2,695: | ||
if not (whitelist == 'ALL' or whitelist:find(fieldname)) then return nil end | if not (whitelist == 'ALL' or whitelist:find(fieldname)) then return nil end | ||
local qid = args.qid or "" | local qid = mw.text.trim(args.qid or "") | ||
if qid == "" then qid = | if qid == "" then qid = nil end | ||
local | local entity = mw.wikibase.getEntity(qid) | ||
if not entity then return nil end | |||
local aliases = entity.aliases | |||
if not aliases then return nil end | if not aliases then return nil end | ||
if not qid then qid= mw.wikibase.getEntityIdForCurrentPage() end | |||
args.langobj = findLang(args.lang) | args.langobj = findLang(args.lang) | ||
Line 2,992: | Line 2,905: | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- followQid takes | -- followQid takes three optional parameters: qid, props, and all. | ||
-- If qid is not given, it uses the qid for the connected page | -- If qid is not given, it uses the qid for the connected page | ||
-- or returns nil if there isn't one. | -- or returns nil if there isn't one. | ||
Line 2,998: | Line 2,911: | ||
-- If props is given, the Wikidata item for the qid is examined for each property in turn. | -- If props is given, the Wikidata item for the qid is examined for each property in turn. | ||
-- If that property contains a value that is another Wikibase-item, that item's qid is returned, | -- If that property contains a value that is another Wikibase-item, that item's qid is returned, | ||
-- and the search terminates, unless |all=y when all of the qids are returned, | -- and the search terminates, unless |all=y when all of the qids are returned, sparated by spaces. | ||
-- If props is not given, the qid is returned. | -- If props is not given, the qid is returned. | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- Dependencies: parseParam() | -- Dependencies: parseParam() | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
p. | p.followQid = function(frame) | ||
local qid = (args.qid or ""):upper() | local qid = (frame.args.qid or ""):upper() | ||
local all = parseParam(args.all, false) | local all = parseParam(frame.args.all, false) | ||
if qid == "" then | if qid == "" then | ||
qid = mw.wikibase.getEntityIdForCurrentPage() | qid = mw.wikibase.getEntityIdForCurrentPage() | ||
Line 3,014: | Line 2,924: | ||
if not qid then return nil end | if not qid then return nil end | ||
local out = {} | local out = {} | ||
local props = (args.props or ""):upper() | local props = (frame.args.props or ""):upper() | ||
if props ~= "" then | if props ~= "" then | ||
for p in mw.text.gsplit(props, "%p") do -- split at punctuation and iterate | for p in mw.text.gsplit(props, "%p") do -- split at punctuation and iterate | ||
Line 3,031: | Line 2,941: | ||
end | end | ||
if #out > 0 then | if #out > 0 then | ||
return table.concat(out, " ") | |||
else | else | ||
return qid | return qid | ||
end | end | ||
end | end | ||
Line 3,184: | Line 3,084: | ||
if qid:sub(1,1) ~= "Q" then qid = mw.wikibase.getEntityIdForCurrentPage() end | if qid:sub(1,1) ~= "Q" then qid = mw.wikibase.getEntityIdForCurrentPage() end | ||
if not qid then return nil end | if not qid then return nil end | ||
ranks = setRanks(args.rank) | |||
local stats = {} | local stats = {} | ||
if ranks.b then | if ranks.b then | ||
Line 3,194: | Line 3,094: | ||
if stats[1].mainsnak.datatype == "wikibase-item" then | if stats[1].mainsnak.datatype == "wikibase-item" then | ||
for k, v in pairs( stats ) do | for k, v in pairs( stats ) do | ||
ms = v.mainsnak | |||
if ranks[v.rank:sub(1,1)] and ms.snaktype == "value" and ms.datavalue.value.id == val then | if ranks[v.rank:sub(1,1)] and ms.snaktype == "value" and ms.datavalue.value.id == val then | ||
return val | return val | ||
Line 3,210: | Line 3,110: | ||
-- but it keeps the "edit at Wikidata" pen icon out of the microformat. | -- but it keeps the "edit at Wikidata" pen icon out of the microformat. | ||
-- Usually it will take its url parameter directly from a Wikidata call: | -- Usually it will take its url parameter directly from a Wikidata call: | ||
-- e.g. {{#invoke:WikidataIB |url2 |url={{wdib |P856 |qid=Q23317 |fwd=ALL |osd=no | -- e.g. {{#invoke:WikidataIB |url2 |url={{wdib |P856 |qid=Q23317 |fwd=ALL |osd=no}} | ||
------------------------------------------------------------------------------- | ------------------------------------------------------------------------------- | ||
-- Dependencies: none | -- Dependencies: none | ||
Line 3,217: | Line 3,117: | ||
local txt = frame.args.url or "" | local txt = frame.args.url or "" | ||
if txt == "" then return nil end | if txt == "" then return nil end | ||
local url, icon = txt:match("(.+) (.+)") | local url, icon = txt:match("(.+) (.+)") | ||
url = url or txt | |||
url = | |||
icon = icon or "" | icon = icon or "" | ||
local prot, addr = url:match("(http[s]*://)(.+)") | |||
local prot = url:match("( | prot = prot or url | ||
addr = addr or "" | |||
local disp, n = addr:gsub("%.", "<wbr/>%.") | |||
local disp, n = addr:gsub( | |||
return '<span class="url">[' .. prot .. addr .. " " .. disp .. "]</span> " .. icon | return '<span class="url">[' .. prot .. addr .. " " .. disp .. "]</span> " .. icon | ||
end | end | ||
Line 3,311: | Line 3,201: | ||
local args = frame.args or frame:getParent().args or {} | local args = frame.args or frame:getParent().args or {} | ||
local qid = args.qid | local qid = args.qid | ||
if qid == "" then qid = mw.wikibase. | if qid == "" then qid = nil end | ||
if not | |||
local entity = mw.wikibase.getEntity(qid) | |||
if not entity then return i18n["entity-not-found"] end | |||
local labels = | local labels = entity.labels | ||
if not labels then return i18n["labels-not-found"] end | if not labels then return i18n["labels-not-found"] end | ||
Line 3,336: | Line 3,228: | ||
local args = frame.args or frame:getParent().args or {} | local args = frame.args or frame:getParent().args or {} | ||
local qid = args.qid | local qid = args.qid | ||
if qid == "" then qid = mw.wikibase. | if qid == "" then qid = nil end | ||
if not | |||
local entity = mw.wikibase.getEntity(qid) | |||
if not entity then return i18n["entity-not-found"] end | |||
local descriptions = | local descriptions = entity.descriptions | ||
if not descriptions then return i18n["descriptions-not-found"] end | if not descriptions then return i18n["descriptions-not-found"] end | ||
Line 3,361: | Line 3,255: | ||
local args = frame.args or frame:getParent().args or {} | local args = frame.args or frame:getParent().args or {} | ||
local qid = args.qid | local qid = args.qid | ||
if qid == "" then qid = mw.wikibase. | if qid == "" then qid = nil end | ||
if not | |||
local entity = mw.wikibase.getEntity(qid) | |||
if not entity then return i18n["entity-not-found"] end | |||
local aliases = | local aliases = entity.aliases | ||
if not aliases then return i18n["aliases-not-found"] end | if not aliases then return i18n["aliases-not-found"] end | ||
Line 3,432: | Line 3,328: | ||
local site = args.site or "" | local site = args.site or "" | ||
local showdab = parseParam(args.showdab, true) | local showdab = parseParam(args.showdab, true) | ||
qid = mw.wikibase.getEntityIdForTitle(title, site) | |||
if qid then | if qid then | ||
local prop31 = mw.wikibase.getBestStatements(qid, "P31")[1] | local prop31 = mw.wikibase.getBestStatements(qid, "P31")[1] |