// Copyright (C) 2016 Kohei YOSHIDA. All rights reserved.
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of The BSD 3-Clause License
// that can be found in the LICENSE file.

package uritemplate

import (
	
	
	
)

type parseOp int

const (
	parseOpSimple parseOp = iota
	parseOpPlus
	parseOpCrosshatch
	parseOpDot
	parseOpSlash
	parseOpSemicolon
	parseOpQuestion
	parseOpAmpersand
)

var (
	rangeVarchar = &unicode.RangeTable{
		R16: []unicode.Range16{
			{Lo: 0x0030, Hi: 0x0039, Stride: 1}, // '0' - '9'
			{Lo: 0x0041, Hi: 0x005A, Stride: 1}, // 'A' - 'Z'
			{Lo: 0x005F, Hi: 0x005F, Stride: 1}, // '_'
			{Lo: 0x0061, Hi: 0x007A, Stride: 1}, // 'a' - 'z'
		},
		LatinOffset: 4,
	}
	rangeLiterals = &unicode.RangeTable{
		R16: []unicode.Range16{
			{Lo: 0x0021, Hi: 0x0021, Stride: 1}, // '!'
			{Lo: 0x0023, Hi: 0x0024, Stride: 1}, // '#' - '$'
			{Lo: 0x0026, Hi: 0x003B, Stride: 1}, // '&' ''' '(' - ';'. '''/27 used to be excluded but an errata is in the review process https://www.rfc-editor.org/errata/eid6937
			{Lo: 0x003D, Hi: 0x003D, Stride: 1}, // '='
			{Lo: 0x003F, Hi: 0x005B, Stride: 1}, // '?' - '['
			{Lo: 0x005D, Hi: 0x005D, Stride: 1}, // ']'
			{Lo: 0x005F, Hi: 0x005F, Stride: 1}, // '_'
			{Lo: 0x0061, Hi: 0x007A, Stride: 1}, // 'a' - 'z'
			{Lo: 0x007E, Hi: 0x007E, Stride: 1}, // '~'
			{Lo: 0x00A0, Hi: 0xD7FF, Stride: 1}, // ucschar
			{Lo: 0xE000, Hi: 0xF8FF, Stride: 1}, // iprivate
			{Lo: 0xF900, Hi: 0xFDCF, Stride: 1}, // ucschar
			{Lo: 0xFDF0, Hi: 0xFFEF, Stride: 1}, // ucschar
		},
		R32: []unicode.Range32{
			{Lo: 0x00010000, Hi: 0x0001FFFD, Stride: 1}, // ucschar
			{Lo: 0x00020000, Hi: 0x0002FFFD, Stride: 1}, // ucschar
			{Lo: 0x00030000, Hi: 0x0003FFFD, Stride: 1}, // ucschar
			{Lo: 0x00040000, Hi: 0x0004FFFD, Stride: 1}, // ucschar
			{Lo: 0x00050000, Hi: 0x0005FFFD, Stride: 1}, // ucschar
			{Lo: 0x00060000, Hi: 0x0006FFFD, Stride: 1}, // ucschar
			{Lo: 0x00070000, Hi: 0x0007FFFD, Stride: 1}, // ucschar
			{Lo: 0x00080000, Hi: 0x0008FFFD, Stride: 1}, // ucschar
			{Lo: 0x00090000, Hi: 0x0009FFFD, Stride: 1}, // ucschar
			{Lo: 0x000A0000, Hi: 0x000AFFFD, Stride: 1}, // ucschar
			{Lo: 0x000B0000, Hi: 0x000BFFFD, Stride: 1}, // ucschar
			{Lo: 0x000C0000, Hi: 0x000CFFFD, Stride: 1}, // ucschar
			{Lo: 0x000D0000, Hi: 0x000DFFFD, Stride: 1}, // ucschar
			{Lo: 0x000E1000, Hi: 0x000EFFFD, Stride: 1}, // ucschar
			{Lo: 0x000F0000, Hi: 0x000FFFFD, Stride: 1}, // iprivate
			{Lo: 0x00100000, Hi: 0x0010FFFD, Stride: 1}, // iprivate
		},
		LatinOffset: 10,
	}
)

type parser struct {
	r     string
	start int
	stop  int
	state parseState
}

func ( *parser) ( rune,  string,  ...interface{}) error {
	return fmt.Errorf("%s: %s%s", fmt.Sprintf(, ...), .r[0:.stop], string())
}

func ( *parser) () (rune, int) {
	,  := utf8.DecodeRuneInString(.r[.stop:])
	if  != utf8.RuneError {
		.stop += 
	}
	return , 
}

func ( *parser) ( rune) {
	.stop -= utf8.RuneLen()
}

type parseState int

const (
	parseStateDefault = parseState(iota)
	parseStateOperator
	parseStateVarList
	parseStateVarName
	parseStatePrefix
)

func ( *parser) ( parseState) {
	.state = 
	.start = .stop
}

func ( *parser) () (*Template, error) {
	 := Template{
		raw:   .r,
		exprs: []template{},
	}

	var  *expression
	for {
		,  := .rune()
		if  == utf8.RuneError {
			if  == 0 {
				if .state != parseStateDefault {
					return nil, .errorf('_', "incomplete expression")
				}
				if .start < .stop {
					.exprs = append(.exprs, literals(.r[.start:.stop]))
				}
				return &, nil
			}
			return nil, .errorf('_', "invalid UTF-8 sequence")
		}

		switch .state {
		case parseStateDefault:
			switch  {
			case '{':
				if  := .stop - ;  > .start {
					.exprs = append(.exprs, literals(.r[.start:]))
				}
				 = &expression{}
				.exprs = append(.exprs, )
				.setState(parseStateOperator)
			case '%':
				.unread()
				if  := .consumeTriplet();  != nil {
					return nil, 
				}
			default:
				if !unicode.Is(rangeLiterals, ) {
					.unread()
					return nil, .errorf('_', "unacceptable character (hint: use %%XX encoding)")
				}
			}
		case parseStateOperator:
			switch  {
			default:
				.unread()
				.op = parseOpSimple
			case '+':
				.op = parseOpPlus
			case '#':
				.op = parseOpCrosshatch
			case '.':
				.op = parseOpDot
			case '/':
				.op = parseOpSlash
			case ';':
				.op = parseOpSemicolon
			case '?':
				.op = parseOpQuestion
			case '&':
				.op = parseOpAmpersand
			case '=', ',', '!', '@', '|': // op-reserved
				return nil, .errorf('|', "unimplemented operator (op-reserved)")
			}
			.setState(parseStateVarName)
		case parseStateVarList:
			switch  {
			case ',':
				.setState(parseStateVarName)
			case '}':
				.init()
				.setState(parseStateDefault)
			default:
				.unread()
				return nil, .errorf('_', "unrecognized value modifier")
			}
		case parseStateVarName:
			switch  {
			case ':', '*':
				 := .r[.start : .stop-]
				if !isValidVarname() {
					return nil, .errorf('|', "unacceptable variable name")
				}
				 :=  == '*'
				.vars = append(.vars, varspec{
					name:    ,
					explode: ,
				})
				if  {
					.setState(parseStateVarList)
				} else {
					.setState(parseStatePrefix)
				}
			case ',', '}':
				.unread()
				 := .r[.start:.stop]
				if !isValidVarname() {
					return nil, .errorf('|', "unacceptable variable name")
				}
				.vars = append(.vars, varspec{
					name: ,
				})
				.setState(parseStateVarList)
			case '%':
				.unread()
				if  := .consumeTriplet();  != nil {
					return nil, 
				}
			case '.':
				if  := .stop - ;  == .start || .r[-1] == '.' {
					return nil, .errorf('|', "unacceptable variable name")
				}
			default:
				if !unicode.Is(rangeVarchar, ) {
					.unread()
					return nil, .errorf('_', "unacceptable variable name")
				}
			}
		case parseStatePrefix:
			 := &(.vars[len(.vars)-1])
			switch {
			case '0' <=  &&  <= '9':
				.maxlen *= 10
				.maxlen += int( - '0')
				if .maxlen == 0 || .maxlen > 9999 {
					return nil, .errorf('|', "max-length must be (0, 9999]")
				}
			default:
				.unread()
				if .maxlen == 0 {
					return nil, .errorf('_', "max-length must be (0, 9999]")
				}
				.setState(parseStateVarList)
			}
		default:
			.unread()
			panic(.errorf('_', "unhandled parseState(%d)", .state))
		}
	}
}

func isValidVarname( string) bool {
	if  := len();  == 0 || [0] == '.' || [-1] == '.' {
		return false
	}
	for  := 1;  < len()-1; ++ {
		switch  := [];  {
		case '.':
			if [-1] == '.' {
				return false
			}
		}
	}
	return true
}

func ( *parser) () error {
	if len(.r)-.stop < 3 || .r[.stop] != '%' || !ishex(.r[.stop+1]) || !ishex(.r[.stop+2]) {
		return .errorf('_', "incomplete pct-encodeed")
	}
	.stop += 3
	return nil
}