// Copyright 2010 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

package textproto

import (
	
	
	
)

// A Writer implements convenience methods for writing
// requests or responses to a text protocol network connection.
type Writer struct {
	W   *bufio.Writer
	dot *dotWriter
}

// NewWriter returns a new [Writer] writing to w.
func ( *bufio.Writer) *Writer {
	return &Writer{W: }
}

var crnl = []byte{'\r', '\n'}
var dotcrnl = []byte{'.', '\r', '\n'}

// PrintfLine writes the formatted output followed by \r\n.
func ( *Writer) ( string,  ...any) error {
	.closeDot()
	fmt.Fprintf(.W, , ...)
	.W.Write(crnl)
	return .W.Flush()
}

// DotWriter returns a writer that can be used to write a dot-encoding to w.
// It takes care of inserting leading dots when necessary,
// translating line-ending \n into \r\n, and adding the final .\r\n line
// when the DotWriter is closed. The caller should close the
// DotWriter before the next call to a method on w.
//
// See the documentation for the [Reader.DotReader] method for details about dot-encoding.
func ( *Writer) () io.WriteCloser {
	.closeDot()
	.dot = &dotWriter{w: }
	return .dot
}

func ( *Writer) () {
	if .dot != nil {
		.dot.Close() // sets w.dot = nil
	}
}

type dotWriter struct {
	w     *Writer
	state int
}

const (
	wstateBegin     = iota // initial state; must be zero
	wstateBeginLine        // beginning of line
	wstateCR               // wrote \r (possibly at end of line)
	wstateData             // writing data in middle of line
)

func ( *dotWriter) ( []byte) ( int,  error) {
	 := .w.W
	for  < len() {
		 := []
		switch .state {
		case wstateBegin, wstateBeginLine:
			.state = wstateData
			if  == '.' {
				// escape leading dot
				.WriteByte('.')
			}
			fallthrough

		case wstateData:
			if  == '\r' {
				.state = wstateCR
			}
			if  == '\n' {
				.WriteByte('\r')
				.state = wstateBeginLine
			}

		case wstateCR:
			.state = wstateData
			if  == '\n' {
				.state = wstateBeginLine
			}
		}
		if  = .WriteByte();  != nil {
			break
		}
		++
	}
	return
}

func ( *dotWriter) () error {
	if .w.dot ==  {
		.w.dot = nil
	}
	 := .w.W
	switch .state {
	default:
		.WriteByte('\r')
		fallthrough
	case wstateCR:
		.WriteByte('\n')
		fallthrough
	case wstateBeginLine:
		.Write(dotcrnl)
	}
	return .Flush()
}