/*
 * 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.
 */

#include "flatten/XmlFlattener.h"
#include "link/Linkers.h"
#include "test/Builders.h"
#include "test/Context.h"
#include "util/BigBuffer.h"
#include "util/Util.h"

#include <androidfw/ResourceTypes.h>
#include <gtest/gtest.h>

namespace aapt {

class XmlFlattenerTest : public ::testing::Test {
public:
    void SetUp() override {
        mContext = test::ContextBuilder()
                .setCompilationPackage(u"com.app.test")
                .setNameManglerPolicy(NameManglerPolicy{ u"com.app.test" })
                .addSymbolSource(test::StaticSymbolSourceBuilder()
                        .addSymbol(u"@android:attr/id", ResourceId(0x010100d0),
                                   test::AttributeBuilder().build())
                        .addSymbol(u"@com.app.test:id/id", ResourceId(0x7f020000))
                        .addSymbol(u"@android:attr/paddingStart", ResourceId(0x010103b3),
                                   test::AttributeBuilder().build())
                        .addSymbol(u"@android:attr/colorAccent", ResourceId(0x01010435),
                                   test::AttributeBuilder().build())
                        .build())
                .build();
    }

    ::testing::AssertionResult flatten(xml::XmlResource* doc, android::ResXMLTree* outTree,
                                       XmlFlattenerOptions options = {}) {
        using namespace android; // For NO_ERROR on windows because it is a macro.

        BigBuffer buffer(1024);
        XmlFlattener flattener(&buffer, options);
        if (!flattener.consume(mContext.get(), doc)) {
            return ::testing::AssertionFailure() << "failed to flatten XML Tree";
        }

        std::unique_ptr<uint8_t[]> data = util::copy(buffer);
        if (outTree->setTo(data.get(), buffer.size(), true) != NO_ERROR) {
            return ::testing::AssertionFailure() << "flattened XML is corrupt";
        }
        return ::testing::AssertionSuccess();
    }

protected:
    std::unique_ptr<IAaptContext> mContext;
};

TEST_F(XmlFlattenerTest, FlattenXmlWithNoCompiledAttributes) {
    std::unique_ptr<xml::XmlResource> doc = test::buildXmlDom(R"EOF(
            <View xmlns:test="http://com.test"
                  attr="hey">
              <Layout test:hello="hi" />
              <Layout>Some text</Layout>
            </View>)EOF");


    android::ResXMLTree tree;
    ASSERT_TRUE(flatten(doc.get(), &tree));

    ASSERT_EQ(tree.next(), android::ResXMLTree::START_NAMESPACE);

    size_t len;
    const char16_t* namespacePrefix = tree.getNamespacePrefix(&len);
    EXPECT_EQ(StringPiece16(namespacePrefix, len), u"test");

    const char16_t* namespaceUri = tree.getNamespaceUri(&len);
    ASSERT_EQ(StringPiece16(namespaceUri, len), u"http://com.test");

    ASSERT_EQ(tree.next(), android::ResXMLTree::START_TAG);

    ASSERT_EQ(tree.getElementNamespace(&len), nullptr);
    const char16_t* tagName = tree.getElementName(&len);
    EXPECT_EQ(StringPiece16(tagName, len), u"View");

    ASSERT_EQ(1u, tree.getAttributeCount());
    ASSERT_EQ(tree.getAttributeNamespace(0, &len), nullptr);
    const char16_t* attrName = tree.getAttributeName(0, &len);
    EXPECT_EQ(StringPiece16(attrName, len), u"attr");

    EXPECT_EQ(0, tree.indexOfAttribute(nullptr, 0, u"attr", StringPiece16(u"attr").size()));

    ASSERT_EQ(tree.next(), android::ResXMLTree::START_TAG);

    ASSERT_EQ(tree.getElementNamespace(&len), nullptr);
    tagName = tree.getElementName(&len);
    EXPECT_EQ(StringPiece16(tagName, len), u"Layout");

    ASSERT_EQ(1u, tree.getAttributeCount());
    const char16_t* attrNamespace = tree.getAttributeNamespace(0, &len);
    EXPECT_EQ(StringPiece16(attrNamespace, len), u"http://com.test");

    attrName = tree.getAttributeName(0, &len);
    EXPECT_EQ(StringPiece16(attrName, len), u"hello");

    ASSERT_EQ(tree.next(), android::ResXMLTree::END_TAG);
    ASSERT_EQ(tree.next(), android::ResXMLTree::START_TAG);

    ASSERT_EQ(tree.getElementNamespace(&len), nullptr);
    tagName = tree.getElementName(&len);
    EXPECT_EQ(StringPiece16(tagName, len), u"Layout");
    ASSERT_EQ(0u, tree.getAttributeCount());

    ASSERT_EQ(tree.next(), android::ResXMLTree::TEXT);
    const char16_t* text = tree.getText(&len);
    EXPECT_EQ(StringPiece16(text, len), u"Some text");

    ASSERT_EQ(tree.next(), android::ResXMLTree::END_TAG);
    ASSERT_EQ(tree.getElementNamespace(&len), nullptr);
    tagName = tree.getElementName(&len);
    EXPECT_EQ(StringPiece16(tagName, len), u"Layout");

    ASSERT_EQ(tree.next(), android::ResXMLTree::END_TAG);
    ASSERT_EQ(tree.getElementNamespace(&len), nullptr);
    tagName = tree.getElementName(&len);
    EXPECT_EQ(StringPiece16(tagName, len), u"View");

    ASSERT_EQ(tree.next(), android::ResXMLTree::END_NAMESPACE);
    namespacePrefix = tree.getNamespacePrefix(&len);
    EXPECT_EQ(StringPiece16(namespacePrefix, len), u"test");

    namespaceUri = tree.getNamespaceUri(&len);
    ASSERT_EQ(StringPiece16(namespaceUri, len), u"http://com.test");

    ASSERT_EQ(tree.next(), android::ResXMLTree::END_DOCUMENT);
}

TEST_F(XmlFlattenerTest, FlattenCompiledXmlAndStripSdk21) {
    std::unique_ptr<xml::XmlResource> doc = test::buildXmlDom(R"EOF(
            <View xmlns:android="http://schemas.android.com/apk/res/android"
                android:paddingStart="1dp"
                android:colorAccent="#ffffff"/>)EOF");

    XmlReferenceLinker linker;
    ASSERT_TRUE(linker.consume(mContext.get(), doc.get()));
    ASSERT_TRUE(linker.getSdkLevels().count(17) == 1);
    ASSERT_TRUE(linker.getSdkLevels().count(21) == 1);

    android::ResXMLTree tree;
    XmlFlattenerOptions options;
    options.maxSdkLevel = 17;
    ASSERT_TRUE(flatten(doc.get(), &tree, options));

    while (tree.next() != android::ResXMLTree::START_TAG) {
        ASSERT_NE(tree.getEventType(), android::ResXMLTree::BAD_DOCUMENT);
        ASSERT_NE(tree.getEventType(), android::ResXMLTree::END_DOCUMENT);
    }

    ASSERT_EQ(1u, tree.getAttributeCount());
    EXPECT_EQ(uint32_t(0x010103b3), tree.getAttributeNameResID(0));
}

TEST_F(XmlFlattenerTest, AssignSpecialAttributeIndices) {
    std::unique_ptr<xml::XmlResource> doc = test::buildXmlDom(R"EOF(
            <View xmlns:android="http://schemas.android.com/apk/res/android"
                  android:id="@id/id"
                  class="str"
                  style="@id/id"/>)EOF");

    android::ResXMLTree tree;
    ASSERT_TRUE(flatten(doc.get(), &tree));

    while (tree.next() != android::ResXMLTree::START_TAG) {
        ASSERT_NE(tree.getEventType(), android::ResXMLTree::BAD_DOCUMENT);
        ASSERT_NE(tree.getEventType(), android::ResXMLTree::END_DOCUMENT);
    }

    EXPECT_EQ(tree.indexOfClass(), 0);
    EXPECT_EQ(tree.indexOfStyle(), 1);
}

/*
 * The device ResXMLParser in libandroidfw differentiates between empty namespace and null
 * namespace.
 */
TEST_F(XmlFlattenerTest, NoNamespaceIsNotTheSameAsEmptyNamespace) {
    std::unique_ptr<xml::XmlResource> doc = test::buildXmlDom("<View package=\"android\"/>");

    android::ResXMLTree tree;
    ASSERT_TRUE(flatten(doc.get(), &tree));

    while (tree.next() != android::ResXMLTree::START_TAG) {
        ASSERT_NE(tree.getEventType(), android::ResXMLTree::BAD_DOCUMENT);
        ASSERT_NE(tree.getEventType(), android::ResXMLTree::END_DOCUMENT);
    }

    const StringPiece16 kPackage = u"package";
    EXPECT_GE(tree.indexOfAttribute(nullptr, 0, kPackage.data(), kPackage.size()), 0);
}

} // namespace aapt