samedi 11 juin 2016

Improve Function performance in Postgresql

Following function is creating a temp table and then populating data in it.

That temp table is supposed to be displayed on click of a button on a website.

Steps I tried to improve performance: 1. Modified postgresql.conf file. 2. Added index in temp table

CREATE OR REPLACE FUNCTION lastonemonth()
  RETURNS void AS
$BODY$
  DECLARE
  query1 text;
  query2 text;
  var_loop1 RECORD;
  dealerName text;
  cur_minus_2_month text; 
  var_sum_of_quantity numeric;
  var_net_value numeric;
  var_average_price_nsr numeric; 
  BEGIN
  EXECUTE 'DROP TABLE IF EXISTS LOM CASCADE';
  EXECUTE 'CREATE TEMP TABLE LOM ( dealer_name text PRIMARY KEY , sum_of_quantity numeric, net_value numeric, average_price_nsr numeric)';
  EXECUTE 'CREATE UNIQUE INDEX dealer_name_idx ON LOM (dealer_name)';

  query1:= 'SELECT DISTINCT dealer FROM customernotorderdb_temp WHERE month IN ( to_char( now() - interval ''2 month'', ''YYYYMM'') , to_char( now() - interval ''3 month'', ''YYYYMM''))EXCEPT SELECT DISTINCT dealer FROM customernotorderdb_temp WHERE month IN ( to_char( now() - interval ''1 month'', ''YYYYMM''), to_char(now(), ''YYYYMM'') )';
  FOR var_loop1 IN EXECUTE (query1)
  LOOP
      dealerName:= var_loop1.dealer;
      INSERT INTO LOM(dealer_name) VALUES(dealerName);
      EXECUTE 'SELECT month FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN(to_char( now() - interval ''2 month'', ''YYYYMM''))' INTO cur_minus_2_month;

      --RAISE NOTICE 'cur_minus_2_month( % )', cur_minus_2_month;
      IF cur_minus_2_month IS NOT NULL 
      THEN
          EXECUTE 'SELECT SUM(saleqtypermt) FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN (to_char( now() - interval ''2 month'', ''YYYYMM''),to_char( now() - interval ''3 month'', ''YYYYMM''),to_char( now() - interval ''4 month'', ''YYYYMM''),to_char( now() - interval ''5 month'', ''YYYYMM''),to_char( now() - interval ''6 month'', ''YYYYMM''),to_char( now() - interval ''7 month'', ''YYYYMM''),to_char( now() -interval ''8 month'', ''YYYYMM''),to_char( now() - interval ''9 month'', ''YYYYMM''),to_char( now() - interval ''10 month'', ''YYYYMM''),to_char( now() - interval ''11 month'', ''YYYYMM''),to_char( now() - interval ''12 month'', ''YYYYMM''),to_char( now() - interval ''13 month'', ''YYYYMM''))' INTO var_sum_of_quantity;
          EXECUTE 'SELECT SUM(basic_value-rate_diff) FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN (to_char( now() - interval ''2 month'', ''YYYYMM''),to_char( now() - interval ''3 month'', ''YYYYMM''),to_char( now() - interval ''4 month'', ''YYYYMM''),to_char( now() - interval ''5 month'', ''YYYYMM''),to_char( now() - interval ''6 month'', ''YYYYMM''),to_char( now() - interval ''7 month'', ''YYYYMM''),to_char( now() -interval ''8 month'', ''YYYYMM''),to_char( now() - interval ''9 month'', ''YYYYMM''),to_char( now() - interval ''10 month'', ''YYYYMM''),to_char( now() - interval ''11 month'', ''YYYYMM''),to_char( now() - interval ''12 month'', ''YYYYMM''),to_char( now() - interval ''13 month'', ''YYYYMM''))' INTO var_net_value;
          EXECUTE 'SELECT SUM(avgpricensr) FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN (to_char( now() - interval ''2 month'', ''YYYYMM''),to_char( now() - interval ''3 month'', ''YYYYMM''),to_char( now() - interval ''4 month'', ''YYYYMM''),to_char( now() - interval ''5 month'', ''YYYYMM''),to_char( now() - interval ''6 month'', ''YYYYMM''),to_char( now() - interval ''7 month'', ''YYYYMM''),to_char( now() -interval ''8 month'', ''YYYYMM''),to_char( now() - interval ''9 month'', ''YYYYMM''),to_char( now() - interval ''10 month'', ''YYYYMM''),to_char( now() - interval ''11 month'', ''YYYYMM''),to_char( now() - interval ''12 month'', ''YYYYMM''),to_char( now() - interval ''13 month'', ''YYYYMM''))' INTO var_average_price_nsr;
          --RAISE NOTICE 'A [ %   % ]', dealername,var_net_value;
          UPDATE LOM SET sum_of_quantity=var_sum_of_quantity,
                         net_value=var_net_value,
                         average_price_nsr=var_average_price_nsr 
          WHERE dealer_name=var_loop1.dealer;

      ELSE
          EXECUTE 'SELECT SUM(saleqtypermt) FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN (to_char( now() - interval ''3 month'', ''YYYYMM''),to_char( now() - interval ''4 month'', ''YYYYMM''),to_char( now() - interval ''5 month'', ''YYYYMM''),to_char( now() - interval ''6 month'', ''YYYYMM''),to_char( now() - interval ''7 month'', ''YYYYMM''),to_char( now() -interval ''8 month'', ''YYYYMM''),to_char( now() - interval ''9 month'', ''YYYYMM''),to_char( now() - interval ''10 month'', ''YYYYMM''),to_char( now() - interval ''11 month'', ''YYYYMM''),to_char( now() - interval ''12 month'', ''YYYYMM''),to_char( now() - interval ''13 month'', ''YYYYMM''),to_char( now() - interval ''14 month'', ''YYYYMM''))' INTO var_sum_of_quantity;
          EXECUTE 'SELECT SUM(basic_value-rate_diff) FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN (to_char( now() - interval ''3 month'', ''YYYYMM''),to_char( now() - interval ''4 month'', ''YYYYMM''),to_char( now() - interval ''5 month'', ''YYYYMM''),to_char( now() - interval ''6 month'', ''YYYYMM''),to_char( now() - interval ''7 month'', ''YYYYMM''),to_char( now() -interval ''8 month'', ''YYYYMM''),to_char( now() - interval ''9 month'', ''YYYYMM''),to_char( now() - interval ''10 month'', ''YYYYMM''),to_char( now() - interval ''11 month'', ''YYYYMM''),to_char( now() - interval ''12 month'', ''YYYYMM''),to_char( now() - interval ''13 month'', ''YYYYMM''),to_char( now() - interval ''14 month'', ''YYYYMM''))' INTO var_net_value;
          EXECUTE 'SELECT SUM(avgpricensr) FROM customernotorderdb_temp WHERE dealer LIKE ''%'||dealerName||'%'' AND month IN (to_char( now() - interval ''3 month'', ''YYYYMM''),to_char( now() - interval ''4 month'', ''YYYYMM''),to_char( now() - interval ''5 month'', ''YYYYMM''),to_char( now() - interval ''6 month'', ''YYYYMM''),to_char( now() - interval ''7 month'', ''YYYYMM''),to_char( now() -interval ''8 month'', ''YYYYMM''),to_char( now() - interval ''9 month'', ''YYYYMM''),to_char( now() - interval ''10 month'', ''YYYYMM''),to_char( now() - interval ''11 month'', ''YYYYMM''),to_char( now() - interval ''12 month'', ''YYYYMM''),to_char( now() - interval ''13 month'', ''YYYYMM''),to_char( now() - interval ''14 month'', ''YYYYMM''))' INTO var_average_price_nsr;
          --RAISE NOTICE 'B [ %  % ]', dealername,var_net_value;
          UPDATE LOM SET sum_of_quantity=var_sum_of_quantity,net_value=var_net_value,average_price_nsr=var_average_price_nsr WHERE dealer_name=var_loop1.dealer;
      END IF;         
 END LOOP;
END;
$BODY$
  LANGUAGE plpgsql VOLATILE 

Aucun commentaire:

Enregistrer un commentaire