/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.
 */
/*
 * Description  : This test case is to verify the fix for issue785
 * https://code.google.com/p/asterixdb/issues/detail?id=785
 * Expected Res : SUCCESS
 * Date         : 2nd Oct. 2014
 */

drop  dataverse tpch if exists;
create  dataverse tpch;

use tpch;


create type tpch.OrderType as
 closed {
  o_orderkey : int64,
  o_custkey : int64,
  o_orderstatus : string,
  o_totalprice : double,
  o_orderdate : string,
  o_orderpriority : string,
  o_clerk : string,
  o_shippriority : int64,
  o_comment : string
}

create type tpch.CustomerType as
 closed {
  c_custkey : int64,
  c_name : string,
  c_address : string,
  c_nationkey : int64,
  c_phone : string,
  c_acctbal : double,
  c_mktsegment : string,
  c_comment : string
}

create type tpch.SupplierType as
 closed {
  s_suppkey : int64,
  s_name : string,
  s_address : string,
  s_nationkey : int64,
  s_phone : string,
  s_acctbal : double,
  s_comment : string
}

create type tpch.NationType as
 closed {
  n_nationkey : int64,
  n_name : string,
  n_regionkey : int64,
  n_comment : string
}

create type tpch.RegionType as
 closed {
  r_regionkey : int64,
  r_name : string,
  r_comment : string
}

create  dataset Orders(OrderType) primary key o_orderkey;

create  dataset Supplier(SupplierType) primary key s_suppkey;

create  dataset Region(RegionType) primary key r_regionkey;

create  dataset Nation(NationType) primary key n_nationkey;

create  dataset Customer(CustomerType) primary key c_custkey;

create  dataset SelectedNation(NationType) primary key n_nationkey;


create index customer_fk_nation on Customer (c_nationkey);
create index nation_fk_region on Nation(n_regionkey);
create index orders_fk_customer on Orders (o_custkey);
create index orders_orderdateIx on Orders (o_orderdate);
create index supplier_fk_nation on Supplier (s_nationkey);
