nips/edufeed.md
2025-11-29 06:21:19 +01:00

14 KiB

Edufeed: AMB-NIP

Abstract

This NIP defines how to handle the metadata profile "Allgemeines Metadatenprofil für Bildungsressourcen" (AMB) in nostr:

  • How to convert AMB metadata to an AMB nostr event
  • How to convert an AMB nostr-event to AMB metadata
  • How to query for AMB nostr-events in supporting relays

Event Kind

This NIP defines kind:30142 as an AMB Metadata Event. This means this is a replaceable event, that can be addressed using kind:pubkey:d-tag.

How to convert AMB metadata to an AMB nostr event

The transformation uses JSON-flattening with : as the delimiter to convert nested AMB metadata structures into flat Nostr tags. Additionally, Nostr-native tag conventions are used where applicable for better interoperability and query efficiency.

Nostr-Native Conventions

This NIP follows Nostr conventions where they align with AMB requirements:

  • t tags: Used for keywords/topics (instead of flattened keywords tags)
  • p tags: Used for creator/contributor references when the creator has a Nostr identity (pubkey), with fallback to flattened structure for non-Nostr identifiers
  • a tags: Used for references to other addressable events on Nostr (including other AMB events), with fallback to flattened URIs for external resources

Flattening Rules

  1. Simple properties: Map directly to tags

    • AMB: {"name": "Resource Title"}
    • Nostr: ["name", "Resource Title"]
  2. Nested objects: Flatten using : delimiter

    • AMB: {"creator": {"name": "John", "id": "123"}}
    • Nostr: ["creator:name", "John"], ["creator:id", "123"]
  3. Arrays: Repeat the same flattened tag key (order is preserved by tag array position)

    • AMB: {"keywords": ["Math", "Physics"]}
    • Nostr: ["t", "Math"], ["t", "Physics"] (Nostr-native t tag)
  4. Arrays of objects: Repeat flattened keys for each object

    • AMB: {"creator": [{"name": "John"}, {"name": "Jane"}]}
    • Nostr: ["creator:name", "John"], ["creator:name", "Jane"]
  5. Deep nesting: Continue flattening with additional : delimiters

    • AMB: {"creator": {"affiliation": {"name": "MIT"}}}
    • Nostr: ["creator:affiliation:name", "MIT"]

Property Mappings

This is how we convert each property of the AMB:

General:

  • id["d", <id>] (special case: use Nostr's d tag as identifier)
  • type["type", <value>] (repeat for multiple types)
  • name["name", <value>]
  • description["description", <value>]
  • about (array of concept objects) → Repeat for each:
    • ["about:id", <uri>]
    • ["about:prefLabel:lang", <label>]
    • ["about:type", "Concept"]
  • keywords["t", <keyword>] (repeat for each keyword, using Nostr t tag)
  • inLanguage["inLanguage", <languageCode>] (repeat for each language)
  • image["image", <uri>]
  • trailer (MediaObject) →
    • ["trailer:contentUrl", <url>]
    • ["trailer:type", <"VideoObject"|"AudioObject">]
    • ["trailer:encodingFormat", <format>] (optional)
    • ["trailer:contentSize", <bytes>] (optional)
    • ["trailer:sha256", <hash>] (optional)
    • ["trailer:embedUrl", <url>] (optional)
    • ["trailer:bitrate", <kbps>] (optional)

Provenance:

  • creator (array of Person/Organization objects) → Repeat for each:
    • Nostr-native (if creator has Nostr pubkey): ["p", <npub-hex>, <relay>, "creator"]
    • Fallback (for non-Nostr identifiers):
      • ["creator:id", <uri>] (optional, e.g., ORCID, GND)
      • ["creator:name", <name>]
      • ["creator:type", <"Person"|"Organization">]
      • ["creator:honorificPrefix", <title>] (optional, for persons)
      • ["creator:affiliation:id", <uri>] (optional)
      • ["creator:affiliation:name", <name>] (optional)
      • ["creator:affiliation:type", "Organization"] (optional)
  • contributor (array of Person/Organization objects) → Same structure as creator
  • dateCreated["dateCreated", <ISO8601Date>]
  • datePublished["datePublished", <ISO8601Date>]
  • dateModified["dateModified", <ISO8601Date>]
  • publisher (array of Organization/Person objects) → Repeat for each:
    • ["publisher:id", <uri>] (optional)
    • ["publisher:name", <name>]
    • ["publisher:type", <"Organization"|"Person">]
  • funder (array of Person/Organization/FundingScheme objects) → Repeat for each:
    • ["funder:id", <uri>] (optional)
    • ["funder:name", <name>]
    • ["funder:type", <"Person"|"Organization"|"FundingScheme">]

Costs and Rights:

  • isAccessibleForFree["isAccessibleForFree", <"true"|"false">]
  • license (object) →
    • ["license:id", <license_uri>]
  • conditionsOfAccess (Concept object) →
    • ["conditionsOfAccess:id", <uri>]
    • ["conditionsOfAccess:prefLabel:lang", <label>] (optional)
    • ["conditionsOfAccess:type", "Concept"] (optional)

Educational:

  • learningResourceType (array of Concept objects) → Repeat for each:
    • ["learningResourceType:id", <uri>]
    • ["learningResourceType:prefLabel:lang", <label>] (optional)
    • ["learningResourceType:type", "Concept"] (optional)
  • audience (array of Concept objects) → Repeat for each:
    • ["audience:id", <uri>]
    • ["audience:prefLabel:lang", <label>] (optional)
    • ["audience:type", "Concept"] (optional)
  • teaches (array of Concept objects) → Repeat for each:
    • ["teaches:id", <uri>]
    • ["teaches:prefLabel:lang", <label>] (optional)
  • assesses (array of Concept objects) → Repeat for each:
    • ["assesses:id", <uri>]
    • ["assesses:prefLabel:lang", <label>] (optional)
  • competencyRequired (array of Concept objects) → Repeat for each:
    • ["competencyRequired:id", <uri>]
    • ["competencyRequired:prefLabel:lang", <label>] (optional)
  • educationalLevel (array of Concept objects) → Repeat for each:
    • ["educationalLevel:id", <uri>]
    • ["educationalLevel:prefLabel:lang", <label>] (optional)
    • ["educationalLevel:type", "Concept"] (optional)
  • interactivityType (Concept object) →
    • ["interactivityType:id", <uri>]
    • ["interactivityType:prefLabel:lang", <label>] (optional)
    • ["interactivityType:type", "Concept"] (optional)

Relations:

  • isBasedOn (array of objects) → Repeat for each:
    • Nostr-native (if referenced resource is addressable AMB event): ["a", "30142:<pubkey>:<d-value>", <relay>, "isBasedOn"]
    • Fallback (for external URIs):
      • ["isBasedOn:id", <uri>]
      • ["isBasedOn:name", <name>] (optional)
  • isPartOf (array of objects) → Repeat for each:
    • Nostr-native (if referenced resource is addressable AMB event): ["a", "30142:<pubkey>:<d-value>", <relay>, "isPartOf"]
    • Fallback (for external URIs):
      • ["isPartOf:id", <uri>]
      • ["isPartOf:name", <name>] (optional)
      • ["isPartOf:type", <type>] (optional)
  • hasPart (array of objects) → Repeat for each:
    • Nostr-native (if referenced resource is addressable AMB event): ["a", "30142:<pubkey>:<d-value>", <relay>, "hasPart"]
    • Fallback (for external URIs):
      • ["hasPart:id", <uri>]
      • ["hasPart:name", <name>] (optional)
      • ["hasPart:type", <type>] (optional)

Meta-Metadata:

  • mainEntityOfPage (array of WebPage objects) → Repeat for each:
    • ["mainEntityOfPage:id", <uri>]
    • ["mainEntityOfPage:type", "WebContent"]
    • ["mainEntityOfPage:provider:id", <uri>] (optional)
    • ["mainEntityOfPage:provider:name", <name>] (optional)
    • ["mainEntityOfPage:provider:type", <type>] (optional)
    • ["mainEntityOfPage:dateCreated", <ISO8601Date>] (optional)
    • ["mainEntityOfPage:dateModified", <ISO8601Date>] (optional)

Technical:

  • duration["duration", <ISO8601Duration>] (format: PnYnMnDTnHnMnS)
  • encoding (array of MediaObject objects) → Repeat for each:
    • ["encoding:type", "MediaObject"]
    • ["encoding:contentUrl", <url>] (or use embedUrl)
    • ["encoding:embedUrl", <url>] (or use contentUrl)
    • ["encoding:encodingFormat", <format>] (optional, IANA media type)
    • ["encoding:contentSize", <bytes>] (optional)
    • ["encoding:sha256", <hash>] (optional)
    • ["encoding:bitrate", <kbps>] (optional)
  • caption (array of MediaObject objects) → Repeat for each:
    • ["caption:id", <uri>]
    • ["caption:type", "MediaObject"]
    • ["caption:encodingFormat", <format>] (optional, IANA media type)
    • ["caption:inLanguage", <languageCode>] (optional)

How to convert an AMB nostr-event to AMB metadata

To convert a Nostr event back to AMB metadata:

  1. Extract tags: Get the tags array from the Nostr event
  2. Group by prefix: Collect all tags that share the same prefix (before the first :)
  3. Reconstruct nesting: Use the : delimiter to rebuild nested object structure
  4. Handle arrays: Multiple tags with identical keys become array elements
  5. Preserve order: Array order is determined by tag order in the event
  6. Special mappings:
    • d tag → id property
    • Convert string booleans to actual booleans
    • Parse ISO8601 dates if needed for validation

How to query for AMB nostr-events in supporting relays

TODO: This section will be completed once relay support is implemented.

Query capabilities should include:

  • Filter by kind:30142 for all AMB events
  • Filter by author (pubkey)
  • Filter by specific subjects using #about:id tag
  • Filter by learning resource type using #learningResourceType:id tag
  • Filter by educational level using #educationalLevel:id tag
  • Full-text search in name and description tags

Examples

Example 1: Simple Educational Resource

{
  "kind": 30142,
  "id": "6ba638a3786cfce89af1702a36c59e0bd9206863afa5cb6b1299aaf0d9f48c84",
  "pubkey": "79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798",
  "created_at": 1743419457,
  "tags": [
    ["d", "oersi.org/resources/aHR0cHM6Ly9hdi50aWIuZXUvbWVkaWEvNjY5ODM=11"],
    ["type", "LearningResource"],
    ["name", "Pythagorean Theorem Video"],
    ["description", "An introductory video explaining the Pythagorean theorem"],
    ["about:id", "http://w3id.org/kim/schulfaecher/s1017"],
    ["about:prefLabel:de", "Mathematik"],
    ["about:type", "Concept"],
    ["about:id", "http://w3id.org/kim/schulfaecher/s1005"],
    ["about:prefLabel:de", "Deutsch"],
    ["about:type", "Concept"],
    ["learningResourceType:id", "http://w3id.org/openeduhub/vocabs/new_lrt/7a6e9608-2554-4981-95dc-47ab9ba924de"],
    ["learningResourceType:prefLabel:de", "Video"],
    ["learningResourceType:type", "Concept"],
    ["t", "Pythagoras"],
    ["t", "Geometrie"],
    ["t", "Mathematik"],
    ["inLanguage", "de"],
    ["license:id", "https://creativecommons.org/licenses/by/4.0/"],
    ["isAccessibleForFree", "true"]
  ],
  "content": "",
  "sig": "6b0b78d56dea322864d35ea3b6d7e892d0e62bed96cd11ecb27d6c1d0b6d0cd68cd9ec82419946a5fb3c8d4a21eca88c9a5dad47a3b3e466ba18787224a613ef"
}

Example 2: Resource with Creator and Affiliation

{
  "kind": 30142,
  "id": "7ca749b4897efdc98fe2803dc60f68c9e1cd29764e8a55d1e9ef47a46ba4fe75",
  "pubkey": "79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798",
  "created_at": 1743419500,
  "tags": [
    ["d", "https://example.org/courses/physics-101"],
    ["type", "LearningResource"],
    ["type", "Course"],
    ["name", "Introduction to Physics"],
    ["description", "A comprehensive introduction to classical mechanics"],
    ["p", "79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798", "wss://relay.example.com", "creator"],
    ["creator:id", "https://orcid.org/0000-0001-2345-6789"],
    ["creator:name", "Dr. Jane Smith"],
    ["creator:type", "Person"],
    ["creator:honorificPrefix", "Dr."],
    ["creator:affiliation:id", "https://ror.org/example123"],
    ["creator:affiliation:name", "Massachusetts Institute of Technology"],
    ["creator:affiliation:type", "Organization"],
    ["p", "2c4b52e2a4f7c8d9e0f1a2b3c4d5e6f7a8b9c0d1e2f3a4b5c6d7e8f9a0b1c", "wss://relay.example.com", "creator"],
    ["creator:id", "https://orcid.org/0000-0009-8765-4321"],
    ["creator:name", "Prof. John Doe"],
    ["creator:type", "Person"],
    ["creator:honorificPrefix", "Prof."],
    ["creator:affiliation:name", "Stanford University"],
    ["creator:affiliation:type", "Organization"],
    ["dateCreated", "2024-01-15"],
    ["datePublished", "2024-02-01"],
    ["about:id", "https://w3id.org/kim/hochschulfaechersystematik/n079"],
    ["about:prefLabel:de", "Informatik"],
    ["about:type", "Concept"],
    ["learningResourceType:id", "https://w3id.org/kim/hcrt/course"],
    ["learningResourceType:prefLabel:de", "Kurs"],
    ["audience:id", "http://purl.org/dcx/lrmi-vocabs/educationalAudienceRole/student"],
    ["audience:prefLabel:de", "Student"],
    ["audience:type", "Concept"],
    ["educationalLevel:id", "https://w3id.org/kim/educationalLevel/level_06"],
    ["educationalLevel:prefLabel:en", "Bachelor or equivalent"],
    ["inLanguage", "en"],
    ["license:id", "https://creativecommons.org/licenses/by-sa/4.0/"],
    ["isAccessibleForFree", "true"]
  ],
  "content": "",
  "sig": "8d1c89f5da33ec9a2b456def78a90b1cd23e456f78a90b12cd34e567f89a012b34c56d78e9f0a12bc3d45e6f78901a23b45c67d89e0f1a2b3c4d5e6f7890123a"
}

Tools

Using nak to create AMB events

You can use nak to create AMB events with the flattened tag structure:

# Example 1: Simple resource with Nostr-native t tags
nak event \
  --kind 30142 \
  --tag d="oersi.org/resources/example123" \
  --tag type="LearningResource" \
  --tag name="Pythagorean Theorem Video" \
  --tag description="An introductory video" \
  --tag about:id="http://w3id.org/kim/schulfaecher/s1017" \
  --tag about:prefLabel:de="Mathematik" \
  --tag about:type="Concept" \
  --tag t="Pythagoras" \
  --tag t="Geometrie" \
  --tag inLanguage="de" \
  --tag license:id="https://creativecommons.org/licenses/by/4.0/"

# Example 2: Resource with creator (with Nostr identity p tag)
nak event \
  --kind 30142 \
  --tag d="https://example.org/resource/456" \
  --tag name="Physics Course" \
  --tag p="79be667ef9dcbbac55a06295ce870b07029bfcdb2dce28d959f2815b16f81798;wss://relay.example.com;creator" \
  --tag creator:affiliation:name="MIT" \
  --tag creator:affiliation:type="Organization"


## References

- [AMB Specification](https://dini-ag-kim.github.io/amb/latest/)
- [Nostr Protocol (NIP-01)](https://github.com/nostr-protocol/nips/blob/master/01.md)
- [Addressable Events (NIP-33)](https://github.com/nostr-protocol/nips/blob/master/33.md)
- [JSON-Flattening Concept](https://localizely.com/json-flattener/)