Created 10 detailed GitHub issues covering: - Project activation and management UI (#1-2) - Worker node coordination and visualization (#3-4) - Automated GitHub repository scanning (#5) - Intelligent model-to-issue matching (#6) - Multi-model task execution system (#7) - N8N workflow integration (#8) - Hive-Bzzz P2P bridge (#9) - Peer assistance protocol (#10) Each issue includes detailed specifications, acceptance criteria, technical implementation notes, and dependency mapping. 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
		
			
				
	
	
		
			213 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			213 lines
		
	
	
		
			5.7 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { isSpace, normalizeReference } from '../common/utils.mjs'
 | |
| 
 | |
| export default function reference (state, startLine, _endLine, silent) {
 | |
|   let pos = state.bMarks[startLine] + state.tShift[startLine]
 | |
|   let max = state.eMarks[startLine]
 | |
|   let nextLine = startLine + 1
 | |
| 
 | |
|   // if it's indented more than 3 spaces, it should be a code block
 | |
|   if (state.sCount[startLine] - state.blkIndent >= 4) { return false }
 | |
| 
 | |
|   if (state.src.charCodeAt(pos) !== 0x5B/* [ */) { return false }
 | |
| 
 | |
|   function getNextLine (nextLine) {
 | |
|     const endLine = state.lineMax
 | |
| 
 | |
|     if (nextLine >= endLine || state.isEmpty(nextLine)) {
 | |
|       // empty line or end of input
 | |
|       return null
 | |
|     }
 | |
| 
 | |
|     let isContinuation = false
 | |
| 
 | |
|     // this would be a code block normally, but after paragraph
 | |
|     // it's considered a lazy continuation regardless of what's there
 | |
|     if (state.sCount[nextLine] - state.blkIndent > 3) { isContinuation = true }
 | |
| 
 | |
|     // quirk for blockquotes, this line should already be checked by that rule
 | |
|     if (state.sCount[nextLine] < 0) { isContinuation = true }
 | |
| 
 | |
|     if (!isContinuation) {
 | |
|       const terminatorRules = state.md.block.ruler.getRules('reference')
 | |
|       const oldParentType = state.parentType
 | |
|       state.parentType = 'reference'
 | |
| 
 | |
|       // Some tags can terminate paragraph without empty line.
 | |
|       let terminate = false
 | |
|       for (let i = 0, l = terminatorRules.length; i < l; i++) {
 | |
|         if (terminatorRules[i](state, nextLine, endLine, true)) {
 | |
|           terminate = true
 | |
|           break
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       state.parentType = oldParentType
 | |
|       if (terminate) {
 | |
|         // terminated by another block
 | |
|         return null
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     const pos = state.bMarks[nextLine] + state.tShift[nextLine]
 | |
|     const max = state.eMarks[nextLine]
 | |
| 
 | |
|     // max + 1 explicitly includes the newline
 | |
|     return state.src.slice(pos, max + 1)
 | |
|   }
 | |
| 
 | |
|   let str = state.src.slice(pos, max + 1)
 | |
| 
 | |
|   max = str.length
 | |
|   let labelEnd = -1
 | |
| 
 | |
|   for (pos = 1; pos < max; pos++) {
 | |
|     const ch = str.charCodeAt(pos)
 | |
|     if (ch === 0x5B /* [ */) {
 | |
|       return false
 | |
|     } else if (ch === 0x5D /* ] */) {
 | |
|       labelEnd = pos
 | |
|       break
 | |
|     } else if (ch === 0x0A /* \n */) {
 | |
|       const lineContent = getNextLine(nextLine)
 | |
|       if (lineContent !== null) {
 | |
|         str += lineContent
 | |
|         max = str.length
 | |
|         nextLine++
 | |
|       }
 | |
|     } else if (ch === 0x5C /* \ */) {
 | |
|       pos++
 | |
|       if (pos < max && str.charCodeAt(pos) === 0x0A) {
 | |
|         const lineContent = getNextLine(nextLine)
 | |
|         if (lineContent !== null) {
 | |
|           str += lineContent
 | |
|           max = str.length
 | |
|           nextLine++
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (labelEnd < 0 || str.charCodeAt(labelEnd + 1) !== 0x3A/* : */) { return false }
 | |
| 
 | |
|   // [label]:   destination   'title'
 | |
|   //         ^^^ skip optional whitespace here
 | |
|   for (pos = labelEnd + 2; pos < max; pos++) {
 | |
|     const ch = str.charCodeAt(pos)
 | |
|     if (ch === 0x0A) {
 | |
|       const lineContent = getNextLine(nextLine)
 | |
|       if (lineContent !== null) {
 | |
|         str += lineContent
 | |
|         max = str.length
 | |
|         nextLine++
 | |
|       }
 | |
|     } else if (isSpace(ch)) {
 | |
|       /* eslint no-empty:0 */
 | |
|     } else {
 | |
|       break
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // [label]:   destination   'title'
 | |
|   //            ^^^^^^^^^^^ parse this
 | |
|   const destRes = state.md.helpers.parseLinkDestination(str, pos, max)
 | |
|   if (!destRes.ok) { return false }
 | |
| 
 | |
|   const href = state.md.normalizeLink(destRes.str)
 | |
|   if (!state.md.validateLink(href)) { return false }
 | |
| 
 | |
|   pos = destRes.pos
 | |
| 
 | |
|   // save cursor state, we could require to rollback later
 | |
|   const destEndPos = pos
 | |
|   const destEndLineNo = nextLine
 | |
| 
 | |
|   // [label]:   destination   'title'
 | |
|   //                       ^^^ skipping those spaces
 | |
|   const start = pos
 | |
|   for (; pos < max; pos++) {
 | |
|     const ch = str.charCodeAt(pos)
 | |
|     if (ch === 0x0A) {
 | |
|       const lineContent = getNextLine(nextLine)
 | |
|       if (lineContent !== null) {
 | |
|         str += lineContent
 | |
|         max = str.length
 | |
|         nextLine++
 | |
|       }
 | |
|     } else if (isSpace(ch)) {
 | |
|       /* eslint no-empty:0 */
 | |
|     } else {
 | |
|       break
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // [label]:   destination   'title'
 | |
|   //                          ^^^^^^^ parse this
 | |
|   let titleRes = state.md.helpers.parseLinkTitle(str, pos, max)
 | |
|   while (titleRes.can_continue) {
 | |
|     const lineContent = getNextLine(nextLine)
 | |
|     if (lineContent === null) break
 | |
|     str += lineContent
 | |
|     pos = max
 | |
|     max = str.length
 | |
|     nextLine++
 | |
|     titleRes = state.md.helpers.parseLinkTitle(str, pos, max, titleRes)
 | |
|   }
 | |
|   let title
 | |
| 
 | |
|   if (pos < max && start !== pos && titleRes.ok) {
 | |
|     title = titleRes.str
 | |
|     pos = titleRes.pos
 | |
|   } else {
 | |
|     title = ''
 | |
|     pos = destEndPos
 | |
|     nextLine = destEndLineNo
 | |
|   }
 | |
| 
 | |
|   // skip trailing spaces until the rest of the line
 | |
|   while (pos < max) {
 | |
|     const ch = str.charCodeAt(pos)
 | |
|     if (!isSpace(ch)) { break }
 | |
|     pos++
 | |
|   }
 | |
| 
 | |
|   if (pos < max && str.charCodeAt(pos) !== 0x0A) {
 | |
|     if (title) {
 | |
|       // garbage at the end of the line after title,
 | |
|       // but it could still be a valid reference if we roll back
 | |
|       title = ''
 | |
|       pos = destEndPos
 | |
|       nextLine = destEndLineNo
 | |
|       while (pos < max) {
 | |
|         const ch = str.charCodeAt(pos)
 | |
|         if (!isSpace(ch)) { break }
 | |
|         pos++
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (pos < max && str.charCodeAt(pos) !== 0x0A) {
 | |
|     // garbage at the end of the line
 | |
|     return false
 | |
|   }
 | |
| 
 | |
|   const label = normalizeReference(str.slice(1, labelEnd))
 | |
|   if (!label) {
 | |
|     // CommonMark 0.20 disallows empty labels
 | |
|     return false
 | |
|   }
 | |
| 
 | |
|   // Reference can not terminate anything. This check is for safety only.
 | |
|   /* istanbul ignore if */
 | |
|   if (silent) { return true }
 | |
| 
 | |
|   if (typeof state.env.references === 'undefined') {
 | |
|     state.env.references = {}
 | |
|   }
 | |
|   if (typeof state.env.references[label] === 'undefined') {
 | |
|     state.env.references[label] = { title, href }
 | |
|   }
 | |
| 
 | |
|   state.line = nextLine
 | |
|   return true
 | |
| }
 |