sqloracle-databaseviewora-00928

Is the following code valid? I am getting errors.


CREATE OR REPLACE VIEW POINTS AS
DECLARE
  avgDurationOurFault       number(5);
  avgDurationCustomersFault number(5);
  avgDuration           number(5);

BEGIN

    (select ceil(avg(abs(total_time))) into avgDuration from inquiry);

    select ceil(avg(total_duration))  into avgDurationOurFault
    from
    (
        select customer_no, sum(abs(total_time)) total_duration
        from inquiry
        where cat_id in ('C900', 'C901', 'C902', 'C905', 'C907', 'C908', 'C909')
        GROUP BY customer_no);

    select ceil(avg(total_duration))  into avgDurationCustomersFault
    from
    (
        select customer_no, sum(abs(total_time)) total_duration
        from inquiry
        where cat_id in ('C903','C904', 'C906')
        group by customer_no);

    select t1.customer_no, t1.callPoints, t1.durationPoints, t2.catgPoints, 
          t1.callPoints+t1.durationPoints+t2.catgPoints as totalPoints
    from 
    (
        select customer_no, count(inquiry_id)*avgDuration callPoints , sum(abs(total_time)) durationPoints
        from inquiry 
        group by customer_no
        ) t1

        inner join (

        select customer_no, sum(points) catgPoints
        from
        (
        select customer_no,
            case
                when cat_id in ('C903','C904', 'C906')
                then 0

            when cat_id in ('C900', 'C901', 'C902', 'C905', 'C907', 'C908', 'C909')
                then 2*avgDuration + abs(avgDurationCustomersFault - avgDurationOurFault)

            else
                0

            end as points
            from inquiry
            )
            group by customer_no
            ) t2

            on t1.customer_no = t2.customer_no;


END;
/

--------------------ERRORS BELOW---------------------------------------------------

Error starting at line 1 in command: CREATE OR REPLACE VIEW POINTS AS DECLARE avgDurationOurFault number(5) Error at Command Line:1 Column:32 Error report: SQL Error: ORA-00928: missing SELECT keyword 00928. 00000 - "missing SELECT keyword" *Cause:
*Action:

Error starting at line 4 in command: avgDurationCustomersFault number(5) Error report: Unknown Command

Error starting at line 5 in command: avgDuration number(5) Error report: Unknown Command

Error starting at line 7 in command:

BEGIN

(select ceil(avg(abs(total_time))) into avgDuration from inquiry);

select ceil(avg(total_duration))  into avgDurationOurFault
from
(
    select customer_no, sum(abs(total_time)) total_duration
    from inquiry
    where cat_id in ('C900', 'C901', 'C902', 'C905', 'C907', 'C908', 'C909')
    GROUP BY customer_no);

select ceil(avg(total_duration))  into avgDurationCustomersFault
from
(
    select customer_no, sum(abs(total_time)) total_duration
    from inquiry
    where cat_id in ('C903','C904', 'C906')
    group by customer_no);
select t1.customer_no, t1.callPoints, t1.durationPoints, t2.catgPoints, 
      t1.callPoints+t1.durationPoints+t2.catgPoints as totalPoints
from 
(
    select customer_no, count(inquiry_id)*avgDuration callPoints , sum(abs(total_time)) durationPoints
    from inquiry 
    group by customer_no
    ) t1

    inner join (
    select customer_no, sum(points) catgPoints
    from
    (
    select customer_no,
        case
            when cat_id in ('C903','C904', 'C906')
            then 0

        when cat_id in ('C900', 'C901', 'C902', 'C905', 'C907', 'C908', 'C909')
            then 2*avgDuration + abs(avgDurationCustomersFault - avgDurationOurFault)
        else
            0

        end as points
        from inquiry
        )
        group by customer_no
        ) t2

        on t1.customer_no = t2.customer_no;

END;

Error report: ORA-06550: line 3, column 2: PLS-00103: Encountered the symbol "(" when expecting one of the following:

begin case declare exit for goto if loop mod null pragma raise return select update while with << close current delete fetch lock insert open rollback savepoint set sql execute commit forall merge pipe The symbol "update" was substituted for "(" to continue. ORA-06550: line 3, column 37: PLS-00103: Encountered the symbol "INTO" when expecting one of the following:

. ( , * % & - + / at mod rem as from || The symbol ". was inserted before "I ORA-06550: line 3, column 67: PLS-00103: Encountered the symbol ";" when expecting one of the following:

set ORA-06550: line 30, column 3: PLS-00103: Encountered the symbol "INNER" when expecting one of the following:

, ; for group having intersect minus order start union where
connect 06550. 00000 - "line %s, column %s:\n%s" *Cause: Usually a PL/SQL compilation error. *Action:


Solution

  • Use:

    CREATE OR REPLACE VIEW POINTS AS
    SELECT a.customer_no, 
           a.callPoints, 
           a.durationPoints,
           a.catgPoints, 
           a.callPoints + a.durationPoints + a.catgPoints as totalPoints
      FROM (SELECT i.customer_no, 
                   COUNT(i.inquiry_id) * x.avgDuration AS callPoints, 
                   SUM(ABS(i.total_time)) durationPoints,
                   SUM(CASE
                         WHEN i.cat_id IN ('C900', 'C901', 'C902', 'C905', 'C907', 'C908', 'C909') THEN 
                           2 * x.avgDuration + ABS(z.avgDurationCustomersFault - y.avgDurationOurFault)
                         ELSE 0
                       END) AS catgpoints
              FROM INQUIRY i
        CROSS JOIN (SELECT CEIL(AVG(ABS(t.total_time))) AS avgDuration 
                      FROM INQUIRY t) x
        CROSS JOIN (SELECT CEIL(AVG(total_duration)) AS avgDurationOurFault
                      FROM (SELECT SUM(ABS(t.total_time)) AS total_duration
                              FROM INQUIRY t
                             WHERE t.cat_id IN ('C900', 'C901', 'C902', 'C905', 'C907', 'C908', 'C909')
                          GROUP BY t.customer_no) y
        CROSS JOIN (SELECT CEIL(AVG(total_duration)) AS avgDurationCustomersFault
                      FROM (SELECT SUM(ABS(t.total_time)) AS total_duration
                              FROM INQUIRY
                             WHERE t.cat_id IN ('C903','C904', 'C906')
                          GROUP BY t.customer_no) z
          GROUP BY i.customer_no) a
    

    It is possible to combine "y" and "z", by using a CASE statement to sum values according to the cat_id. Someone else can golf with it.

    The problem with your query is that you were trying to use multiple, unrelated SELECT statements. A view is a single SELECT statement - you can use subqueries, derived tables/inline views, etc but they have to be inside the single query like you see in my example. What you posted is more like what you'd find in a stored procedure or function. You can't use variables like how you were attempting, and you didn't need to -- just needed a CROSS JOIN.

    Subquery factoring (AKA WITH clause, CTE) is possible to use, but there's generally little to no performance benefit.