work-with-adf
npx skills add https://github.com/dawiddutoit/custom-claude --skill work-with-adf
Agent 安装分布
Skill 文档
Work with ADF (Atlassian Document Format)
Purpose
This skill helps you work with Atlassian Document Format (ADF), the JSON-based format used for rich text content in Jira. Whether you’re creating documents programmatically, understanding the structure, validating content, or debugging Jira API errors, this skill provides clear guidance and patterns.
Quick Start
Create a simple ADF paragraph:
from jira_tool.formatter import JiraDocumentBuilder
doc = JiraDocumentBuilder()
doc.add_paragraph(doc.add_text("Hello, Jira!"))
adf = doc.build()
print(adf)
Output:
{
"version": 1,
"type": "doc",
"content": [
{
"type": "paragraph",
"content": [{"type": "text", "text": "Hello, Jira!"}]
}
]
}
Instructions
Step 1: Understand ADF Structure
ADF documents follow a strict hierarchical structure:
- Root Node (
doc): Every document must haveversion,type: "doc", andcontentarray - Block Nodes: Top-level structure (headings, paragraphs, lists, panels, code blocks, tables)
- Inline Nodes: Content within blocks (text, emoji, links, mentions)
- Marks: Formatting applied to text (bold, italic, code, strikethrough, links)
Minimal Valid Document:
{
"version": 1,
"type": "doc",
"content": []
}
Key Principle: ADF uses a single sequential path – traversing nodes linearly produces correct reading order.
Step 2: Choose Your Approach
Pick the right tool for your use case:
Option A: Use JiraDocumentBuilder (Recommended)
- Built-in Python class in
src/jira_tool/formatter.py - Fluent API with method chaining
- Handles correct nesting automatically
- Best for: Most common tasks
Option B: Use Specialized Builders
EpicBuilder: Pre-formatted template for epicsIssueBuilder: Pre-formatted template for issues- Best for: Creating standardized documents
Option C: Build Raw ADF (Advanced)
- Direct JSON dictionaries
- Full control over structure
- Best for: Complex or non-standard layouts
Step 3: Build Your Document
Basic Elements
Add a heading:
doc.add_heading("My Epic", level=1)
doc.add_heading("Problem Statement", level=2)
Add a paragraph with formatting:
doc.add_paragraph(
doc.bold("Priority: "),
doc.add_text("P0")
)
Add lists:
doc.add_bullet_list(["Item 1", "Item 2", "Item 3"])
doc.add_ordered_list(["First step", "Second step"], start=1)
Add code blocks:
doc.add_code_block(
'def hello():\n print("world")',
language="python"
)
Add panels (info, note, warning, success, error):
doc.add_panel("warning",
{"type": "paragraph", "content": [doc.add_text("Important!")]}
)
Add visual elements:
doc.add_rule() # Horizontal rule
emoji = doc.add_emoji(":rocket:", "ð")
Formatting Options
Text formatting (use in text nodes):
doc.bold("Bold text")
doc.italic("Italic text")
doc.code("inline_code")
doc.strikethrough("Strikethrough")
doc.link("Click here", "https://example.com")
Combine formatting:
doc.add_paragraph(
doc.bold("Status: "),
doc.add_text("In Progress")
)
Step 4: Build and Export
Get the final ADF:
adf_dict = doc.build()
Use with Jira API:
from jira_tool.client import JiraClient
client = JiraClient()
client.create_issue(
project="PROJ",
issue_type="Epic",
summary="My Epic",
description=adf_dict # Pass ADF directly
)
Examples
Example 1: Create a Simple Epic
from jira_tool.formatter import EpicBuilder
epic = EpicBuilder(
title="User Authentication System",
priority="P0",
dependencies="OAuth 2.0 library",
services="Auth Service, API Gateway"
)
epic.add_problem_statement(
"Current authentication is insecure and lacks OAuth support"
)
epic.add_description(
"Implement OAuth 2.0 integration with support for multiple providers"
)
epic.add_technical_details(
requirements=[
"Implement OAuth 2.0 flow",
"Support Google and GitHub providers",
"Add token refresh mechanism"
],
code_example="oauth = OAuth2Handler(provider='google')",
code_language="python"
)
epic.add_acceptance_criteria([
"Users can log in with Google",
"Users can log in with GitHub",
"Tokens refresh automatically"
])
epic.add_edge_cases([
"Handle provider outages gracefully",
"Support token expiration",
"Manage scope conflicts"
])
epic.add_testing_considerations([
"Mock OAuth provider responses",
"Test token refresh edge cases",
"Verify scope permissions"
])
adf = epic.build()
Example 2: Create an Issue with Mixed Content
from jira_tool.formatter import IssueBuilder
issue = IssueBuilder(
title="Implement OAuth Google Provider",
component="Auth Service",
story_points=13,
epic_key="PROJ-100"
)
issue.add_description(
"Add Google OAuth 2.0 provider support to the authentication system"
)
issue.add_implementation_details([
"Register application with Google Cloud Console",
"Implement OAuth callback endpoint",
"Add token validation and refresh logic",
"Update user profile with Google user ID"
])
issue.add_acceptance_criteria([
"Users can authenticate with Google account",
"Tokens are validated on each request",
"Token refresh works correctly",
"Tests pass with >90% coverage"
])
adf = issue.build()
Example 3: Create Raw ADF with Full Control
For cases where builders don’t provide enough flexibility:
adf_document = {
"version": 1,
"type": "doc",
"content": [
{
"type": "heading",
"attrs": {"level": 1},
"content": [{"type": "text", "text": "Custom Report"}]
},
{
"type": "paragraph",
"content": [
{"type": "text", "text": "Data as of: "},
{"type": "text", "text": "2025-11-03", "marks": [{"type": "code"}]}
]
},
{
"type": "table",
"attrs": {"isNumberColumnEnabled": False, "layout": "default"},
"content": [
{
"type": "tableRow",
"content": [
{
"type": "tableHeader",
"content": [
{"type": "paragraph", "content": [{"type": "text", "text": "Metric"}]}
]
},
{
"type": "tableHeader",
"content": [
{"type": "paragraph", "content": [{"type": "text", "text": "Value"}]}
]
}
]
}
]
}
]
}
Example 4: Add Emoji to Documents
from jira_tool.formatter import JiraDocumentBuilder
doc = JiraDocumentBuilder()
# Emoji in paragraph
doc.add_paragraph(
doc.add_emoji(":rocket:", "ð"),
doc.add_text(" "),
doc.bold("Important Update")
)
# Emoji in heading
doc.add_heading(doc.add_emoji(":warning:", "â ï¸") + " Critical Issue", 1)
adf = doc.build()
Validation
Validate ADF Structure
Using the Official JSON Schema:
Atlassian provides an official JSON Schema for ADF validation:
- Schema URL (latest): https://unpkg.com/@atlaskit/adf-schema@latest/dist/json-schema/v1/full.json
- Schema URL (pinned v51.3.2): https://unpkg.com/@atlaskit/adf-schema@51.3.2/dist/json-schema/v1/full.json
- Short link: https://go.atlassian.com/adf-json-schema (redirects to latest)
Using the Built-in Validator:
# Validate using the skill's validation script
python .claude/skills/work-with-adf/scripts/validate_adf.py your-adf.json
This validator checks:
- Root document structure (version, type, content)
- Required properties for all node types
- Valid parent-child relationships
- Attribute constraints (heading levels, panel types, etc.)
- Text mark validity
Common Validation Errors
Error: "content is required for block nodes"
- Cause: Block node (paragraph, heading, list) missing
contentarray - Fix: Add
"content": []even if empty, or add child nodes
Error: "version is required"
- Cause: Document missing version field at root
- Fix: Ensure
"version": 1at document root
Error: "Node type X is not allowed as child of Y"
- Cause: Invalid nesting (e.g., heading inside paragraph)
- Fix: Review node hierarchy – ensure block nodes only contain valid children
Error: "attrs is required for node type X"
- Cause: Node requires attributes but they’re missing
- Fix: Add
attrsobject with required properties
Debug Validation Errors
When Jira API returns validation errors:
- Print the ADF:
import json; print(json.dumps(adf, indent=2)) - Check nesting: Block nodes at top level, inline nodes in blocks
- Validate attributes: Use
.build()output as reference - Test incrementally: Build document piece by piece
Advanced Patterns
Pattern 1: Conditional Content
Add content based on conditions:
doc = JiraDocumentBuilder()
doc.add_heading("Report", 1)
if has_dependencies:
doc.add_heading("Dependencies", 2)
doc.add_bullet_list(dependencies)
if has_code_example:
doc.add_code_block(code, language)
adf = doc.build()
Pattern 2: Reusable Templates
Create functions for common patterns:
def create_info_section(title: str, content: str) -> dict:
"""Create a titled info panel."""
from jira_tool.formatter import JiraDocumentBuilder
doc = JiraDocumentBuilder()
doc.add_heading(title, 2)
doc.add_panel("info",
{"type": "paragraph", "content": [doc.add_text(content)]}
)
return doc.content[0] # Return just the heading
return doc.content[1] # Return just the panel
# Use in larger document
epic = EpicBuilder("My Epic", "P0")
epic.content.extend(create_info_section("Background", "...").values())
Pattern 3: Multi-line Code Blocks
Preserve formatting in code:
code = """
def calculate(a, b):
result = a + b
return result
"""
doc.add_code_block(code.strip(), language="python")
Requirements
- Python 3.8+
jira-toolpackage (from this repository)- Knowledge of JSON structure
- Jira instance with Cloud API access
See Also
docs/reference/adf_reference_guide.md– Complete ADF node referencesrc/jira_tool/formatter.py– JiraDocumentBuilder source codedocs/guides/jira_formatting_guide.md– Jira formatting best practices- Atlassian ADF Specification – Official documentation
- Official ADF JSON Schema – Schema definition
examples/examples.md– Real-world use cases and patternsreferences/reference.md– Complete node type referencescripts/validate_adf.py– Validation utility