pulumi/pkg/codegen/docs/description.go

161 lines
5.2 KiB
Go

// Copyright 2016-2020, Pulumi Corporation.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package docs
import (
"fmt"
"strings"
"github.com/pulumi/pulumi/sdk/v3/go/common/util/contract"
)
const (
beginCodeBlock = "<!--Start PulumiCodeChooser -->"
endCodeBlock = "<!--End PulumiCodeChooser -->"
)
type codeLocation struct {
open int
close int
}
func getCodeSection(doc string) []codeLocation {
var fences []codeLocation
startIndex := 0
for {
open := strings.Index(doc[startIndex:], beginCodeBlock)
if open == -1 {
break
}
var fence codeLocation
fence.open = startIndex + open
startIndex += open + len(beginCodeBlock)
closing := strings.Index(doc[startIndex:], endCodeBlock)
contract.Assertf(closing != -1, "this should never happen: "+
"there should be equal amounts of opening and closing code block markers")
fence.close = startIndex + closing
startIndex += closing + len(endCodeBlock)
fences = append(fences, fence)
}
return fences
}
func markupBlock(block, supportedSnippetLanguages string) string {
languages := []struct{ tag, choosable string }{
{"typescript", "<div>\n<pulumi-choosable type=\"language\" values=\"javascript,typescript\">\n\n"},
{"python", "<div>\n<pulumi-choosable type=\"language\" values=\"python\">\n\n"},
{"go", "<div>\n<pulumi-choosable type=\"language\" values=\"go\">\n\n"},
{"csharp", "<div>\n<pulumi-choosable type=\"language\" values=\"csharp\">\n\n"},
{"java", "<div>\n<pulumi-choosable type=\"language\" values=\"java\">\n\n"},
{"yaml", "<div>\n<pulumi-choosable type=\"language\" values=\"yaml\">\n\n"},
}
const (
//nolint:lll
chooserStartFmt = "<div>\n<pulumi-chooser type=\"language\" options=\"%s\"></pulumi-chooser>\n</div>\n"
choosableEnd = "</pulumi-choosable>\n</div>\n"
)
var markedUpBlock strings.Builder
// first, append the start chooser
markedUpBlock.WriteString(fmt.Sprintf(chooserStartFmt, supportedSnippetLanguages))
for _, lang := range languages {
// Add language specific open choosable
markedUpBlock.WriteString(lang.choosable)
// find our language - because we have no guarantee of order from our input, we need to find
// both code fences and then append the content in the order that docsgen expects.
start := strings.Index(block, "```"+lang.tag)
if start == -1 {
markedUpBlock.WriteString("```\n")
markedUpBlock.WriteString(defaultMissingExampleSnippetPlaceholder)
markedUpBlock.WriteString("\n```\n")
} else {
// find end index - this is the next code fence.
endLangBlock := start + len("```"+lang.tag) + strings.Index(block[start+len("```"+lang.tag):], "```")
// append code to block, and include code fences
markedUpBlock.WriteString(block[start : endLangBlock+len("```")])
markedUpBlock.WriteRune('\n')
}
// add closing choosable
markedUpBlock.WriteString(choosableEnd)
}
return markedUpBlock.String()
}
func (dctx *docGenContext) processDescription(description, supportedSnippetLanguages string) docInfo {
importDetails := ""
parts := strings.Split(description, "\n\n## Import")
if len(parts) > 1 {
importDetails = parts[1]
description = parts[0]
}
codeBlocks := getCodeSection(description)
startIndex := 0
var markedUpDescription string
for _, block := range codeBlocks {
// append text
markedUpDescription += description[startIndex:block.open]
codeBlock := description[block.open:block.close]
// append marked up block
markedUpDescription += markupBlock(codeBlock, supportedSnippetLanguages)
startIndex = block.close + len(endCodeBlock)
}
// append remainder of description, if any
markedUpDescription += description[startIndex:]
// sanitize to remove unclosed remaining "```" if any exist
sanitizedDescription, err := sanitizeMarkdown(markedUpDescription)
contract.AssertNoErrorf(err, "error rendering docs")
return docInfo{
description: sanitizedDescription,
importDetails: importDetails,
}
}
func sanitizeMarkdown(description string) (string, error) {
sanitized, _, err := removeUnclosedFence(description, 0)
return sanitized, err
}
func removeUnclosedFence(description string, start int) (string, int, error) {
// find the first occurrence of backticks
openingIndex := strings.Index(description[start:], "```")
if openingIndex == -1 {
// if no more backticks are found, return the content (this means everything was fine)
return description, start, nil
}
openingIndex += start
// get closing backticks after the opening set
closingIndex := strings.Index(description[openingIndex+3:], "```")
if closingIndex == -1 {
// remove the unmatched opening ticks
description = description[:openingIndex] + description[openingIndex+3:]
return description, openingIndex, nil
}
closingIndex += openingIndex + 3
return removeUnclosedFence(description, closingIndex+3)
}