모듈:Pipeline: 두 판 사이의 차이
잔글편집 요약 없음 |
편집 요약 없음 |
||
(같은 사용자의 중간 판 8개는 보이지 않습니다) | |||
7번째 줄: | 7번째 줄: | ||
local passTagPattern = 'pipe%-passed%-data%-%-' .. passRandStr | local passTagPattern = 'pipe%-passed%-data%-%-' .. passRandStr | ||
local passPattern = '<' .. passTagPattern .. '>(.-)</' .. passTagPattern.. '>' | local passPattern = '<' .. passTagPattern .. '>(.-)</' .. passTagPattern.. '>' | ||
function iferror(wt) | |||
local frame = mw.getCurrentFrame() | |||
local result = frame:callParserFunction('#iferror', wt, 1, 0) | |||
if result == '1' then | |||
return true | |||
elseif result == '0' then | |||
return false | |||
else | |||
error('Cannot detect error') | |||
end | |||
end | |||
function p.pipe(frame) | function p.pipe(frame) | ||
42번째 줄: | 55번째 줄: | ||
function p.from(args) | function p.from(args) | ||
return '<' .. passTag .. '>' | return '<' .. passTag .. '>' | ||
.. mw.text.jsonEncode(args) | .. mw.text.nowiki(mw.text.jsonEncode(args)) | ||
.. '</' .. passTag .. '>' | .. '</' .. passTag .. '>' | ||
end | end | ||
function p.pass(frame) | function p.pass(frame) | ||
return p.from( | local args = {} | ||
for key, value in pairs(frame.args) do | |||
args[key] = mw.text.unstripNoWiki(value) | |||
end | |||
return p.from(args) | |||
end | end | ||
54번째 줄: | 73번째 줄: | ||
local returned = '' | local returned = '' | ||
local passed = nil | local passed = nil | ||
local err = {} | |||
local printed = 'passTag = ' .. passTag .. '<br />' | local printed = 'passTag = ' .. passTag .. '<br />' | ||
-- 한 단계씩 파싱 | -- 한 단계씩 파싱 | ||
for i, source in ipairs(frame.args) do | for i, source in ipairs(frame.args) do | ||
err = {} | |||
local title = mw.title.new(source, 'Template') | local title = mw.title.new(source, 'Template') | ||
94번째 줄: | 115번째 줄: | ||
returned = tf.create(source):parse(passed) -- 이전 반환값 위키텍스트 파싱 | returned = tf.create(source):parse(passed) -- 이전 반환값 위키텍스트 파싱 | ||
passed = returned:match(passPattern) -- pass 인자로 넘겨줄 부분: #pass 부분만 추출 | passed = returned:match(passPattern) -- pass 인자로 넘겨줄 부분: #pass 부분만 추출 | ||
passed = passed and mw.text.jsonDecode(passed) -- pass 부분 테이블로 파싱 | passed = passed and select(2, xpcall(function() return mw.text.jsonDecode(mw.text.unstripNoWiki(mw.text.decode(passed))) end, function() printed = printed .. '<br /><strong class=error>pass 파싱중 에러 (내용: ' .. passed .. ')</strong>' end)) -- pass 부분 테이블로 파싱 | ||
returned = returned:gsub(passPattern, '') -- RETURNED로 넘겨줄 부분: #pass 부분 | returned = returned:gsub(passPattern, '') -- RETURNED로 넘겨줄 부분: #pass 부분 | ||
printed = printed .. 'RETURNED raw = ' .. frame:extensionTag('pre', returned) .. '<br />RETURNED parsed = ' .. returned | printed = printed .. 'RETURNED raw = ' .. frame:extensionTag('pre', returned) .. '<br />RETURNED parsed = ' .. returned | ||
112번째 줄: | 133번째 줄: | ||
end | end | ||
end | end | ||
if iferror(returned) then | |||
table.insert(err, returned) | |||
end | |||
if passed then | |||
for key, value in pairs(passed) do | |||
if iferror(value) then | |||
table.insert(err, value) | |||
end | |||
end | |||
end | |||
printed = printed .. '<br />errors = ' .. table.concat(err, ',') | |||
end | end | ||
2024년 12월 21일 (토) 20:05 기준 최신판
이전 값에 종속되는 여러 틀을 연속적으로 실행합니다. 이를 통해 복잡한 로직을 함수형으로 구현하거나 중첩된 틀을 가독성 좋게 바꿀 수 있습니다.
사용법[원본 편집]
code_blocks 코드
{{#invoke:Pipeline|pipe
|{{#invoke:pipeline|pass|9|a=10}}11
|<nowiki>{{#expr:{{{1}}}+{{{a}}}+{{RETURNED}}}}</nowiki>
}}
code
description 결과
30
이전에 실행된 함수의 값을 RETURNED 상수를 통해 통째로 다음 함수에 넘겨줍니다. pass 함수를 활용하여 여러 값을 넘겨줄 수 있습니다.
위 설명은 모듈:Pipeline/설명문서의 내용을 가져와 보여주고 있습니다. (편집 | 역사) 이 모듈에 대한 수정 연습과 시험은 연습장 (만들기 | 미러)과 시험장 (만들기)에서 할 수 있습니다. 분류는 /설명문서에 넣어주세요. 이 모듈에 딸린 문서. |
local p = {}
local tf = require('모듈:TemplateFunction')
local random = require('모듈:Random')
local passRandStr = random.id(6, os.time())
local passTag = 'pipe-passed-data--' .. passRandStr
local passTagPattern = 'pipe%-passed%-data%-%-' .. passRandStr
local passPattern = '<' .. passTagPattern .. '>(.-)</' .. passTagPattern.. '>'
function iferror(wt)
local frame = mw.getCurrentFrame()
local result = frame:callParserFunction('#iferror', wt, 1, 0)
if result == '1' then
return true
elseif result == '0' then
return false
else
error('Cannot detect error')
end
end
function p.pipe(frame)
local returned = ''
local passed = nil
-- 한 단계씩 파싱
for _, source in ipairs(frame.args) do
local title = mw.title.new(source, 'Template')
-- RETURNED 파싱
source = mw.text.unstripNoWiki(mw.text.decode(source)):gsub('{{RETURNED}}', function(default)
return returned or ''
end)
if title and title.exists then
-- 틀/문서 제목과 일치할 경우 틀로 처리
returned = tf.load(source):parse(passed)
elseif not pcall(function()
-- 파서 함수 이름과 일치할 경우 파서 함수로 처리
returned = frame:callParserFunction(source, passed)
end) then
-- 아니면 인라인 위키텍스트로 처리
returned = tf.create(source):parse(passed) -- 이전 반환값 위키텍스트 파싱
passed = returned:match(passPattern) -- pass 인자로 넘겨줄 부분: #pass 부분만 추출
passed = passed and mw.text.jsonDecode(passed) -- pass 부분 테이블로 파싱
returned = returned:gsub(passPattern, '') -- RETURNED로 넘겨줄 부분: #pass 부분 제외
end
end
return returned
end
function p.from(args)
return '<' .. passTag .. '>'
.. mw.text.nowiki(mw.text.jsonEncode(args))
.. '</' .. passTag .. '>'
end
function p.pass(frame)
local args = {}
for key, value in pairs(frame.args) do
args[key] = mw.text.unstripNoWiki(value)
end
return p.from(args)
end
function p.debug(frame)
local tableView = require('모듈:TableView')
local returned = ''
local passed = nil
local err = {}
local printed = 'passTag = ' .. passTag .. '<br />'
-- 한 단계씩 파싱
for i, source in ipairs(frame.args) do
err = {}
local title = mw.title.new(source, 'Template')
printed = printed .. '[#' .. i .. ']<br />source = ' .. frame:extensionTag('pre', source)
if passed then
printed = printed .. '#passed = ' .. tostring(tableView.kv(passed, {
header = true,
attr = {
class = 'wikitable'
},
transformer = function(key, value)
return key, mw.text.nowiki(value)
end
}))
end
-- RETURNED 파싱
source = mw.text.unstripNoWiki(mw.text.decode(source)):gsub('{{RETURNED}}', function(default)
return returned or ''
end)
if title and title.exists then
-- 틀/문서 제목과 일치할 경우 틀로 처리
printed = printed .. 'template = ' .. source .. '<br />'
returned = tf.load(source):parse(passed)
printed = printed .. 'RETURNED raw = ' .. frame:extensionTag('pre', returned) .. '<br />RETURNED parsed = ' .. returned
elseif not pcall(function()
-- 파서 함수 이름과 일치할 경우 파서 함수로 처리
returned = frame:callParserFunction(source, passed)
printed = printed .. 'parserFunction = ' .. source .. '<br />'
printed = printed .. 'RETURNED raw = ' .. frame:extensionTag('pre', returned) .. '<br />RETURNED parsed = ' .. returned
end) then
-- 아니면 인라인 위키텍스트로 처리
printed = printed .. '<br />wikitext = ' .. frame:extensionTag('pre', source) .. '<br />'
returned = tf.create(source):parse(passed) -- 이전 반환값 위키텍스트 파싱
passed = returned:match(passPattern) -- pass 인자로 넘겨줄 부분: #pass 부분만 추출
passed = passed and select(2, xpcall(function() return mw.text.jsonDecode(mw.text.unstripNoWiki(mw.text.decode(passed))) end, function() printed = printed .. '<br /><strong class=error>pass 파싱중 에러 (내용: ' .. passed .. ')</strong>' end)) -- pass 부분 테이블로 파싱
returned = returned:gsub(passPattern, '') -- RETURNED로 넘겨줄 부분: #pass 부분
printed = printed .. 'RETURNED raw = ' .. frame:extensionTag('pre', returned) .. '<br />RETURNED parsed = ' .. returned
printed = printed .. '<br />matched #pass = ' .. mw.text.jsonEncode(passed)
if passed then
printed = printed .. '<br />#passed = ' .. tostring(tableView.kv(passed, {
header = true,
attr = {
class = 'wikitable'
},
transformer = function(key, value)
return key, mw.text.nowiki(value)
end
}))
end
end
if iferror(returned) then
table.insert(err, returned)
end
if passed then
for key, value in pairs(passed) do
if iferror(value) then
table.insert(err, value)
end
end
end
printed = printed .. '<br />errors = ' .. table.concat(err, ',')
end
printed = printed .. '<br />\n----'
return printed
end
return p