| /* |
| * Copyright (C) 2015 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef AAPT_DIAGNOSTICS_H |
| #define AAPT_DIAGNOSTICS_H |
| |
| #include "Source.h" |
| #include "util/StringPiece.h" |
| #include "util/Util.h" |
| |
| #include <android-base/macros.h> |
| #include <iostream> |
| #include <sstream> |
| #include <string> |
| |
| namespace aapt { |
| |
| struct DiagMessageActual { |
| Source source; |
| std::string message; |
| }; |
| |
| struct DiagMessage { |
| private: |
| Source mSource; |
| std::stringstream mMessage; |
| |
| public: |
| DiagMessage() = default; |
| |
| DiagMessage(const StringPiece& src) : mSource(src) { |
| } |
| |
| DiagMessage(const Source& src) : mSource(src) { |
| } |
| |
| DiagMessage(size_t line) : mSource(Source().withLine(line)) { |
| } |
| |
| template <typename T> |
| DiagMessage& operator<<(const T& value) { |
| mMessage << value; |
| return *this; |
| } |
| |
| DiagMessageActual build() const { |
| return DiagMessageActual{ mSource, mMessage.str() }; |
| } |
| }; |
| |
| struct IDiagnostics { |
| virtual ~IDiagnostics() = default; |
| |
| enum class Level { |
| Note, |
| Warn, |
| Error |
| }; |
| |
| virtual void log(Level level, DiagMessageActual& actualMsg) = 0; |
| |
| virtual void error(const DiagMessage& message) { |
| DiagMessageActual actual = message.build(); |
| log(Level::Error, actual); |
| } |
| |
| virtual void warn(const DiagMessage& message) { |
| DiagMessageActual actual = message.build(); |
| log(Level::Warn, actual); |
| } |
| |
| virtual void note(const DiagMessage& message) { |
| DiagMessageActual actual = message.build(); |
| log(Level::Note, actual); |
| } |
| }; |
| |
| class StdErrDiagnostics : public IDiagnostics { |
| public: |
| StdErrDiagnostics() = default; |
| |
| void log(Level level, DiagMessageActual& actualMsg) override { |
| const char* tag; |
| |
| switch (level) { |
| case Level::Error: |
| mNumErrors++; |
| if (mNumErrors > 20) { |
| return; |
| } |
| tag = "error"; |
| break; |
| |
| case Level::Warn: |
| tag = "warn"; |
| break; |
| |
| case Level::Note: |
| tag = "note"; |
| break; |
| } |
| |
| if (!actualMsg.source.path.empty()) { |
| std::cerr << actualMsg.source << ": "; |
| } |
| std::cerr << tag << ": " << actualMsg.message << "." << std::endl; |
| } |
| |
| private: |
| size_t mNumErrors = 0; |
| |
| DISALLOW_COPY_AND_ASSIGN(StdErrDiagnostics); |
| }; |
| |
| class SourcePathDiagnostics : public IDiagnostics { |
| public: |
| SourcePathDiagnostics(const Source& src, IDiagnostics* diag) : mSource(src), mDiag(diag) { |
| } |
| |
| void log(Level level, DiagMessageActual& actualMsg) override { |
| actualMsg.source.path = mSource.path; |
| mDiag->log(level, actualMsg); |
| } |
| |
| private: |
| Source mSource; |
| IDiagnostics* mDiag; |
| |
| DISALLOW_COPY_AND_ASSIGN(SourcePathDiagnostics); |
| }; |
| |
| } // namespace aapt |
| |
| #endif /* AAPT_DIAGNOSTICS_H */ |