scanner.h 3.05 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
	class Scanner
	{
	public:
		Scanner(std::istream& in);
		~Scanner();

		// token queue management (hopefully this looks kinda stl-ish)
		bool empty();
		void pop();
		Token& peek();
34
        Mark mark() const;
35
36

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

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

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

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

			void Validate();
			void Invalidate();
88
89
90
			
			Mark mark;
			int flowLevel;
91
			IndentMarker *pIndent;
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
118
			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)
119
		std::queue<Token> m_tokens;
120
121
122
123

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

#endif // SCANNER_H_62B23520_7C8E_11DE_8A39_0800200C9A66
133