blob: f3463e26463cee619e0c7992a05788a42b0118e1 [file] [log] [blame]
-- Copyright (c) 2015-2018, Regents of the University of California.
--
-- This file is part of ndn-tools (Named Data Networking Essential Tools).
-- See AUTHORS.md for complete list of ndn-tools authors and contributors.
--
-- ndn-tools is free software: you can redistribute it and/or modify it under the terms
-- of the GNU General Public License as published by the Free Software Foundation,
-- either version 3 of the License, or (at your option) any later version.
--
-- ndn-tools is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
-- without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
-- PURPOSE. See the GNU General Public License for more details.
--
-- You should have received a copy of the GNU General Public License along with
-- ndn-tools, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
--
-- @author Qi Zhao <https://www.linkedin.com/pub/qi-zhao/73/835/9a3>
-- @author Seunghyun Yoo <http://relue2718.com/>
-- @author Seungbae Kim <https://sites.google.com/site/sbkimcv/>
-- @author Alexander Afanasyev <http://lasr.cs.ucla.edu/afanasyev/index.html>
-- @author Zipeng Wang
-- @author Qianshan Yu
-- inspect.lua (https://github.com/kikito/inspect.lua) can be used for debugging.
-- See more at http://stackoverflow.com/q/15175859/2150331
-- local inspect = require('inspect')
-- NDN protocol
ndn = Proto("ndn", "Named Data Networking (NDN)")
-- TODO with NDNLPv2 processing:
-- * mark field "unknown" when the field is recognized but the relevant feature is disabled
-- * colorize "unknown field"
-- * for a field that appears out-of-order, display "out-of-order field " in red
-----------------------------------------------------
-----------------------------------------------------
-- Field formatting helpers
-- Borrowed from http://lua-users.org/wiki/StringRecipes
function escapeString(str)
if (str) then
str = string.gsub(str, "\n", "\r\n")
str = string.gsub(str, "([^%w %-%_%.%~])",
function (c) return string.format ("%%%02X", string.byte(c)) end)
str = string.gsub(str, " ", "+")
end
return str
end
-- @return TLV-VALUE portion of a TLV block
function getValue(b)
return b.tvb(b.offset + b.typeLen + b.lengthLen, b.length)
end
function getUriFromNameComponent(b)
-- @todo Implement proper proper URL escaping
return getValue(b):string()
end
function getUriFromName(nameBlock)
if (nameBlock.elements == nil) then
return ""
else
components = {}
for i, block in pairs(nameBlock.elements) do
table.insert(components, getUriFromNameComponent(block))
end
return "/" .. table.concat(components, "/")
end
end
function getUriFromExclude(block)
-- @todo
return ""
end
function getNackReasonDetail(b)
local code = getNonNegativeInteger(b)
if (code == 0) then return "None"
elseif (code == 50) then return "Congestion"
elseif (code == 100) then return "Duplicate"
elseif (code == 150) then return "NoRoute"
else return "Unknown"
end
end
function getCachePolicyDetail(b)
local code = getNonNegativeInteger(b)
if (code == 1) then return "NoCache"
else return "Unknown"
end
end
function getNonNegativeInteger(b)
if (b.length == 1 or b.length == 2 or b.length == 4) then
return getValue(b):uint()
-- Something strange with uint64, not supporting it for now
-- elseif (b.length == 8) then
-- return b.tvb(b.offset + b.typeLen + b.lengthLen, 8):uint64()
else
return 0xFFFFFFFF;
end
end
function getNonce(b)
assert(b.type == 10)
if (b.length ~= 4) then
return "invalid (should have 4 octets)"
end
return getValue(b):uint()
end
function getTrue(block)
return "Yes"
end
local AppPrivateBlock1 = 100
local AppPrivateBlock2 = 800
local AppPrivateBlock3 = 1000
function canIgnoreTlvType(t)
if (t < AppPrivateBlock2 or t >= AppPrivateBlock3) then
return false
else
local mod = math.fmod(t, 2)
if (mod == 1) then
return true
else
return false
end
end
end
function getGenericBlockInfo(block)
local name = ""
-- TODO: Properly format informational message based type value reservations
-- (http://named-data.net/doc/ndn-tlv/types.html#type-value-reservations)
if (block.type <= AppPrivateBlock1) then
name = "Unrecognized from the reserved range " .. 0 .. "-" .. AppPrivateBlock1 .. ""
elseif (AppPrivateBlock1 < block.type and block.type < AppPrivateBlock2) then
name = "Unrecognized from the reserved range " .. (AppPrivateBlock1 + 1) .. "-" .. (AppPrivateBlock2 - 1) .. ""
elseif (AppPrivateBlock2 <= block.type and block.type <= AppPrivateBlock3) then
if (canIgnoreTlvType(block.type)) then
name = "Unknown field (ignored)"
else
name = "Unknown field"
end
else
name = "RESERVED_3"
end
return name .. ", Type: " .. block.type .. ", Length: " .. block.length
end
-----------------------------------------------------
-----------------------------------------------------
local NDN_DICT = {
-- Name and NameComponent
[7] = {name = "Name" , field = ProtoField.string("ndn.name", "Name") , value = getUriFromName},
[1] = {name = "ImplicitSha256DigestComponent", field = ProtoField.string("ndn.implicitsha256", "ImplicitSha256DigestComponent"), value = getUriFromNameComponent},
[8] = {name = "GenericNameComponent" , field = ProtoField.string("ndn.namecomponent", "NameComponent") , value = getUriFromNameComponent},
-- Interest and its sub-elements in Packet Format v0.3
[5] = {name = "Interest" , summary = true},
[33] = {name = "CanBePrefix" , field = ProtoField.string("ndn.canbeprefix", "CanBePrefix") , value = getTrue},
[18] = {name = "MustBeFresh" , field = ProtoField.string("ndn.mustbefresh", "MustBeFresh") , value = getTrue},
-- [30] = {name = "ForwardingHint" , summary = true},
-- ForwardingHint and LinkPreference have the same TLV-TYPE number, so we can't recognize both for now (see #4185).
[31] = {name = "LinkDelegation" , summary = true},
[30] = {name = "LinkPreference" , field = ProtoField.uint32("ndn.link_preference", "LinkPreference", base.DEC) , value = getNonNegativeInteger},
[10] = {name = "Nonce" , field = ProtoField.uint32("ndn.nonce", "Nonce", base.HEX) , value = getNonce},
[12] = {name = "InterestLifetime" , field = ProtoField.uint32("ndn.interestlifetime", "InterestLifetime", base.DEC) , value = getNonNegativeInteger},
[34] = {name = "HopLimit" , field = ProtoField.uint32("ndn.hoplimit", "HopLimit", base.DEC) , value = getNonNegativeInteger},
[35] = {name = "Parameters" , field = ProtoField.string("ndn.parameters", "Parameters")},
-- Data and its sub-elements in Packet Format v0.3
[6] = {name = "Data" , summary = true},
[20] = {name = "MetaInfo" , summary = true},
[24] = {name = "ContentType" , field = ProtoField.uint32("ndn.contenttype", "Content Type", base.DEC) , value = getNonNegativeInteger},
[25] = {name = "FreshnessPeriod" , field = ProtoField.uint32("ndn.freshnessperiod", "FreshnessPeriod", base.DEC) , value = getNonNegativeInteger},
[26] = {name = "FinalBlockId" , field = ProtoField.string("ndn.finalblockid", "FinalBlockId") , value = getUriFromNameComponent},
[21] = {name = "Content" , field = ProtoField.string("ndn.content", "Content")},
[22] = {name = "SignatureInfo" , summary = true},
[27] = {name = "SignatureType" , field = ProtoField.uint32("ndn.signaturetype", "SignatureType", base.DEC) , value = getNonNegativeInteger},
[28] = {name = "KeyLocator" , summary = true},
[29] = {name = "KeyDigest" , field = ProtoField.bytes("ndn.keydigest", "KeyDigest")},
[23] = {name = "SignatureValue" , field = ProtoField.bytes("ndn.signaturevalue", "SignatureValue")},
-- NDNLPv2 headers
[80] = {name = "Fragment" },
[81] = {name = "Sequence" , field = ProtoField.uint32("ndn.sequence", "Sequence", base.DEC), value = getNonNegativeInteger},
[82] = {name = "FragIndex" , field = ProtoField.uint32("ndn.fragindex", "FragIndex", base.DEC), value = getNonNegativeInteger},
[83] = {name = "FragCount" , field = ProtoField.uint32("ndn.fragcount", "FragCount", base.DEC), value = getNonNegativeInteger},
[100] = {name = "LpPacket" , summary = true},
[800] = {name = "Nack" , summary = true},
[801] = {name = "NackReason" , field = ProtoField.string("ndn.nack_reason", "NackReason"), value = getNackReasonDetail},
[816] = {name = "NextHopFaceId" , field = ProtoField.uint32("ndn.nexthop_faceid", "NextHopFaceId", base.DEC), value = getNonNegativeInteger},
[817] = {name = "IncomingFaceId" , field = ProtoField.uint32("ndn.incoming_faceid", "IncomingFaceId", base.DEC), value = getNonNegativeInteger},
[820] = {name = "CachePolicy" , summary = true},
[821] = {name = "CachePolicyType" , field = ProtoField.string("ndn.cachepolicy_type", "CachePolicyType"), value = getCachePolicyDetail},
-- Deprecated elements
[9] = {name = "Selectors" , summary = true},
[13] = {name = "MinSuffixComponents" , field = ProtoField.uint32("ndn.minsuffix", "MinSuffixComponents") , value = getNonNegativeInteger},
[14] = {name = "MaxSuffixComponents" , field = ProtoField.uint32("ndn.maxsuffix", "MaxSuffixComponents") , value = getNonNegativeInteger},
[15] = {name = "PublisherPublicKeyLocator" , summary = true},
[16] = {name = "Exclude" , field = ProtoField.string("ndn.exclude", "Exclude") , value = getUriFromExclude},
[17] = {name = "ChildSelector" , field = ProtoField.uint32("ndn.childselector", "ChildSelector", base.DEC) , value = getNonNegativeInteger},
[19] = {name = "Any" , field = ProtoField.string("ndn.any", "Any") , value = getTrue},
[32] = {name = "SelectedDelegation" , field = ProtoField.uint32("ndn.selected_delegation", "SelectedDelegation", base.DEC), value = getNonNegativeInteger},
}
-- -- Add protofields in NDN protocol
ndn.fields = {
}
for key, value in pairs(NDN_DICT) do
table.insert(ndn.fields, value.field)
end
-----------------------------------------------------
-----------------------------------------------------
-- block
-- .tvb
-- .offset
-- .type
-- .typeLen
-- .length
-- .lengthLen
-- .size = .typeLen + .lengthLen + .length
function addInfo(block, root) -- may be add additional context later
local info = NDN_DICT[block.type]
if (info == nil) then
info = {}
info.value = getGenericBlockInfo
-- color
end
local treeInfo
if (info.value == nil) then
if (info.field ~= nil) then
treeInfo = root:add(info.field, block.tvb(block.offset, block.size))
else
treeInfo = root:add(block.tvb(block.offset, block.size), info.name)
end
treeInfo:append_text(", Type: " .. block.type .. ", Length: " .. block.length)
else
block.value = info.value(block)
if (info.field ~= nil) then
treeInfo = root:add(info.field, block.tvb(block.offset, block.size), block.value)
else
treeInfo = root:add(block.tvb(block.offset, block.size), block.value)
end
end
block.root = treeInfo
return block.root
end
function addSummary(block)
if (block.elements == nil) then
return
end
local info = NDN_DICT[block.type]
if (info == nil or info.summary == nil) then
return
end
local summary = {}
for k, subblock in pairs(block.elements) do
if (subblock.value ~= nil) then
local info = NDN_DICT[subblock.type]
if (info ~= nil) then
table.insert(summary, info.name .. ": " .. subblock.value)
end
end
end
if (#summary > 0) then
block.summary = table.concat(summary, ", ")
if (block.value == nil) then
block.value = block.summary
end
block.root:append_text(", " .. block.summary)
end
end
-----------------------------------------------------
-----------------------------------------------------
function readVarNumber(tvb, offset)
if offset >= tvb:len() then
return 0, 0
end
local firstOctet = tvb(offset, 1):uint()
if (firstOctet < 253) then
return firstOctet, 1
elseif (firstOctet == 253) and (offset + 3 < tvb:len()) then
return tvb(offset + 1, 2):uint(), 3
elseif (firstOctet == 254) and (offset + 5 < tvb:len()) then
return tvb(offset + 1, 4):uint(), 5
elseif (firstOctet == 255) and (offset + 9 < tvb:len()) then
return tvb(offset + 1, 8):uint64(), 9
end
return 0, 0
end
function getBlock(tvb, offset)
local block = {}
block.tvb = tvb
block.offset = offset
block.type, block.typeLen = readVarNumber(block.tvb, block.offset)
if block.typeLen == 0 then
return nil
end
block.length, block.lengthLen = readVarNumber(block.tvb, block.offset + block.typeLen)
if block.lengthLen == 0 then
return nil
end
block.size = block.typeLen + block.lengthLen + block.length
return block
end
function canBeValidNdnPacket(block)
if ((block.type == 5 or block.type == 6 or block.type == 100) and block.length <= 8800) then
return true
else
return false
end
end
function findNdnPacket(tvb)
offset = 0
while offset + 2 < tvb:len() do
local block = getBlock(tvb, offset)
if (block ~= nil) and canBeValidNdnPacket(block) then
return block
end
offset = offset + 1
end
return nil
end
function getSubBlocks(block)
local valueLeft = block.length
local subBlocks = {}
while valueLeft > 0 do
local offset = block.offset + block.typeLen + block.lengthLen + (block.length - valueLeft)
local child = getBlock(block.tvb, offset)
if child == nil then
return nil
end
valueLeft = valueLeft - child.size
table.insert(subBlocks, child)
end
if (valueLeft == 0) then
return subBlocks
else
return nil
end
end
-----------------------------------------------------
-----------------------------------------------------
-- NDN protocol dissector function
function ndn.dissector(tvb, pInfo, root) -- Tvb, Pinfo, TreeItem
if (tvb:len() ~= tvb:reported_len()) then
return 0 -- ignore partially captured packets
-- this can/may be re-enabled only for unfragmented UDP packets
end
local ok, block = pcall(findNdnPacket, tvb)
if (not ok) then
return 0
end
if (block == nil or block.offset == nil) then
-- no valid NDN packets found
return 0
end
local nBytesLeft = tvb:len() - block.offset
-- print (pInfo.number .. ":: Found block: " .. block.type .. " of length " .. block.size .. " bytesLeft: " .. nBytesLeft)
local pktType = ""
local pktName = ""
while (block.size <= nBytesLeft) do
-- Create TreeItems
block.tree = root:add(ndn, tvb(block.offset, block.size))
local queue = {block}
while (#queue > 0) do
local block = queue[1]
table.remove(queue, 1)
block.elements = getSubBlocks(block)
local subtree = addInfo(block, block.tree)
if (block.elements ~= nil) then
for i, subBlock in pairs(block.elements) do
subBlock.tree = subtree
table.insert(queue, subBlock)
end
end
end
-- Make summaries
local queue = {block}
while (#queue > 0) do
local block = queue[1]
if (block.visited ~= nil or block.elements == nil) then
-- try to make summary
table.remove(queue, 1)
addSummary(block)
else
for i, subBlock in pairs(block.elements) do
table.insert(queue, 1, subBlock)
end
block.visited = true
end
-- prepare information to fill info column
if block.type == 5 and pktType ~= "Nack" then
pktType = "Interest"
elseif block.type == 6 then
pktType = "Data"
elseif block.type == 800 then
pktType = "Nack"
end
if pktName == "" and block.type == 7 then
pktName = getUriFromName(block)
end
end
local info = NDN_DICT[block.type]
if (info ~= nil) then
if (block.summary ~= nil) then
block.tree:append_text(", " .. NDN_DICT[block.type].name .. ", " .. block.summary)
else
block.tree:append_text(", " .. NDN_DICT[block.type].name)
end
end
nBytesLeft = nBytesLeft - block.size
if (nBytesLeft > 0) then
ok, block = pcall(getBlock, tvb, tvb:len() - nBytesLeft)
if (not ok or block == nil or not canBeValidNdnPacket(block)) then
break
end
end
end -- while(block.size <= nBytesLeft)
pInfo.cols.protocol = tostring(pInfo.cols.protocol) .. " (" .. ndn.name .. ")"
pInfo.cols.info = pktType .. " " .. pktName
if (nBytesLeft > 0 and block ~= nil and block.size ~= nil and block.size > nBytesLeft) then
pInfo.desegment_offset = tvb:len() - nBytesLeft
-- Originally, I set desegment_len to the exact lenght, but it mysteriously didn't work for TCP
-- pInfo.desegment_len = block.size -- this will not work to desegment TCP streams
pInfo.desegment_len = DESEGMENT_ONE_MORE_SEGMENT
end
end
local udpDissectorTable = DissectorTable.get("udp.port")
udpDissectorTable:add("6363", ndn)
udpDissectorTable:add("56363", ndn)
local tcpDissectorTable = DissectorTable.get("tcp.port")
tcpDissectorTable:add("6363", ndn)
local websocketDissectorTable = DissectorTable.get("ws.port")
-- websocketDissectorTable:add("9696", ndn)
websocketDissectorTable:add("1-65535", ndn)
local ethernetDissectorTable = DissectorTable.get("ethertype")
ethernetDissectorTable:add(0x8624, ndn)
io.stderr:write("NDN dissector successfully loaded\n")