Index Markdown blobs and files in Azure AI Search

Note

This feature is currently in public preview. This preview is provided without a service-level agreement and isn't recommended for production workloads. Certain features might not be supported or might have constrained capabilities. For more information, see Supplemental Terms of Use for Azure Previews.

In Azure AI Search, indexers for Azure Blob Storage, Azure Files, and Azure OneLake support a markdown parsing mode for Markdown files. Markdown files can be indexed in two ways:

  • One-to-many parsing mode, creating multiple search documents per Markdown file
  • One-to-one parsing mode, creating one search document per Markdown file

Prerequisites

  • A supported data source: Azure Blob storage, Azure File storage.

    Azure Storage for blob indexers and file indexers is a standard performance (general-purpose v2) instance that supports hot and cool access tiers.

Markdown parsing mode parameters

Parsing mode parameters are specified in an indexer definition when you create or update an indexer.

POST https://[service name].search.azure.cn/indexers?api-version=2025-08-01-preview
Content-Type: application/json
api-key: [admin key]

{
  "name": "my-markdown-indexer",
  "dataSourceName": "my-blob-datasource",
  "targetIndexName": "my-target-index",
  "parameters": {
    "configuration": {
      "parsingMode": "markdown",
      "markdownParsingSubmode": "oneToMany",
      "markdownHeaderDepth": "h6"
    }
  },
}

The blob indexer provides a submode parameter to determine the output of structure of the search documents. Markdown parsing mode provides the following submode options:

parsingMode submode Search document Description
markdown oneToMany Multiple per blob (default) Breaks the Markdown into multiple search documents, each representing a content (nonheader) section of the Markdown file. You can omit submode unless you want one-to-one parsing.
markdown oneToOne One per blob Parses the Markdown into one search document, with sections mapped to specific headers in the Markdown file.

For oneToMany submode, you should review Indexing one blob to produce many search documents to understand how the blob indexer handles disambiguation of the document key for multiple search documents produced from the same blob.

Later sections describe each submode in more detail. If you're unfamiliar with indexer clients and concepts, see Create a search indexer. You should also be familiar with the details of basic blob indexer configuration, which isn't repeated here.

Optional Markdown parsing parameters

Parameters are case-sensitive.

Parameter name Allowed Values Description
markdownHeaderDepth h1, h2, h3, h4, h5, h6(default) This parameter determines the deepest header level that is considered when parsing, allowing for flexible handling of document structure (for example, when markdownHeaderDepth is set to h1, the parser only recognizes top-level headers that begin with "#", and all lower-level headers are treated as plain text). If not specified, it defaults to h6.

This setting can be changed after initial creation of the indexer, however the structure of the resulting search documents might change depending on the Markdown content.

Supported Markdown elements

Markdown parsing only splits content based on headers. All other elements such as lists, code blocks, tables, and so forth, are treated as plain text and passed into a content field.

Sample Markdown content

The following Markdown content is used for the examples on this page:

# Section 1
Content for section 1.

## Subsection 1.1
Content for subsection 1.1.

# Section 2
Content for section 2.

Use one-to-many parsing mode

The one-to-many parsing mode parses Markdown files into multiple search documents, where each document corresponds to a specific content section of the Markdown file based on the header metadata at that point in the document. The Markdown is parsed based on headers into search documents, which contain the following content:

  • content: A string that contains the raw Markdown found in a specific location, based on the header metadata at that point in the document.

  • sections: An object that contains subfields for the header metadata up to the desired header level. For example, when markdownHeaderDepth is set to h3, contains string fields h1, h2, and h3. These fields are indexed by mirroring this structure in the index, or through field mappings in the format /sections/h1, sections/h2, etc. See index and indexer configurations in the following samples for in-context examples. The subfields contained are:

    • h1 - A string containing the h1 header value. Empty string if not set at this point in the document.
    • (Optional) h2- A string containing the h2 header value. Empty string if not set at this point in the document.
    • (Optional) h3- A string containing the h3 header value. Empty string if not set at this point in the document.
    • (Optional) h4- A string containing the h4 header value. Empty string if not set at this point in the document.
    • (Optional) h5- A string containing the h5 header value. Empty string if not set at this point in the document.
    • (Optional) h6- A string containing the h6 header value. Empty string if not set at this point in the document.
  • ordinal_position: An integer value indicating the position of the section within the document hierarchy. This field is used for ordering the sections in their original sequence as they appear in the document, beginning with an ordinal position of 1 and incrementing sequentially for each header.

Index schema for one-to-many parsing

An example index configuration might look something like this:

{
  "name": "my-markdown-index",
  "fields": [
  {
    "name": "id",
    "type": "Edm.String",
    "key": true
  },
  {
    "name": "content",
    "type": "Edm.String",
  },
  {
    "name": "ordinal_position",
    "type": "Edm.Int32"
  },
  {
    "name": "sections",
    "type": "Edm.ComplexType",
    "fields": [
    {
      "name": "h1",
      "type": "Edm.String"
    },
    {
      "name": "h2",
      "type": "Edm.String"
    }]
  }]
}

Indexer definition for one-to-many parsing

If field names and data types align, the blob indexer can infer the mapping without an explicit field mapping present in the request, so an indexer configuration corresponding to the provided index configuration might look like this:

POST https://[service name].search.azure.cn/indexers?api-version=2025-08-01-preview
Content-Type: application/json
api-key: [admin key]

{
  "name": "my-markdown-indexer",
  "dataSourceName": "my-blob-datasource",
  "targetIndexName": "my-target-index",
  "parameters": {
    "configuration": { "parsingMode": "markdown" }
  },
}

Note

The submode doesn't need to be set explicitly here because oneToMany is the default.

Indexer output for one-to-many parsing

This Markdown file would result in three search documents after indexing, due to the three content sections. The search document resulting from the first content section of the provided Markdown document would contain the following values for content, sections, h1, and h2:

{
  {
    "content": "Content for section 1.\r\n",
    "sections": {
      "h1": "Section 1",
      "h2": ""
    },
    "ordinal_position": 1
  },
  {
    "content": "Content for subsection 1.1.\r\n",
    "sections": {
      "h1": "Section 1",
      "h2": "Subsection 1.1"
    },
    "ordinal_position": 2
  },
  {
    "content": "Content for section 2.\r\n",
    "sections": {
      "h1": "Section 2",
      "h2": ""
    },
    "ordinal_position": 3
  }
}   

Map one-to-many fields in a search index

Field mappings associate a source field with a destination field in situations where the field names and types aren't identical. But field mappings can also be used to match parts of a Markdown document and "lift" them into top-level fields of the search document.

The following example illustrates this scenario. For more information about field mappings in general, see field mappings.

Assume a search index with the following fields: raw_content of type Edm.String, h1_header of type Edm.String, and h2_header of type Edm.String. To map your Markdown into the desired shape, use the following field mappings:

"fieldMappings" : [
    { "sourceFieldName" : "/content", "targetFieldName" : "raw_content" },
    { "sourceFieldName" : "/sections/h1", "targetFieldName" : "h1_header" },
    { "sourceFieldName" : "/sections/h2", "targetFieldName" : "h2_header" },
  ]

The resulting search document in the index would look as follows:

{
  {
    "raw_content": "Content for section 1.\r\n",
    "h1_header": "Section 1",
    "h2_header": "",
  },
  {
    "raw_content": "Content for section 1.1.\r\n",
    "h1_header": "Section 1",
    "h2_header": "Subsection 1.1",
  },
  {
    "raw_content": "Content for section 2.\r\n",
    "h1_header": "Section 2",
    "h2_header": "",
  }
}

Use one-to-one parsing mode

In the one-to-one parsing mode, the entire Markdown document is indexed as a single search document, preserving the hierarchy and structure of the original content. This mode is most useful when the files to be indexed share a common structure, so that you can use this common structure in the index to make the relevant fields searchable.

Within the indexer definition, set the parsingMode to "markdown" and use the optional markdownHeaderDepth parameter to define the maximum heading depth for chunking. If not specified, it defaults to h6, capturing all possible header depths.

The Markdown is parsed based on headers into search documents, which contain the following content:

  • document_content: Contains the full Markdown text as a single string. This field serves as a raw representation of the input document.

  • sections: An array of objects that contains the hierarchical representation of the sections within the Markdown document. Each section is represented as an object within this array and captures the structure of the document in a nested manner corresponding to the headers and their respective content. The fields are accessible through field mappings by referencing the path, for example /sections/content. The objects in this array have the following properties:

    • header_level: A string that indicates the level of the header (h1, h2, h3, etc.) in Markdown syntax. This field helps in understanding the hierarchy and structuring of the content.

    • header_name: A string containing the text of the header as it appears in the Markdown document. This field provides a label or title for the section.

    • content: A string containing text content that immediately follows the header, up to the next header. This field captures the detailed information or description associated with the header. If there's no content directly under a header, the value is an empty string.

    • ordinal_position: An integer value indicating the position of the section within the document hierarchy. This field is used for ordering the sections in their original sequence as they appear in the document, beginning with an ordinal position of 1 and incrementing sequentially for each content block.

    • sections: An array that contains objects representing subsections nested under the current section. This array follows the same structure as the top-level sections array, allowing for the representation of multiple levels of nested content. Each subsection object also includes header_level, header_name, content, and ordinal_position properties, enabling a recursive structure that represents and hierarchy of the Markdown content.

Here's the sample Markdown that we're using to explain the index schemas designed around each parsing mode.

# Section 1
Content for section 1.

## Subsection 1.1
Content for subsection 1.1.

# Section 2
Content for section 2.

Index schema for one-to-one parsing

If you aren't utilizing field mappings, the shape of the index should reflect the shape of the Markdown content. Given the structure of sample Markdown with its two sections and single subsection, the index should look similar to the following example:

{
  "name": "my-markdown-index",
  "fields": [
  {
    "name": "id",
    "type": "Edm.String",
    "key": true
  },
  {
    "name": "document_content",
    "type": "Edm.String"
  },
  {
    "name": "sections",
    "type": "Collection(Edm.ComplexType)",
    "fields": [
    {
      "name": "header_level",
      "type": "Edm.String"
    },
    {
      "name": "header_name",
      "type": "Edm.String"
    },
    {
      "name": "content",
      "type": "Edm.String"
    },
    {
      "name": "ordinal_position",
      "type": "Edm.Int32"
    },
    {
      "name": "sections",
      "type": "Collection(Edm.ComplexType)",
      "fields": [
      {
        "name": "header_level",
        "type": "Edm.String"
      },
      {
        "name": "header_name",
        "type": "Edm.String"
      },
      {
        "name": "content",
        "type": "Edm.String"
      },
      {
        "name": "ordinal_position",
        "type": "Edm.Int32"
      }]
    }]
  }]
}

Indexer definition for one-to-one parsing

POST https://[service name].search.azure.cn/indexers?api-version=2025-08-01-preview
Content-Type: application/json
api-key: [admin key]

{
  "name": "my-markdown-indexer",
  "dataSourceName": "my-blob-datasource",
  "targetIndexName": "my-target-index",
  "parameters": {
    "configuration": {
      "parsingMode": "markdown",
      "markdownParsingSubmode": "oneToOne",
    }
  }
}

Indexer output for one-to-one parsing

Because the Markdown we want to index only goes to a depth of h2 ("##"), we need sections fields nested to a depth of 2 to match that. This configuration would result in the following data in the index:

  "document_content": "# Section 1\r\nContent for section 1.\r\n## Subsection 1.1\r\nContent for subsection 1.1.\r\n# Section 2\r\nContent for section 2.\r\n",
  "sections": [
    {
      "header_level": "h1",
      "header_name": "Section 1",
      "content": "Content for section 1.",
      "ordinal_position": 1,
      "sections": [
        {
          "header_level": "h2",
          "header_name": "Subsection 1.1",
          "content": "Content for subsection 1.1.",
          "ordinal_position": 2,
        }]
    }],
    {
      "header_level": "h1",
      "header_name": "Section 2",
      "content": "Content for section 2.",
      "ordinal_position": 3,
      "sections": []
    }]
  }

As you can see, the ordinal position increments based on the location of the content within the document.

It should also be noted that if header levels are skipped in the content, then structure of the resulting document reflects the headers that are present in the Markdown content, not necessarily containing nested sections for h1 through h6 consecutively. For example, when the document begins at h2, then the first element in the top-level sections array is h2.

Map one-to-one fields in a search index

If you would like to extract fields with custom names from the document, you can use field mappings to do so. Using the same Markdown sample as before, consider the following index configuration:

{
  "name": "my-markdown-index",
  "fields": [
    {
      "name": "document_content",
      "type": "Edm.String",
    },
    {
      "name": "document_title",
      "type": "Edm.String",
    },
    {
      "name": "opening_subsection_title"
      "type": "Edm.String",
    }
    {
      "name": "summary_content",
      "type": "Edm.String",
    }
  ]
}

Extracting specific fields from the parsed Markdown is handled similar to how the document paths are in outputFieldMappings, except the path begins with /sections instead of /document. So, for example, /sections/0/content would map to the content under the item at position 0 in the sections array.

An example of a strong use case might look something like this: all Markdown files have a document title in the first h1, a subsection title in the first h2, and a summary in the content of the final paragraph underneath the final h1. You could use the following field mappings to index only that content:

"fieldMappings" : [
  { "sourceFieldName" : "/content", "targetFieldName" : "raw_content" },
  { "sourceFieldName" : "/sections/0/header_name", "targetFieldName" : "document_title" },
  { "sourceFieldName" : "/sections/0/sections/header_name", "targetFieldName" : "opening_subsection_title" },
  { "sourceFieldName" : "/sections/1/content", "targetFieldName" : "summary_content" },
]

Here you would extract only the relevant pieces from that document. To most effectively use this functionality, documents you plan to index should share the same hierarchical header structure.

The resulting search document in the index would look as follows:

{
  "content": "Content for section 1.\r\n",
  "document_title": "Section 1",
  "opening_subsection_title": "Subsection 1.1",
  "summary_content": "Content for section 2."
}

Note

These examples specify how to use these parsing modes entirely with or without field mappings, but you can apply both in one scenario if it suits your needs.

Managing stale documents from Markdown re-indexing

When using one-to-many parsing mode, re-indexing a modified Markdown file can result in stale or duplicate documents if sections are removed. This behavior is specific to one-to-many mode and doesn't apply to one-to-one parsing.

Behavior overview

One-to-many parsing mode

In oneToMany mode, each Markdown section (based on headers) is indexed as a separate search document. When the file is re-indexed:

  • No automatic deletion: The indexer overwrites existing documents with new ones, but it does not delete documents that no longer correspond to any content in the updated file.
  • Potential for duplicates: This issue specifically arises only when more sections are deleted than inserted between indexing runs. In such cases, leftover documents from the previous version remain in the index, leading to stale entries that no longer reflect the current state of the source file.

One-to-one parsing mode

In oneToOne mode, the entire Markdown file is indexed as a single search document. When the file is re-indexed:

  • Overwrite behavior: The existing document is replaced entirely with the new version.
  • No stale sections: When the file is re-indexed, the existing document is replaced with the updated version and removed content is no longer included. The only exception is if the file path or blob URI changes, which could result in a new document being created alongside the old one.

Workaround options

To ensure the index reflects the current state of your Markdown files, consider one of the following approaches:

Option 1. Soft delete with metadata

This method uses a soft-delete to delete documents associated with a specific blob. For more information, see Change and delete detection using indexers for Azure Storage in Azure AI Search.

Steps:

  1. Mark the blob as deleted by setting a metadata field.
  2. Let the indexer run. It deletes all documents in the index associated with that blob.
  3. Remove the soft-delete marker and re-index the file.

Option 2. Use the delete API

Before re-indexing a modified Markdown file, explicitly delete the existing documents associated with that file using the delete API. You can either:

  • Manually indentify individual stale documents by identifying duplicates in the index to be deleted. This may be feasible for small, well-understood changes but can be time-consuming.
  • (Recommended) Remove all documents generated from the same parent file before re-indexing, ensuring inconsistencies are avoided.

Steps:

  1. Identify the id of the documents associated with the file. Use a query like the following example to retrieve the document key IDs (for example, id, chunk_id, etc.) for all documents tied to a specific file. Replace metadata_storage_path with the appropriate field in your index that maps to the file path or blob URI. This field must be a key.

    GET https://[service name].search.azure.cn/indexes/[index name]/docs?api-version=2025-05-01-preview
    Content-Type: application/json
    api-key: [admin key]
    
      {  
          "filter": "metadata_storage_path eq 'https://storage-account>.blob.core.chinacloudapi.cn/<container-name>/<file-name>'",
          "select": "id"
      }
    
  2. Issue a delete request for the documents with the identified keys.

    POST https://[service name].search.azure.cn/indexes/[index name]/docs/index?api-version=2025-05-01-preview
    Content-Type: application/json
    api-key: [admin key]
    
    {  
      "value": [  
        {  
          "@search.action": "delete",  
          "id": "aHR0c...jI1"  
        },
        {  
          "@search.action": "delete",  
          "id": "aHR0...MQ2"  
        }  
      ]  
    }
    
  3. Re-index the updated file.

Next steps