or run

npx @tessl/cli init
Log in

Version

Tile

Overview

Evals

Files

Files

docs

advanced-configuration.mdcomponent-interface.mdcomponent-overrides.mdcore-compilation.mdcustom-rendering.mdindex.mdutility-functions.md

utility-functions.mddocs/

0

# Utility Functions

1

2

Helper functions for content sanitization, slug generation, and other common markdown processing tasks.

3

4

## Capabilities

5

6

### Content Sanitization

7

8

Built-in sanitizer function for securing URLs and content against XSS and other security vulnerabilities.

9

10

```typescript { .api }

11

/**

12

* Default sanitizer function for URLs and attribute values

13

* @param input - The input string to sanitize

14

* @returns Sanitized string or null if blocked

15

*/

16

function sanitizer(input: string): string | null;

17

18

/**

19

* Custom sanitizer function type for advanced security controls

20

* @param value - The attribute value to sanitize

21

* @param tag - The HTML tag containing the attribute

22

* @param attribute - The attribute name being sanitized

23

* @returns Sanitized value or null to remove the attribute

24

*/

25

type SanitizerFunction = (

26

value: string,

27

tag: HTMLTags,

28

attribute: string

29

) => string | null;

30

```

31

32

**Usage Examples:**

33

34

```typescript

35

import { sanitizer } from "markdown-to-jsx";

36

37

// Basic sanitization

38

const safeUrl = sanitizer("https://example.com"); // "https://example.com"

39

const blockedJs = sanitizer("javascript:alert('xss')"); // null

40

const blockedVbs = sanitizer("vbscript:msgbox('xss')"); // null

41

const blockedData = sanitizer("data:text/html,<script>"); // null

42

43

// Safe data URLs are allowed

44

const safeImage = sanitizer("..."); // allowed

45

46

// Custom sanitizer implementation

47

const strictSanitizer = (value, tag, attribute) => {

48

// Block all external URLs

49

if (attribute === 'href' || attribute === 'src') {

50

if (value.startsWith('http://') || value.startsWith('https://')) {

51

return null; // Block external resources

52

}

53

}

54

55

// Use default sanitizer for other cases

56

return sanitizer(value);

57

};

58

59

// Apply custom sanitizer

60

const options = {

61

sanitizer: strictSanitizer

62

};

63

64

const result = compiler(markdownWithLinks, options);

65

```

66

67

### Slug Generation

68

69

Convert text strings to URL-safe slugs for heading IDs and anchor links.

70

71

```typescript { .api }

72

/**

73

* Convert string to URL-safe slug for heading IDs

74

* @param str - Input string to convert to slug

75

* @returns URL-safe slug string

76

*/

77

function slugify(str: string): string;

78

79

/**

80

* Custom slug generation function type

81

* @param input - Input string to convert

82

* @param defaultFn - Default slugify function to fall back to

83

* @returns Custom slug string

84

*/

85

type SlugifyFunction = (

86

input: string,

87

defaultFn: (input: string) => string

88

) => string;

89

```

90

91

**Usage Examples:**

92

93

```typescript

94

import { slugify } from "markdown-to-jsx";

95

96

// Basic slug generation

97

slugify("Hello World"); // "hello-world"

98

slugify("Special Characters!@#"); // "special-characters"

99

slugify("Àccented Tëxt"); // "accented-text"

100

slugify("Multiple Spaces"); // "multiple-spaces"

101

slugify("123 Numbers"); // "123-numbers"

102

103

// Custom slugify function

104

const customSlugify = (input, defaultFn) => {

105

// Add timestamp to make unique

106

const baseSlug = defaultFn(input);

107

return `${baseSlug}-${Date.now()}`;

108

};

109

110

// Custom slugify with prefix

111

const prefixedSlugify = (input, defaultFn) => {

112

const baseSlug = defaultFn(input);

113

return `section-${baseSlug}`;

114

};

115

116

// Apply custom slugify

117

const options = {

118

slugify: customSlugify

119

};

120

121

const markdownWithHeadings = `

122

# Main Title

123

## Sub Title

124

### Another Heading

125

`;

126

127

const result = compiler(markdownWithHeadings, options);

128

// Generates unique IDs: main-title-1634567890123, etc.

129

```

130

131

### Sanitizer Configuration in Options

132

133

Configure sanitization behavior through the options system for comprehensive security control.

134

135

**Usage Examples:**

136

137

```typescript

138

// Disable sanitization (not recommended for untrusted content)

139

const unsafeOptions = {

140

sanitizer: (value) => value // Pass through all values

141

};

142

143

// Strict sanitization

144

const strictOptions = {

145

sanitizer: (value, tag, attribute) => {

146

// Only allow specific domains

147

const allowedDomains = ['example.com', 'trusted-site.org'];

148

149

if (attribute === 'href' || attribute === 'src') {

150

try {

151

const url = new URL(value);

152

if (!allowedDomains.includes(url.hostname)) {

153

return null;

154

}

155

} catch {

156

// Invalid URL, allow relative links

157

if (!value.startsWith('/') && !value.startsWith('#')) {

158

return null;

159

}

160

}

161

}

162

163

// Use default sanitizer for validation

164

return sanitizer(value);

165

}

166

};

167

168

// Content-Security-Policy aware sanitizer

169

const cspSanitizer = (value, tag, attribute) => {

170

// Remove inline styles to comply with CSP

171

if (attribute === 'style') {

172

return null;

173

}

174

175

// Only allow relative URLs for images

176

if (tag === 'img' && attribute === 'src') {

177

if (!value.startsWith('/') && !value.startsWith('./')) {

178

return null;

179

}

180

}

181

182

return sanitizer(value);

183

};

184

```

185

186

### Advanced Utility Patterns

187

188

Complex utility function combinations for sophisticated markdown processing.

189

190

**Usage Examples:**

191

192

```typescript

193

// Utility function factory

194

const createSecureProcessor = (config) => {

195

const customSanitizer = (value, tag, attribute) => {

196

// Apply domain whitelist

197

if (config.allowedDomains && (attribute === 'href' || attribute === 'src')) {

198

try {

199

const url = new URL(value);

200

if (!config.allowedDomains.includes(url.hostname)) {

201

return config.blockExternal ? null : '#blocked';

202

}

203

} catch {

204

// Not a full URL, allow if relative

205

}

206

}

207

208

return sanitizer(value);

209

};

210

211

const customSlugify = (input, defaultFn) => {

212

const baseSlug = defaultFn(input);

213

return config.slugPrefix ? `${config.slugPrefix}-${baseSlug}` : baseSlug;

214

};

215

216

return {

217

sanitizer: customSanitizer,

218

slugify: customSlugify

219

};

220

};

221

222

// Usage

223

const secureConfig = createSecureProcessor({

224

allowedDomains: ['example.com', 'cdn.example.com'],

225

blockExternal: true,

226

slugPrefix: 'doc'

227

});

228

229

const options = {

230

...secureConfig,

231

enforceAtxHeadings: true,

232

disableAutoLink: false

233

};

234

235

// Debugging utilities

236

const debugSanitizer = (value, tag, attribute) => {

237

console.log('Sanitizing:', { value, tag, attribute });

238

const result = sanitizer(value);

239

console.log('Result:', result);

240

return result;

241

};

242

243

const debugSlugify = (input, defaultFn) => {

244

console.log('Slugifying:', input);

245

const result = defaultFn(input);

246

console.log('Slug result:', result);

247

return result;

248

};

249

250

// Performance monitoring utilities

251

const performanceSanitizer = (() => {

252

let callCount = 0;

253

let totalTime = 0;

254

255

return (value, tag, attribute) => {

256

const start = performance.now();

257

const result = sanitizer(value);

258

const end = performance.now();

259

260

callCount++;

261

totalTime += (end - start);

262

263

if (callCount % 100 === 0) {

264

console.log(`Sanitizer performance: ${callCount} calls, ${totalTime.toFixed(2)}ms total`);

265

}

266

267

return result;

268

};

269

})();

270

```

271

272

### Security Best Practices

273

274

Guidelines and patterns for secure markdown processing with the utility functions.

275

276

**Usage Examples:**

277

278

```typescript

279

// Secure configuration for user-generated content

280

const userContentOptions = {

281

// Use strict sanitization

282

sanitizer: (value, tag, attribute) => {

283

// Block all JavaScript

284

if (value.toLowerCase().includes('javascript:')) {

285

return null;

286

}

287

288

// Block data URLs except images

289

if (value.startsWith('data:') && !value.startsWith('data:image/')) {

290

return null;

291

}

292

293

// Use default sanitizer

294

return sanitizer(value);

295

},

296

297

// Disable HTML parsing for safety

298

disableParsingRawHTML: true,

299

300

// Use predictable slugs

301

slugify: (input, defaultFn) => {

302

// Remove potentially problematic characters

303

const cleaned = input.replace(/[<>]/g, '');

304

return defaultFn(cleaned);

305

}

306

};

307

308

// Configuration for trusted content

309

const trustedContentOptions = {

310

// Standard sanitization

311

sanitizer: sanitizer,

312

313

// Allow HTML parsing

314

disableParsingRawHTML: false,

315

316

// Enhanced slugs with metadata

317

slugify: (input, defaultFn) => {

318

const baseSlug = defaultFn(input);

319

// Add section prefix for navigation

320

return `section-${baseSlug}`;

321

}

322

};

323

324

// Apply based on content source

325

const processMarkdown = (content, isTrusted = false) => {

326

const options = isTrusted ? trustedContentOptions : userContentOptions;

327

return compiler(content, options);

328

};

329

```