/*
 * 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.
 */
drop dataverse index_search if exists;

create dataverse index_search;

use dataverse index_search;

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


create nodegroup group1 if not exists on asterix_nc1, asterix_nc2;

create dataset Orders(OrderType)
  primary key o_orderkey on group1;

create index idx_Orders_Custkey on Orders(o_custkey);

write output to asterix_nc1:"/tmp/index_search.adm";

for $o in dataset('Orders')
where
  $o.o_custkey = 40
return {  
  "o_orderkey": $o.o_orderkey,
  "o_custkey": $o.o_custkey 
}
