openai.go 18.2 KB
Newer Older
1
// openai package provides core transformation logic for partial compatibility with the OpenAI REST API
2
3
4
package openai

import (
5
	"encoding/base64"
6
	"encoding/json"
Michael Yang's avatar
lint  
Michael Yang committed
7
	"errors"
8
	"fmt"
royjhan's avatar
royjhan committed
9
	"log/slog"
10
11
	"math/rand"
	"net/http"
12
	"strings"
13
14
	"time"

15
	"github.com/ollama/ollama/api"
16
	"github.com/ollama/ollama/types/model"
17
18
)

19
20
var finishReasonToolCalls = "tool_calls"

21
type Error struct {
22
23
24
25
	Message string  `json:"message"`
	Type    string  `json:"type"`
	Param   any     `json:"param"`
	Code    *string `json:"code"`
26
27
28
29
30
31
32
}

type ErrorResponse struct {
	Error Error `json:"error"`
}

type Message struct {
33
34
35
36
37
38
	Role       string     `json:"role"`
	Content    any        `json:"content"`
	Reasoning  string     `json:"reasoning,omitempty"`
	ToolCalls  []ToolCall `json:"tool_calls,omitempty"`
	Name       string     `json:"name,omitempty"`
	ToolCallID string     `json:"tool_call_id,omitempty"`
39
40
41
42
43
44
45
46
47
48
49
50
51
52
}

type Choice struct {
	Index        int     `json:"index"`
	Message      Message `json:"message"`
	FinishReason *string `json:"finish_reason"`
}

type ChunkChoice struct {
	Index        int     `json:"index"`
	Delta        Message `json:"delta"`
	FinishReason *string `json:"finish_reason"`
}

53
54
55
56
57
58
type CompleteChunkChoice struct {
	Text         string  `json:"text"`
	Index        int     `json:"index"`
	FinishReason *string `json:"finish_reason"`
}

59
60
61
62
63
64
65
type Usage struct {
	PromptTokens     int `json:"prompt_tokens"`
	CompletionTokens int `json:"completion_tokens"`
	TotalTokens      int `json:"total_tokens"`
}

type ResponseFormat struct {
66
67
68
69
70
	Type       string      `json:"type"`
	JsonSchema *JsonSchema `json:"json_schema,omitempty"`
}

type JsonSchema struct {
71
	Schema json.RawMessage `json:"schema"`
72
73
}

74
type EmbedRequest struct {
75
76
77
	Input      any    `json:"input"`
	Model      string `json:"model"`
	Dimensions int    `json:"dimensions,omitempty"`
78
79
}

80
81
82
83
type StreamOptions struct {
	IncludeUsage bool `json:"include_usage"`
}

Michael Yang's avatar
Michael Yang committed
84
85
86
87
type Reasoning struct {
	Effort *string `json:"effort,omitempty"`
}

88
89
90
91
type ChatCompletionRequest struct {
	Model            string          `json:"model"`
	Messages         []Message       `json:"messages"`
	Stream           bool            `json:"stream"`
92
	StreamOptions    *StreamOptions  `json:"stream_options"`
93
94
95
96
97
	MaxTokens        *int            `json:"max_tokens"`
	Seed             *int            `json:"seed"`
	Stop             any             `json:"stop"`
	Temperature      *float64        `json:"temperature"`
	FrequencyPenalty *float64        `json:"frequency_penalty"`
98
	PresencePenalty  *float64        `json:"presence_penalty"`
99
100
	TopP             *float64        `json:"top_p"`
	ResponseFormat   *ResponseFormat `json:"response_format"`
royjhan's avatar
royjhan committed
101
	Tools            []api.Tool      `json:"tools"`
Michael Yang's avatar
Michael Yang committed
102
	Reasoning        *Reasoning      `json:"reasoning,omitempty"`
103
	ReasoningEffort  *string         `json:"reasoning_effort,omitempty"`
Devon Rifkin's avatar
Devon Rifkin committed
104
	DebugRenderOnly  bool            `json:"_debug_render_only"`
105
106
107
}

type ChatCompletion struct {
Devon Rifkin's avatar
Devon Rifkin committed
108
109
110
111
112
113
114
115
	Id                string         `json:"id"`
	Object            string         `json:"object"`
	Created           int64          `json:"created"`
	Model             string         `json:"model"`
	SystemFingerprint string         `json:"system_fingerprint"`
	Choices           []Choice       `json:"choices"`
	Usage             Usage          `json:"usage,omitempty"`
	DebugInfo         *api.DebugInfo `json:"_debug_info,omitempty"`
116
117
118
119
120
121
122
123
124
}

type ChatCompletionChunk struct {
	Id                string        `json:"id"`
	Object            string        `json:"object"`
	Created           int64         `json:"created"`
	Model             string        `json:"model"`
	SystemFingerprint string        `json:"system_fingerprint"`
	Choices           []ChunkChoice `json:"choices"`
125
	Usage             *Usage        `json:"usage,omitempty"`
126
127
}

128
129
// TODO (https://github.com/ollama/ollama/issues/5259): support []string, []int and [][]int
type CompletionRequest struct {
130
131
132
133
134
135
136
137
138
139
140
141
	Model            string         `json:"model"`
	Prompt           string         `json:"prompt"`
	FrequencyPenalty float32        `json:"frequency_penalty"`
	MaxTokens        *int           `json:"max_tokens"`
	PresencePenalty  float32        `json:"presence_penalty"`
	Seed             *int           `json:"seed"`
	Stop             any            `json:"stop"`
	Stream           bool           `json:"stream"`
	StreamOptions    *StreamOptions `json:"stream_options"`
	Temperature      *float32       `json:"temperature"`
	TopP             float32        `json:"top_p"`
	Suffix           string         `json:"suffix"`
Devon Rifkin's avatar
Devon Rifkin committed
142
	DebugRenderOnly  bool           `json:"_debug_render_only"`
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
}

type Completion struct {
	Id                string                `json:"id"`
	Object            string                `json:"object"`
	Created           int64                 `json:"created"`
	Model             string                `json:"model"`
	SystemFingerprint string                `json:"system_fingerprint"`
	Choices           []CompleteChunkChoice `json:"choices"`
	Usage             Usage                 `json:"usage,omitempty"`
}

type CompletionChunk struct {
	Id                string                `json:"id"`
	Object            string                `json:"object"`
	Created           int64                 `json:"created"`
	Choices           []CompleteChunkChoice `json:"choices"`
	Model             string                `json:"model"`
	SystemFingerprint string                `json:"system_fingerprint"`
162
	Usage             *Usage                `json:"usage,omitempty"`
163
164
}

royjhan's avatar
royjhan committed
165
166
type ToolCall struct {
	ID       string `json:"id"`
167
	Index    int    `json:"index"`
royjhan's avatar
royjhan committed
168
169
170
171
172
173
174
	Type     string `json:"type"`
	Function struct {
		Name      string `json:"name"`
		Arguments string `json:"arguments"`
	} `json:"function"`
}

175
176
177
178
179
180
181
type Model struct {
	Id      string `json:"id"`
	Object  string `json:"object"`
	Created int64  `json:"created"`
	OwnedBy string `json:"owned_by"`
}

182
183
184
185
186
187
type Embedding struct {
	Object    string    `json:"object"`
	Embedding []float32 `json:"embedding"`
	Index     int       `json:"index"`
}

188
189
190
191
192
type ListCompletion struct {
	Object string  `json:"object"`
	Data   []Model `json:"data"`
}

193
type EmbeddingList struct {
194
195
196
197
198
199
200
201
202
	Object string         `json:"object"`
	Data   []Embedding    `json:"data"`
	Model  string         `json:"model"`
	Usage  EmbeddingUsage `json:"usage,omitempty"`
}

type EmbeddingUsage struct {
	PromptTokens int `json:"prompt_tokens"`
	TotalTokens  int `json:"total_tokens"`
203
204
}

205
206
207
208
209
210
211
212
213
214
215
216
217
218
func NewError(code int, message string) ErrorResponse {
	var etype string
	switch code {
	case http.StatusBadRequest:
		etype = "invalid_request_error"
	case http.StatusNotFound:
		etype = "not_found_error"
	default:
		etype = "api_error"
	}

	return ErrorResponse{Error{Type: etype, Message: message}}
}

219
220
// ToUsage converts an api.ChatResponse to Usage
func ToUsage(r api.ChatResponse) Usage {
221
	return Usage{
222
223
224
		PromptTokens:     r.Metrics.PromptEvalCount,
		CompletionTokens: r.Metrics.EvalCount,
		TotalTokens:      r.Metrics.PromptEvalCount + r.Metrics.EvalCount,
225
226
227
	}
}

royjhan's avatar
royjhan committed
228
229
230
231
232
233
234
235
236
func toolCallId() string {
	const letterBytes = "abcdefghijklmnopqrstuvwxyz0123456789"
	b := make([]byte, 8)
	for i := range b {
		b[i] = letterBytes[rand.Intn(len(letterBytes))]
	}
	return "call_" + strings.ToLower(string(b))
}

237
238
239
func toToolCalls(tc []api.ToolCall) []ToolCall {
	toolCalls := make([]ToolCall, len(tc))
	for i, tc := range tc {
royjhan's avatar
royjhan committed
240
241
242
		toolCalls[i].ID = toolCallId()
		toolCalls[i].Type = "function"
		toolCalls[i].Function.Name = tc.Function.Name
243
		toolCalls[i].Index = tc.Function.Index
royjhan's avatar
royjhan committed
244
245
246
247
248
249
250
251
252

		args, err := json.Marshal(tc.Function.Arguments)
		if err != nil {
			slog.Error("could not marshall function arguments to json", "error", err)
			continue
		}

		toolCalls[i].Function.Arguments = string(args)
	}
253
254
	return toolCalls
}
royjhan's avatar
royjhan committed
255

256
257
// ToChatCompletion converts an api.ChatResponse to ChatCompletion
func ToChatCompletion(id string, r api.ChatResponse) ChatCompletion {
258
	toolCalls := toToolCalls(r.Message.ToolCalls)
259
260
261
262
263
264
265
	return ChatCompletion{
		Id:                id,
		Object:            "chat.completion",
		Created:           r.CreatedAt.Unix(),
		Model:             r.Model,
		SystemFingerprint: "fp_ollama",
		Choices: []Choice{{
266
			Index:   0,
Michael Yang's avatar
Michael Yang committed
267
			Message: Message{Role: r.Message.Role, Content: r.Message.Content, ToolCalls: toolCalls, Reasoning: r.Message.Thinking},
268
			FinishReason: func(reason string) *string {
269
270
271
				if len(toolCalls) > 0 {
					reason = "tool_calls"
				}
272
273
274
275
276
				if len(reason) > 0 {
					return &reason
				}
				return nil
			}(r.DoneReason),
277
		}}, Usage: ToUsage(r),
Devon Rifkin's avatar
Devon Rifkin committed
278
		DebugInfo: r.DebugInfo,
279
280
281
	}
}

282
283
// ToChunk converts an api.ChatResponse to ChatCompletionChunk
func ToChunk(id string, r api.ChatResponse, toolCallSent bool) ChatCompletionChunk {
284
	toolCalls := toToolCalls(r.Message.ToolCalls)
285
286
287
288
289
290
	return ChatCompletionChunk{
		Id:                id,
		Object:            "chat.completion.chunk",
		Created:           time.Now().Unix(),
		Model:             r.Model,
		SystemFingerprint: "fp_ollama",
291
292
		Choices: []ChunkChoice{{
			Index: 0,
Michael Yang's avatar
Michael Yang committed
293
			Delta: Message{Role: "assistant", Content: r.Message.Content, ToolCalls: toolCalls, Reasoning: r.Message.Thinking},
294
295
			FinishReason: func(reason string) *string {
				if len(reason) > 0 {
Michael Yang's avatar
Michael Yang committed
296
					if toolCallSent || len(toolCalls) > 0 {
297
298
						return &finishReasonToolCalls
					}
299
300
301
302
303
					return &reason
				}
				return nil
			}(r.DoneReason),
		}},
304
305
306
	}
}

307
308
// ToUsageGenerate converts an api.GenerateResponse to Usage
func ToUsageGenerate(r api.GenerateResponse) Usage {
309
	return Usage{
310
311
312
		PromptTokens:     r.Metrics.PromptEvalCount,
		CompletionTokens: r.Metrics.EvalCount,
		TotalTokens:      r.Metrics.PromptEvalCount + r.Metrics.EvalCount,
313
314
315
	}
}

316
317
// ToCompletion converts an api.GenerateResponse to Completion
func ToCompletion(id string, r api.GenerateResponse) Completion {
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
	return Completion{
		Id:                id,
		Object:            "text_completion",
		Created:           r.CreatedAt.Unix(),
		Model:             r.Model,
		SystemFingerprint: "fp_ollama",
		Choices: []CompleteChunkChoice{{
			Text:  r.Response,
			Index: 0,
			FinishReason: func(reason string) *string {
				if len(reason) > 0 {
					return &reason
				}
				return nil
			}(r.DoneReason),
		}},
334
		Usage: ToUsageGenerate(r),
335
336
337
	}
}

338
339
// ToCompleteChunk converts an api.GenerateResponse to CompletionChunk
func ToCompleteChunk(id string, r api.GenerateResponse) CompletionChunk {
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
	return CompletionChunk{
		Id:                id,
		Object:            "text_completion",
		Created:           time.Now().Unix(),
		Model:             r.Model,
		SystemFingerprint: "fp_ollama",
		Choices: []CompleteChunkChoice{{
			Text:  r.Response,
			Index: 0,
			FinishReason: func(reason string) *string {
				if len(reason) > 0 {
					return &reason
				}
				return nil
			}(r.DoneReason),
		}},
	}
}

359
360
// ToListCompletion converts an api.ListResponse to ListCompletion
func ToListCompletion(r api.ListResponse) ListCompletion {
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
	var data []Model
	for _, m := range r.Models {
		data = append(data, Model{
			Id:      m.Name,
			Object:  "model",
			Created: m.ModifiedAt.Unix(),
			OwnedBy: model.ParseName(m.Name).Namespace,
		})
	}

	return ListCompletion{
		Object: "list",
		Data:   data,
	}
}

377
378
// ToEmbeddingList converts an api.EmbedResponse to EmbeddingList
func ToEmbeddingList(model string, r api.EmbedResponse) EmbeddingList {
379
380
381
382
383
384
385
386
387
388
389
390
391
392
	if r.Embeddings != nil {
		var data []Embedding
		for i, e := range r.Embeddings {
			data = append(data, Embedding{
				Object:    "embedding",
				Embedding: e,
				Index:     i,
			})
		}

		return EmbeddingList{
			Object: "list",
			Data:   data,
			Model:  model,
393
394
395
396
			Usage: EmbeddingUsage{
				PromptTokens: r.PromptEvalCount,
				TotalTokens:  r.PromptEvalCount,
			},
397
398
399
400
401
402
		}
	}

	return EmbeddingList{}
}

403
404
// ToModel converts an api.ShowResponse to Model
func ToModel(r api.ShowResponse, m string) Model {
405
406
407
408
409
410
411
412
	return Model{
		Id:      m,
		Object:  "model",
		Created: r.ModifiedAt.Unix(),
		OwnedBy: model.ParseName(m).Namespace,
	}
}

413
414
// FromChatRequest converts a ChatCompletionRequest to api.ChatRequest
func FromChatRequest(r ChatCompletionRequest) (*api.ChatRequest, error) {
415
416
	var messages []api.Message
	for _, msg := range r.Messages {
417
418
419
420
421
422
423
		toolName := ""
		if strings.ToLower(msg.Role) == "tool" {
			toolName = msg.Name
			if toolName == "" && msg.ToolCallID != "" {
				toolName = nameFromToolCallID(r.Messages, msg.ToolCallID)
			}
		}
424
425
		switch content := msg.Content.(type) {
		case string:
426
427
428
429
			toolCalls, err := fromCompletionToolCall(msg.ToolCalls)
			if err != nil {
				return nil, err
			}
430
			messages = append(messages, api.Message{Role: msg.Role, Content: content, Thinking: msg.Reasoning, ToolCalls: toolCalls, ToolName: toolName})
431
432
433
434
		case []any:
			for _, c := range content {
				data, ok := c.(map[string]any)
				if !ok {
Michael Yang's avatar
lint  
Michael Yang committed
435
					return nil, errors.New("invalid message format")
436
437
438
439
440
				}
				switch data["type"] {
				case "text":
					text, ok := data["text"].(string)
					if !ok {
Michael Yang's avatar
lint  
Michael Yang committed
441
						return nil, errors.New("invalid message format")
442
					}
443
					messages = append(messages, api.Message{Role: msg.Role, Content: text})
444
445
446
447
				case "image_url":
					var url string
					if urlMap, ok := data["image_url"].(map[string]any); ok {
						if url, ok = urlMap["url"].(string); !ok {
Michael Yang's avatar
lint  
Michael Yang committed
448
							return nil, errors.New("invalid message format")
449
450
451
						}
					} else {
						if url, ok = data["image_url"].(string); !ok {
Michael Yang's avatar
lint  
Michael Yang committed
452
							return nil, errors.New("invalid message format")
453
454
455
						}
					}

456
					types := []string{"jpeg", "jpg", "png", "webp"}
457
458
459
460
461
462
463
464
465
466
467
					valid := false
					for _, t := range types {
						prefix := "data:image/" + t + ";base64,"
						if strings.HasPrefix(url, prefix) {
							url = strings.TrimPrefix(url, prefix)
							valid = true
							break
						}
					}

					if !valid {
Michael Yang's avatar
lint  
Michael Yang committed
468
						return nil, errors.New("invalid image input")
469
470
471
472
					}

					img, err := base64.StdEncoding.DecodeString(url)
					if err != nil {
Michael Yang's avatar
lint  
Michael Yang committed
473
						return nil, errors.New("invalid message format")
474
					}
475
476

					messages = append(messages, api.Message{Role: msg.Role, Images: []api.ImageData{img}})
477
				default:
Michael Yang's avatar
lint  
Michael Yang committed
478
					return nil, errors.New("invalid message format")
479
480
				}
			}
481
482
483
484
485
486
487
488
			// since we might have added multiple messages above, if we have tools
			// calls we'll add them to the last message
			if len(messages) > 0 && len(msg.ToolCalls) > 0 {
				toolCalls, err := fromCompletionToolCall(msg.ToolCalls)
				if err != nil {
					return nil, err
				}
				messages[len(messages)-1].ToolCalls = toolCalls
489
490
491
				if toolName != "" {
					messages[len(messages)-1].ToolName = toolName
				}
492
				messages[len(messages)-1].Thinking = msg.Reasoning
493
			}
494
		default:
495
			// content is only optional if tool calls are present
royjhan's avatar
royjhan committed
496
497
498
499
500
501
502
503
504
			if msg.ToolCalls == nil {
				return nil, fmt.Errorf("invalid message content type: %T", content)
			}

			toolCalls := make([]api.ToolCall, len(msg.ToolCalls))
			for i, tc := range msg.ToolCalls {
				toolCalls[i].Function.Name = tc.Function.Name
				err := json.Unmarshal([]byte(tc.Function.Arguments), &toolCalls[i].Function.Arguments)
				if err != nil {
Michael Yang's avatar
lint  
Michael Yang committed
505
					return nil, errors.New("invalid tool call arguments")
royjhan's avatar
royjhan committed
506
507
				}
			}
508
			messages = append(messages, api.Message{Role: msg.Role, Thinking: msg.Reasoning, ToolCalls: toolCalls})
509
		}
510
511
	}

512
	options := make(map[string]any)
513
514
515
516

	switch stop := r.Stop.(type) {
	case string:
		options["stop"] = []string{stop}
517
	case []any:
518
519
520
521
522
523
524
525
526
527
528
529
530
531
		var stops []string
		for _, s := range stop {
			if str, ok := s.(string); ok {
				stops = append(stops, str)
			}
		}
		options["stop"] = stops
	}

	if r.MaxTokens != nil {
		options["num_predict"] = *r.MaxTokens
	}

	if r.Temperature != nil {
532
		options["temperature"] = *r.Temperature
533
534
535
536
537
538
539
540
541
	} else {
		options["temperature"] = 1.0
	}

	if r.Seed != nil {
		options["seed"] = *r.Seed
	}

	if r.FrequencyPenalty != nil {
542
		options["frequency_penalty"] = *r.FrequencyPenalty
543
544
545
	}

	if r.PresencePenalty != nil {
546
		options["presence_penalty"] = *r.PresencePenalty
547
548
549
550
551
552
553
554
	}

	if r.TopP != nil {
		options["top_p"] = *r.TopP
	} else {
		options["top_p"] = 1.0
	}

555
556
557
558
559
560
561
562
	var format json.RawMessage
	if r.ResponseFormat != nil {
		switch strings.ToLower(strings.TrimSpace(r.ResponseFormat.Type)) {
		// Support the old "json_object" type for OpenAI compatibility
		case "json_object":
			format = json.RawMessage(`"json"`)
		case "json_schema":
			if r.ResponseFormat.JsonSchema != nil {
563
				format = r.ResponseFormat.JsonSchema.Schema
564
565
			}
		}
566
567
	}

Michael Yang's avatar
Michael Yang committed
568
569
570
571
572
	var think *api.ThinkValue
	if r.Reasoning != nil {
		think = &api.ThinkValue{
			Value: *r.Reasoning.Effort,
		}
573
574
575
576
	} else if r.ReasoningEffort != nil {
		think = &api.ThinkValue{
			Value: *r.ReasoningEffort,
		}
Michael Yang's avatar
Michael Yang committed
577
578
	}

579
	return &api.ChatRequest{
Devon Rifkin's avatar
Devon Rifkin committed
580
581
582
583
584
585
586
587
		Model:           r.Model,
		Messages:        messages,
		Format:          format,
		Options:         options,
		Stream:          &r.Stream,
		Tools:           r.Tools,
		Think:           think,
		DebugRenderOnly: r.DebugRenderOnly,
588
	}, nil
589
590
}

591
592
593
594
595
596
597
598
599
600
601
602
603
604
func nameFromToolCallID(messages []Message, toolCallID string) string {
	// iterate backwards to be more resilient to duplicate tool call IDs (this
	// follows "last one wins")
	for i := len(messages) - 1; i >= 0; i-- {
		msg := messages[i]
		for _, tc := range msg.ToolCalls {
			if tc.ID == toolCallID {
				return tc.Function.Name
			}
		}
	}
	return ""
}

605
606
607
608
609
610
611
612
613
614
615
616
617
func fromCompletionToolCall(toolCalls []ToolCall) ([]api.ToolCall, error) {
	apiToolCalls := make([]api.ToolCall, len(toolCalls))
	for i, tc := range toolCalls {
		apiToolCalls[i].Function.Name = tc.Function.Name
		err := json.Unmarshal([]byte(tc.Function.Arguments), &apiToolCalls[i].Function.Arguments)
		if err != nil {
			return nil, errors.New("invalid tool call arguments")
		}
	}

	return apiToolCalls, nil
}

618
619
// FromCompleteRequest converts a CompletionRequest to api.GenerateRequest
func FromCompleteRequest(r CompletionRequest) (api.GenerateRequest, error) {
620
621
622
623
624
	options := make(map[string]any)

	switch stop := r.Stop.(type) {
	case string:
		options["stop"] = []string{stop}
625
626
627
628
629
630
631
632
	case []any:
		var stops []string
		for _, s := range stop {
			if str, ok := s.(string); ok {
				stops = append(stops, str)
			} else {
				return api.GenerateRequest{}, fmt.Errorf("invalid type for 'stop' field: %T", s)
			}
633
		}
634
		options["stop"] = stops
635
636
637
638
639
640
641
	}

	if r.MaxTokens != nil {
		options["num_predict"] = *r.MaxTokens
	}

	if r.Temperature != nil {
642
		options["temperature"] = *r.Temperature
643
644
645
646
647
648
649
650
	} else {
		options["temperature"] = 1.0
	}

	if r.Seed != nil {
		options["seed"] = *r.Seed
	}

651
	options["frequency_penalty"] = r.FrequencyPenalty
652

653
	options["presence_penalty"] = r.PresencePenalty
654
655
656
657
658
659
660
661

	if r.TopP != 0.0 {
		options["top_p"] = r.TopP
	} else {
		options["top_p"] = 1.0
	}

	return api.GenerateRequest{
Devon Rifkin's avatar
Devon Rifkin committed
662
663
664
665
666
667
		Model:           r.Model,
		Prompt:          r.Prompt,
		Options:         options,
		Stream:          &r.Stream,
		Suffix:          r.Suffix,
		DebugRenderOnly: r.DebugRenderOnly,
668
669
	}, nil
}