scanner.h 3.02 KB
Newer Older
1
2
3
#ifndef SCANNER_H_62B23520_7C8E_11DE_8A39_0800200C9A66
#define SCANNER_H_62B23520_7C8E_11DE_8A39_0800200C9A66

4
#if defined(_MSC_VER) || (defined(__GNUC__) && (__GNUC__ == 3 && __GNUC_MINOR__ >= 4) || (__GNUC__ >= 4)) // GCC supports "pragma once" correctly since 3.4
5
6
7
#pragma once
#endif

8

9
10
11
12
13
#include <ios>
#include <string>
#include <queue>
#include <stack>
#include <set>
14
#include <map>
15
#include "ptr_vector.h"
16
17
18
19
20
#include "stream.h"
#include "token.h"

namespace YAML
{
21
	class Node;
22
	class RegEx;
23

24
25
26
27
28
29
30
31
32
33
34
35
	class Scanner
	{
	public:
		Scanner(std::istream& in);
		~Scanner();

		// token queue management (hopefully this looks kinda stl-ish)
		bool empty();
		void pop();
		Token& peek();

	private:
36
37
		struct IndentMarker {
			enum INDENT_TYPE { MAP, SEQ, NONE };
38
39
			enum STATUS { VALID, INVALID, UNKNOWN };
			IndentMarker(int column_, INDENT_TYPE type_): column(column_), type(type_), status(VALID), pStartToken(0) {}
40
41
42
		
			int column;
			INDENT_TYPE type;
43
			STATUS status;
44
			Token *pStartToken;
45
		};
46
47
		
		enum FLOW_MARKER { FLOW_MAP, FLOW_SEQ };
48
49
	
	private:	
50
51
52
53
54
55
		// scanning
		void EnsureTokensInQueue();
		void ScanNextToken();
		void ScanToNextToken();
		void StartStream();
		void EndStream();
56
		Token *PushToken(Token::TYPE type);
57
58
59
60
61
		
		bool InFlowContext() const { return !m_flows.empty(); }
		bool InBlockContext() const { return m_flows.empty(); }
		int GetFlowLevel() const { return m_flows.size(); }
		
62
		Token::TYPE GetStartTokenFor(IndentMarker::INDENT_TYPE type) const;
63
		IndentMarker *PushIndentTo(int column, IndentMarker::INDENT_TYPE type);
64
65
66
67
		void PopIndentToHere();
		void PopAllIndents();
		void PopIndent();
		int GetTopIndent() const;
68
69

		// checking input
70
		bool CanInsertPotentialSimpleKey() const;
71
72
73
74
75
76
		bool ExistsActiveSimpleKey() const;
		void InsertPotentialSimpleKey();
		void InvalidateSimpleKey();
		bool VerifySimpleKey();
		void PopAllSimpleKeys();
		
77
		void ThrowParserException(const std::string& msg) const;
78
79

		bool IsWhitespaceToBeEaten(char ch);
80
		const RegEx& GetValueRegex() const;
81
82

		struct SimpleKey {
83
			SimpleKey(const Mark& mark_, int flowLevel_);
84
85
86

			void Validate();
			void Invalidate();
87
88
89
			
			Mark mark;
			int flowLevel;
90
			IndentMarker *pIndent;
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
			Token *pMapStart, *pKey;
		};

		// and the tokens
		void ScanDirective();
		void ScanDocStart();
		void ScanDocEnd();
		void ScanBlockSeqStart();
		void ScanBlockMapSTart();
		void ScanBlockEnd();
		void ScanBlockEntry();
		void ScanFlowStart();
		void ScanFlowEnd();
		void ScanFlowEntry();
		void ScanKey();
		void ScanValue();
		void ScanAnchorOrAlias();
		void ScanTag();
		void ScanPlainScalar();
		void ScanQuotedScalar();
		void ScanBlockScalar();

	private:
		// the stream
		Stream INPUT;

		// the output (tokens)
118
		std::queue<Token> m_tokens;
119
120
121
122

		// state info
		bool m_startedStream, m_endedStream;
		bool m_simpleKeyAllowed;
123
		bool m_canBeJSONFlow;
124
125
126
127
		std::stack<SimpleKey> m_simpleKeys;
		std::stack<IndentMarker *> m_indents;
		ptr_vector<IndentMarker> m_indentRefs; // for "garbage collection"
		std::stack<FLOW_MARKER> m_flows;
128
129
	};
}
130
131

#endif // SCANNER_H_62B23520_7C8E_11DE_8A39_0800200C9A66
132